UNPKG

@postnord/web-components

Version:
438 lines (437 loc) 17.3 kB
/*! * Built with Stencil * By PostNord. */ import { Host, h } from "@stencil/core"; import { arrow_right, open_in_new } from "pn-design-assets/pn-assets/icons.js"; import { ripple, uuidv4 } from "../../../index"; /** * The card component is more than just a navigational element. * It is a flexible component that can be used as a product item, a clickable link card, etc... * * Setting the `href` prop will transform the entire `pn-card` into a clickable element. * If you need content to be interactable inside the card, do not use the `href` prop. * * @slot - The default slot is the primary content of the card. This is the same area where the `text` prop is located. * @slot image - Set a custom image/svg/illustration. Simply use an `img` tag with the slot "image". * @slot tags - Add `pn-tags` that will be positioned above the `label` and `overline` if used. * @slot header - The header content, same area that the `label` and `overline` props will appear. * @slot navigation - This slot is intended for one or more `pn-button` components. * Only use **without** the `href` prop. Having both at the same time makes the component inaccessible. * * @since v7.12.0 */ export class PnCard { id = `pn-card-${uuidv4()}`; idTitle = `${this.id}-title`; visibleOutline = false; hostElement; /** Headline of the card. Will act as label for the link if there is no `href-label` set. */ label; /** The label HTML tag. Default is `h3`. */ labelTag = 'h3'; /** Set a smaller text above the `label` text. */ overline; /** Text content of the card. */ text; /** Set a HTML id for the card. Optional and not required. */ cardId = null; /** * Uee the horizontal card layout. * @category Features * */ horizontal = false; /** Reverse the order of which the image will wrap. @category Features */ reverse = false; /** * Select image aspect ratio. Example: '3/3', '16/9'. * @see {@link https://developer.mozilla.org/en-US/docs/Web/CSS/aspect-ratio CSS documentation.} * @category Features * */ aspectRatio = ''; /** The card link. @category Href */ href; /** The link text. @category Href */ hrefLabel; /** The target attribute of the link. @category Href */ target; /** The rel attribute of the link. @category Href */ rel = 'noopener noreferrer'; /** The link icon. Defaults to `arrow_right` or `open_in_new` if `target="_blank"`. @category Href */ icon = null; watchCardId() { const id = this.cardId || this.id; this.idTitle = `${id}-title`; } /** Emitted when you click on the card link (if you use the `href` prop). */ pnCard; componentWillLoad() { this.watchCardId(); } isExternal() { return this.target === '_blank'; } showTitle() { return !!this.label; } showHref() { return !!this.href && (this.hrefLabel || this.showTitle()); } getTitleId() { return this.hrefLabel ? null : this.idTitle; } getLabelledby() { return this.hrefLabel ? null : this.showTitle() ? this.idTitle : null; } getIcon() { if (!!this.icon) return this.icon; return this.isExternal() ? open_in_new : arrow_right; } getHeadingTag() { const tags = ['h2', 'h3', 'h4', 'h5', 'h6', 'p']; return tags.includes(this.labelTag) ? this.labelTag : 'h3'; } handleFocus({ type, target }) { this.visibleOutline = type === 'focus' && target.matches(':focus-visible'); } handleClick(event) { ripple(event, this.hostElement, '.pn-card'); this.pnCard.emit({ click: event }); } render() { const TitleTag = this.getHeadingTag(); return (h(Host, { key: '8126385831d763637aed6ebe4fa61fd0bdb42764' }, h("article", { key: '93c75dc2d9e71844e36a9cdd6914f42eae34f7b0', id: this.cardId, class: "pn-card", "data-link": !!this.href, "data-horizontal": this.horizontal, "data-reverse": this.reverse, "data-focus": this.visibleOutline }, h("div", { key: '140936bb72867a5ef4351363dfda52698b9d9df0', class: "pn-card-image" }, h("picture", { key: '60de2f3f53b4d7f1e1cbe37b1ff47686c9e48b72', class: "pn-card-picture", style: { '--pn-card-aspect-ratio': this.aspectRatio || 'unset' } }, h("slot", { key: '47cffafc0d642f6cf0c682cd96f9fd11609f924e', name: "image" }))), h("div", { key: '6768342177fe3884d9bd20b0ca516cf026b69751', class: "pn-card-content" }, h("header", { key: '91e023c92c33bf74c39200387edc7a3d3c2f2d42', class: "pn-card-header" }, h("div", { key: '9d7771c4275569012021a07f9165ba4a8ce8c297', class: "pn-card-tags" }, h("slot", { key: 'c45641bd02093f8001dee361b6b8577a383f72ba', name: "tags" })), !!this.overline && h("p", { key: 'b152b16d2488e82556e9f56f90daf0edfe8ae33f', class: "pn-card-label-overline" }, this.overline), !!this.label && (h(TitleTag, { key: '3bc086490090afc05ad44c0e1736009b82fbe814', id: this.getTitleId(), class: "pn-card-label" }, this.label)), h("slot", { key: '1188ca2c3260dd509811cfb5aa5356488f211e1a', name: "header" })), h("section", { key: 'ba0b7362ddd721041466ac979ba198170b016fb1', class: "pn-card-section" }, !!this.text && h("p", { key: '76aa9e90ac18ff66ec15a436ec8eec9e22c67748', class: "pn-card-section-paragraph" }, this.text), h("slot", { key: 'bd0f035ce418425c1ac1871ec1cb98f850cd97e1' })), h("nav", { key: '1a538e55ed6c9c07f3f8c5dd823f80ed3a149ed6', class: "pn-card-navigation" }, this.showHref() && (h("a", { key: 'a18605058f0dc733ceeeab9fa6a44fb524c93630', class: "pn-card-link", href: this.href, target: this.target, rel: this.rel, "aria-labelledby": this.getLabelledby(), onClick: event => this.handleClick(event), onFocus: event => this.handleFocus(event), onBlur: event => this.handleFocus(event) }, this.hrefLabel, h("pn-icon", { key: '066242008cd49a69d64db93da7a87bbdcdd1beda', color: "blue700", icon: this.getIcon() }))), h("slot", { key: 'bbe0271663622b8adf542232daa573a467b34f16', name: "navigation" })))))); } static get is() { return "pn-card"; } static get originalStyleUrls() { return { "$": ["pn-card.scss"] }; } static get styleUrls() { return { "$": ["pn-card.css"] }; } static get properties() { return { "label": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": false, "docs": { "tags": [], "text": "Headline of the card. Will act as label for the link if there is no `href-label` set." }, "getter": false, "setter": false, "reflect": false, "attribute": "label" }, "labelTag": { "type": "string", "mutable": false, "complexType": { "original": "'h2' | 'h3' | 'h4' | 'h5' | 'h6' | 'p'", "resolved": "\"h2\" | \"h3\" | \"h4\" | \"h5\" | \"h6\" | \"p\"", "references": {} }, "required": false, "optional": true, "docs": { "tags": [], "text": "The label HTML tag. Default is `h3`." }, "getter": false, "setter": false, "reflect": false, "attribute": "label-tag", "defaultValue": "'h3'" }, "overline": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [], "text": "Set a smaller text above the `label` text." }, "getter": false, "setter": false, "reflect": false, "attribute": "overline" }, "text": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [], "text": "Text content of the card." }, "getter": false, "setter": false, "reflect": false, "attribute": "text" }, "cardId": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [], "text": "Set a HTML id for the card. Optional and not required." }, "getter": false, "setter": false, "reflect": false, "attribute": "card-id", "defaultValue": "null" }, "horizontal": { "type": "boolean", "mutable": false, "complexType": { "original": "boolean", "resolved": "boolean", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Features" }], "text": "Uee the horizontal card layout." }, "getter": false, "setter": false, "reflect": false, "attribute": "horizontal", "defaultValue": "false" }, "reverse": { "type": "boolean", "mutable": false, "complexType": { "original": "boolean", "resolved": "boolean", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Features" }], "text": "Reverse the order of which the image will wrap." }, "getter": false, "setter": false, "reflect": false, "attribute": "reverse", "defaultValue": "false" }, "aspectRatio": { "type": "string", "mutable": false, "complexType": { "original": "string | '3/3' | '16/9'", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "see", "text": "{@link https://developer.mozilla.org/en-US/docs/Web/CSS/aspect-ratio CSS documentation.}" }, { "name": "category", "text": "Features" }], "text": "Select image aspect ratio. Example: '3/3', '16/9'." }, "getter": false, "setter": false, "reflect": false, "attribute": "aspect-ratio", "defaultValue": "''" }, "href": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Href" }], "text": "The card link." }, "getter": false, "setter": false, "reflect": false, "attribute": "href" }, "hrefLabel": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Href" }], "text": "The link text." }, "getter": false, "setter": false, "reflect": false, "attribute": "href-label" }, "target": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Href" }], "text": "The target attribute of the link." }, "getter": false, "setter": false, "reflect": false, "attribute": "target" }, "rel": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Href" }], "text": "The rel attribute of the link." }, "getter": false, "setter": false, "reflect": false, "attribute": "rel", "defaultValue": "'noopener noreferrer'" }, "icon": { "type": "string", "mutable": false, "complexType": { "original": "string", "resolved": "string", "references": {} }, "required": false, "optional": true, "docs": { "tags": [{ "name": "category", "text": "Href" }], "text": "The link icon. Defaults to `arrow_right` or `open_in_new` if `target=\"_blank\"`." }, "getter": false, "setter": false, "reflect": false, "attribute": "icon", "defaultValue": "null" } }; } static get states() { return { "visibleOutline": {} }; } static get events() { return [{ "method": "pnCard", "name": "pnCard", "bubbles": true, "cancelable": true, "composed": true, "docs": { "tags": [], "text": "Emitted when you click on the card link (if you use the `href` prop)." }, "complexType": { "original": "{ click: MouseEvent }", "resolved": "{ click: MouseEvent; }", "references": { "MouseEvent": { "location": "global", "id": "global::MouseEvent" } } } }]; } static get elementRef() { return "hostElement"; } static get watchers() { return [{ "propName": "cardId", "methodName": "watchCardId" }]; } }