UNPKG

30.7 kBJavaScriptView Raw
1'use strict';
2
3Object.defineProperty(exports, '__esModule', { value: true });
4
5const index = require('./index-a0a08b2a.js');
6const ionicGlobal = require('./ionic-global-06f21c1a.js');
7const helpers = require('./helpers-d381ec4d.js');
8const theme = require('./theme-30b7a575.js');
9
10let CACHED_MAP;
11const getIconMap = () => {
12 if (typeof window === 'undefined') {
13 return new Map();
14 }
15 else {
16 if (!CACHED_MAP) {
17 const win = window;
18 win.Ionicons = win.Ionicons || {};
19 CACHED_MAP = win.Ionicons.map = win.Ionicons.map || new Map();
20 }
21 return CACHED_MAP;
22 }
23};
24const getUrl = (i) => {
25 let url = getSrc(i.src);
26 if (url) {
27 return url;
28 }
29 url = getName(i.name, i.icon, i.mode, i.ios, i.md);
30 if (url) {
31 return getNamedUrl(url);
32 }
33 if (i.icon) {
34 url = getSrc(i.icon);
35 if (url) {
36 return url;
37 }
38 url = getSrc(i.icon[i.mode]);
39 if (url) {
40 return url;
41 }
42 }
43 return null;
44};
45const getNamedUrl = (iconName) => {
46 const url = getIconMap().get(iconName);
47 if (url) {
48 return url;
49 }
50 return index.getAssetPath(`svg/${iconName}.svg`);
51};
52const getName = (iconName, icon, mode, ios, md) => {
53 // default to "md" if somehow the mode wasn't set
54 mode = (mode && toLower(mode)) === 'ios' ? 'ios' : 'md';
55 // if an icon was passed in using the ios or md attributes
56 // set the iconName to whatever was passed in
57 if (ios && mode === 'ios') {
58 iconName = toLower(ios);
59 }
60 else if (md && mode === 'md') {
61 iconName = toLower(md);
62 }
63 else {
64 if (!iconName && icon && !isSrc(icon)) {
65 iconName = icon;
66 }
67 if (isStr(iconName)) {
68 iconName = toLower(iconName);
69 }
70 }
71 if (!isStr(iconName) || iconName.trim() === '') {
72 return null;
73 }
74 // only allow alpha characters and dash
75 const invalidChars = iconName.replace(/[a-z]|-|\d/gi, '');
76 if (invalidChars !== '') {
77 return null;
78 }
79 return iconName;
80};
81const getSrc = (src) => {
82 if (isStr(src)) {
83 src = src.trim();
84 if (isSrc(src)) {
85 return src;
86 }
87 }
88 return null;
89};
90const isSrc = (str) => str.length > 0 && /(\/|\.)/.test(str);
91const isStr = (val) => typeof val === 'string';
92const toLower = (val) => val.toLowerCase();
93
94const buttonIosCss = ":host{--overflow:hidden;--ripple-color:currentColor;--border-width:initial;--border-color:initial;--border-style:initial;--color-activated:var(--color);--color-focused:var(--color);--color-hover:var(--color);--box-shadow:none;display:inline-block;width:auto;color:var(--color);font-family:var(--ion-font-family, inherit);text-align:center;text-decoration:none;text-overflow:ellipsis;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;vertical-align:top;vertical-align:-webkit-baseline-middle;-webkit-font-kerning:none;font-kerning:none}:host(.button-disabled){cursor:default;opacity:0.5;pointer-events:none}:host(.button-solid){--background:var(--ion-color-primary, #3880ff);--color:var(--ion-color-primary-contrast, #fff)}:host(.button-outline){--border-color:var(--ion-color-primary, #3880ff);--background:transparent;--color:var(--ion-color-primary, #3880ff)}:host(.button-clear){--border-width:0;--background:transparent;--color:var(--ion-color-primary, #3880ff)}:host(.button-block){display:block}:host(.button-block) .button-native{margin-left:0;margin-right:0;display:block;width:100%;clear:both;contain:content}:host(.button-block) .button-native::after{clear:both}:host(.button-full){display:block}:host(.button-full) .button-native{margin-left:0;margin-right:0;display:block;width:100%;contain:content}:host(.button-full:not(.button-round)) .button-native{border-radius:0;border-right-width:0;border-left-width:0}.button-native{border-radius:var(--border-radius);-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;margin-left:0;margin-right:0;margin-top:0;margin-bottom:0;padding-left:var(--padding-start);padding-right:var(--padding-end);padding-top:var(--padding-top);padding-bottom:var(--padding-bottom);font-family:inherit;font-size:inherit;font-style:inherit;font-weight:inherit;letter-spacing:inherit;text-decoration:inherit;text-indent:inherit;text-overflow:inherit;text-transform:inherit;text-align:inherit;white-space:inherit;color:inherit;display:block;position:relative;width:100%;height:100%;-webkit-transition:var(--transition);transition:var(--transition);border-width:var(--border-width);border-style:var(--border-style);border-color:var(--border-color);outline:none;background:var(--background);line-height:1;-webkit-box-shadow:var(--box-shadow);box-shadow:var(--box-shadow);contain:layout style;cursor:pointer;opacity:var(--opacity);overflow:var(--overflow);z-index:0;-webkit-box-sizing:border-box;box-sizing:border-box;-webkit-appearance:none;-moz-appearance:none;appearance:none}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){.button-native{padding-left:unset;padding-right:unset;-webkit-padding-start:var(--padding-start);padding-inline-start:var(--padding-start);-webkit-padding-end:var(--padding-end);padding-inline-end:var(--padding-end)}}.button-native::-moz-focus-inner{border:0}.button-inner{display:-ms-flexbox;display:flex;position:relative;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-negative:0;flex-shrink:0;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:100%;height:100%;z-index:1}::slotted(ion-icon){font-size:1.4em;pointer-events:none}::slotted(ion-icon[slot=start]){margin-left:-0.3em;margin-right:0.3em;margin-top:0;margin-bottom:0}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){::slotted(ion-icon[slot=start]){margin-left:unset;margin-right:unset;-webkit-margin-start:-0.3em;margin-inline-start:-0.3em;-webkit-margin-end:0.3em;margin-inline-end:0.3em}}::slotted(ion-icon[slot=end]){margin-left:0.3em;margin-right:-0.2em;margin-top:0;margin-bottom:0}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){::slotted(ion-icon[slot=end]){margin-left:unset;margin-right:unset;-webkit-margin-start:0.3em;margin-inline-start:0.3em;-webkit-margin-end:-0.2em;margin-inline-end:-0.2em}}::slotted(ion-icon[slot=icon-only]){font-size:1.8em}ion-ripple-effect{color:var(--ripple-color)}.button-native::after{left:0;right:0;top:0;bottom:0;position:absolute;content:\"\";opacity:0}:host(.ion-activated){color:var(--color-activated)}:host(.ion-activated) .button-native::after{background:var(--background-activated);opacity:var(--background-activated-opacity)}:host(.ion-focused){color:var(--color-focused)}:host(.ion-focused) .button-native::after{background:var(--background-focused);opacity:var(--background-focused-opacity)}@media (any-hover: hover){:host(:hover){color:var(--color-hover)}:host(:hover) .button-native::after{background:var(--background-hover);opacity:var(--background-hover-opacity)}}:host(.button-solid.ion-color) .button-native{background:var(--ion-color-base);color:var(--ion-color-contrast)}:host(.button-outline.ion-color) .button-native{border-color:var(--ion-color-base);background:transparent;color:var(--ion-color-base)}:host(.button-clear.ion-color) .button-native{background:transparent;color:var(--ion-color-base)}:host(.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{color:var(--ion-toolbar-color, var(--color))}:host(.button-outline.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{border-color:var(--ion-toolbar-color, var(--color, var(--border-color)))}:host(.button-solid.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{background:var(--ion-toolbar-color, var(--background));color:var(--ion-toolbar-background, var(--color))}:host{--border-radius:10px;--padding-top:0;--padding-bottom:0;--padding-start:1em;--padding-end:1em;--transition:background-color, opacity 100ms linear;margin-left:2px;margin-right:2px;margin-top:4px;margin-bottom:4px;height:2.8em;font-size:16px;font-weight:500;letter-spacing:-0.03em}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){:host{margin-left:unset;margin-right:unset;-webkit-margin-start:2px;margin-inline-start:2px;-webkit-margin-end:2px;margin-inline-end:2px}}:host(.button-solid){--background-activated:var(--ion-color-primary-shade, #3171e0);--background-focused:var(--ion-color-primary-shade, #3171e0);--background-hover:var(--ion-color-primary-tint, #4c8dff);--background-activated-opacity:1;--background-focused-opacity:1;--background-hover-opacity:1}:host(.button-outline){--border-radius:10px;--border-width:1px;--border-style:solid;--background-activated:var(--ion-color-primary, #3880ff);--background-focused:var(--ion-color-primary, #3880ff);--background-hover:transparent;--background-focused-opacity:.1;--color-activated:var(--ion-color-primary-contrast, #fff)}:host(.button-clear){--background-activated:transparent;--background-focused:var(--ion-color-primary, #3880ff);--background-hover:transparent;--background-focused-opacity:.1;font-size:17px;font-weight:normal;letter-spacing:0}:host(.button-large){--border-radius:12px;--padding-top:0;--padding-start:1em;--padding-end:1em;--padding-bottom:0;height:2.8em;font-size:20px}:host(.button-small){--border-radius:6px;--padding-top:0;--padding-start:0.9em;--padding-end:0.9em;--padding-bottom:0;height:2.1em;font-size:13px}:host(.button-round){--border-radius:64px;--padding-top:0;--padding-start:26px;--padding-end:26px;--padding-bottom:0}:host(.button-strong){font-weight:600}:host(.button-clear.ion-activated){opacity:0.4}:host(.button-outline.ion-activated.ion-color) .button-native{color:var(--ion-color-contrast)}:host(.button-outline.ion-activated.ion-color) .button-native::after{background:var(--ion-color-base)}:host(.button-solid.ion-color.ion-activated) .button-native::after{background:var(--ion-color-shade)}:host(.button-outline.ion-focused.ion-color) .button-native,:host(.button-clear.ion-focused.ion-color) .button-native{color:var(--ion-color-base)}:host(.button-outline.ion-focused.ion-color) .button-native::after,:host(.button-clear.ion-focused.ion-color) .button-native::after{background:var(--ion-color-base)}:host(.button-solid.ion-color.ion-focused) .button-native::after{background:var(--ion-color-shade)}@media (any-hover: hover){:host(.button-clear:hover),:host(.button-outline:hover){opacity:0.6}:host(.button-clear.ion-color:hover) .button-native,:host(.button-outline.ion-color:hover) .button-native{color:var(--ion-color-base)}:host(.button-clear.ion-color:hover) .button-native::after,:host(.button-outline.ion-color:hover) .button-native::after{background:transparent}:host(.button-solid.ion-color:hover) .button-native::after{background:var(--ion-color-tint)}:host(:hover.button-solid.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native::after{background:#fff;opacity:0.1}}";
95
96const buttonMdCss = ":host{--overflow:hidden;--ripple-color:currentColor;--border-width:initial;--border-color:initial;--border-style:initial;--color-activated:var(--color);--color-focused:var(--color);--color-hover:var(--color);--box-shadow:none;display:inline-block;width:auto;color:var(--color);font-family:var(--ion-font-family, inherit);text-align:center;text-decoration:none;text-overflow:ellipsis;white-space:nowrap;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;vertical-align:top;vertical-align:-webkit-baseline-middle;-webkit-font-kerning:none;font-kerning:none}:host(.button-disabled){cursor:default;opacity:0.5;pointer-events:none}:host(.button-solid){--background:var(--ion-color-primary, #3880ff);--color:var(--ion-color-primary-contrast, #fff)}:host(.button-outline){--border-color:var(--ion-color-primary, #3880ff);--background:transparent;--color:var(--ion-color-primary, #3880ff)}:host(.button-clear){--border-width:0;--background:transparent;--color:var(--ion-color-primary, #3880ff)}:host(.button-block){display:block}:host(.button-block) .button-native{margin-left:0;margin-right:0;display:block;width:100%;clear:both;contain:content}:host(.button-block) .button-native::after{clear:both}:host(.button-full){display:block}:host(.button-full) .button-native{margin-left:0;margin-right:0;display:block;width:100%;contain:content}:host(.button-full:not(.button-round)) .button-native{border-radius:0;border-right-width:0;border-left-width:0}.button-native{border-radius:var(--border-radius);-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;margin-left:0;margin-right:0;margin-top:0;margin-bottom:0;padding-left:var(--padding-start);padding-right:var(--padding-end);padding-top:var(--padding-top);padding-bottom:var(--padding-bottom);font-family:inherit;font-size:inherit;font-style:inherit;font-weight:inherit;letter-spacing:inherit;text-decoration:inherit;text-indent:inherit;text-overflow:inherit;text-transform:inherit;text-align:inherit;white-space:inherit;color:inherit;display:block;position:relative;width:100%;height:100%;-webkit-transition:var(--transition);transition:var(--transition);border-width:var(--border-width);border-style:var(--border-style);border-color:var(--border-color);outline:none;background:var(--background);line-height:1;-webkit-box-shadow:var(--box-shadow);box-shadow:var(--box-shadow);contain:layout style;cursor:pointer;opacity:var(--opacity);overflow:var(--overflow);z-index:0;-webkit-box-sizing:border-box;box-sizing:border-box;-webkit-appearance:none;-moz-appearance:none;appearance:none}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){.button-native{padding-left:unset;padding-right:unset;-webkit-padding-start:var(--padding-start);padding-inline-start:var(--padding-start);-webkit-padding-end:var(--padding-end);padding-inline-end:var(--padding-end)}}.button-native::-moz-focus-inner{border:0}.button-inner{display:-ms-flexbox;display:flex;position:relative;-ms-flex-flow:row nowrap;flex-flow:row nowrap;-ms-flex-negative:0;flex-shrink:0;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:100%;height:100%;z-index:1}::slotted(ion-icon){font-size:1.4em;pointer-events:none}::slotted(ion-icon[slot=start]){margin-left:-0.3em;margin-right:0.3em;margin-top:0;margin-bottom:0}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){::slotted(ion-icon[slot=start]){margin-left:unset;margin-right:unset;-webkit-margin-start:-0.3em;margin-inline-start:-0.3em;-webkit-margin-end:0.3em;margin-inline-end:0.3em}}::slotted(ion-icon[slot=end]){margin-left:0.3em;margin-right:-0.2em;margin-top:0;margin-bottom:0}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){::slotted(ion-icon[slot=end]){margin-left:unset;margin-right:unset;-webkit-margin-start:0.3em;margin-inline-start:0.3em;-webkit-margin-end:-0.2em;margin-inline-end:-0.2em}}::slotted(ion-icon[slot=icon-only]){font-size:1.8em}ion-ripple-effect{color:var(--ripple-color)}.button-native::after{left:0;right:0;top:0;bottom:0;position:absolute;content:\"\";opacity:0}:host(.ion-activated){color:var(--color-activated)}:host(.ion-activated) .button-native::after{background:var(--background-activated);opacity:var(--background-activated-opacity)}:host(.ion-focused){color:var(--color-focused)}:host(.ion-focused) .button-native::after{background:var(--background-focused);opacity:var(--background-focused-opacity)}@media (any-hover: hover){:host(:hover){color:var(--color-hover)}:host(:hover) .button-native::after{background:var(--background-hover);opacity:var(--background-hover-opacity)}}:host(.button-solid.ion-color) .button-native{background:var(--ion-color-base);color:var(--ion-color-contrast)}:host(.button-outline.ion-color) .button-native{border-color:var(--ion-color-base);background:transparent;color:var(--ion-color-base)}:host(.button-clear.ion-color) .button-native{background:transparent;color:var(--ion-color-base)}:host(.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{color:var(--ion-toolbar-color, var(--color))}:host(.button-outline.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{border-color:var(--ion-toolbar-color, var(--color, var(--border-color)))}:host(.button-solid.in-toolbar:not(.ion-color):not(.in-toolbar-color)) .button-native{background:var(--ion-toolbar-color, var(--background));color:var(--ion-toolbar-background, var(--color))}:host{--border-radius:4px;--padding-top:0;--padding-bottom:0;--padding-start:1.1em;--padding-end:1.1em;--transition:box-shadow 280ms cubic-bezier(.4, 0, .2, 1),\n background-color 15ms linear,\n color 15ms linear;margin-left:2px;margin-right:2px;margin-top:4px;margin-bottom:4px;height:36px;font-size:14px;font-weight:500;letter-spacing:0.06em;text-transform:uppercase}@supports ((-webkit-margin-start: 0) or (margin-inline-start: 0)) or (-webkit-margin-start: 0){:host{margin-left:unset;margin-right:unset;-webkit-margin-start:2px;margin-inline-start:2px;-webkit-margin-end:2px;margin-inline-end:2px}}:host(.button-solid){--background-activated:transparent;--background-hover:var(--ion-color-primary-contrast, #fff);--background-focused:var(--ion-color-primary-contrast, #fff);--background-activated-opacity:0;--background-focused-opacity:.24;--background-hover-opacity:.08;--box-shadow:0 3px 1px -2px rgba(0, 0, 0, 0.2), 0 2px 2px 0 rgba(0, 0, 0, 0.14), 0 1px 5px 0 rgba(0, 0, 0, 0.12)}:host(.button-solid.ion-activated){--box-shadow:0 5px 5px -3px rgba(0, 0, 0, 0.2), 0 8px 10px 1px rgba(0, 0, 0, 0.14), 0 3px 14px 2px rgba(0, 0, 0, 0.12)}:host(.button-outline){--border-width:2px;--border-style:solid;--box-shadow:none;--background-activated:transparent;--background-focused:var(--ion-color-primary, #3880ff);--background-hover:var(--ion-color-primary, #3880ff);--background-activated-opacity:0;--background-focused-opacity:.12;--background-hover-opacity:.04}:host(.button-outline.ion-activated.ion-color) .button-native{background:transparent}:host(.button-clear){--background-activated:transparent;--background-focused:var(--ion-color-primary, #3880ff);--background-hover:var(--ion-color-primary, #3880ff);--background-activated-opacity:0;--background-focused-opacity:.12;--background-hover-opacity:.04}:host(.button-round){--border-radius:64px;--padding-top:0;--padding-start:26px;--padding-end:26px;--padding-bottom:0}:host(.button-large){--padding-top:0;--padding-start:1em;--padding-end:1em;--padding-bottom:0;height:2.8em;font-size:20px}:host(.button-small){--padding-top:0;--padding-start:0.9em;--padding-end:0.9em;--padding-bottom:0;height:2.1em;font-size:13px}:host(.button-strong){font-weight:bold}::slotted(ion-icon[slot=icon-only]){padding-left:0;padding-right:0;padding-top:0;padding-bottom:0}:host(.button-solid.ion-color.ion-focused) .button-native::after{background:var(--ion-color-contrast)}:host(.button-clear.ion-color.ion-focused) .button-native::after,:host(.button-outline.ion-color.ion-focused) .button-native::after{background:var(--ion-color-base)}@media (any-hover: hover){:host(.button-solid.ion-color:hover) .button-native::after{background:var(--ion-color-contrast)}:host(.button-clear.ion-color:hover) .button-native::after,:host(.button-outline.ion-color:hover) .button-native::after{background:var(--ion-color-base)}}";
97
98const Button = class {
99 constructor(hostRef) {
100 index.registerInstance(this, hostRef);
101 this.ionFocus = index.createEvent(this, "ionFocus", 7);
102 this.ionBlur = index.createEvent(this, "ionBlur", 7);
103 this.inItem = false;
104 this.inListHeader = false;
105 this.inToolbar = false;
106 this.inheritedAttributes = {};
107 /**
108 * The type of button.
109 */
110 this.buttonType = 'button';
111 /**
112 * If `true`, the user cannot interact with the button.
113 */
114 this.disabled = false;
115 /**
116 * When using a router, it specifies the transition direction when navigating to
117 * another page using `href`.
118 */
119 this.routerDirection = 'forward';
120 /**
121 * If `true`, activates a button with a heavier font weight.
122 */
123 this.strong = false;
124 /**
125 * The type of the button.
126 */
127 this.type = 'button';
128 this.handleClick = (ev) => {
129 if (this.type === 'button') {
130 theme.openURL(this.href, ev, this.routerDirection, this.routerAnimation);
131 }
132 else if (helpers.hasShadowDom(this.el)) {
133 // this button wants to specifically submit a form
134 // climb up the dom to see if we're in a <form>
135 // and if so, then use JS to submit it
136 const form = this.el.closest('form');
137 if (form) {
138 ev.preventDefault();
139 const fakeButton = document.createElement('button');
140 fakeButton.type = this.type;
141 fakeButton.style.display = 'none';
142 form.appendChild(fakeButton);
143 fakeButton.click();
144 fakeButton.remove();
145 }
146 }
147 };
148 this.onFocus = () => {
149 this.ionFocus.emit();
150 };
151 this.onBlur = () => {
152 this.ionBlur.emit();
153 };
154 }
155 componentWillLoad() {
156 this.inToolbar = !!this.el.closest('ion-buttons');
157 this.inListHeader = !!this.el.closest('ion-list-header');
158 this.inItem = !!this.el.closest('ion-item') || !!this.el.closest('ion-item-divider');
159 this.inheritedAttributes = helpers.inheritAttributes(this.el, ['aria-label']);
160 }
161 get hasIconOnly() {
162 return !!this.el.querySelector('[slot="icon-only"]');
163 }
164 get rippleType() {
165 const hasClearFill = this.fill === undefined || this.fill === 'clear';
166 // If the button is in a toolbar, has a clear fill (which is the default)
167 // and only has an icon we use the unbounded "circular" ripple effect
168 if (hasClearFill && this.hasIconOnly && this.inToolbar) {
169 return 'unbounded';
170 }
171 return 'bounded';
172 }
173 render() {
174 const mode = ionicGlobal.getIonMode(this);
175 const { buttonType, type, disabled, rel, target, size, href, color, expand, hasIconOnly, shape, strong, inheritedAttributes } = this;
176 const finalSize = size === undefined && this.inItem ? 'small' : size;
177 const TagType = href === undefined ? 'button' : 'a';
178 const attrs = (TagType === 'button')
179 ? { type }
180 : {
181 download: this.download,
182 href,
183 rel,
184 target
185 };
186 let fill = this.fill;
187 if (fill === undefined) {
188 fill = this.inToolbar || this.inListHeader ? 'clear' : 'solid';
189 }
190 return (index.h(index.Host, { onClick: this.handleClick, "aria-disabled": disabled ? 'true' : null, class: theme.createColorClasses(color, {
191 [mode]: true,
192 [buttonType]: true,
193 [`${buttonType}-${expand}`]: expand !== undefined,
194 [`${buttonType}-${finalSize}`]: finalSize !== undefined,
195 [`${buttonType}-${shape}`]: shape !== undefined,
196 [`${buttonType}-${fill}`]: true,
197 [`${buttonType}-strong`]: strong,
198 'in-toolbar': theme.hostContext('ion-toolbar', this.el),
199 'in-toolbar-color': theme.hostContext('ion-toolbar[color]', this.el),
200 'button-has-icon-only': hasIconOnly,
201 'button-disabled': disabled,
202 'ion-activatable': true,
203 'ion-focusable': true,
204 }) }, index.h(TagType, Object.assign({}, attrs, { class: "button-native", part: "native", disabled: disabled, onFocus: this.onFocus, onBlur: this.onBlur }, inheritedAttributes), index.h("span", { class: "button-inner" }, index.h("slot", { name: "icon-only" }), index.h("slot", { name: "start" }), index.h("slot", null), index.h("slot", { name: "end" })), mode === 'md' && index.h("ion-ripple-effect", { type: this.rippleType }))));
205 }
206 get el() { return index.getElement(this); }
207};
208Button.style = {
209 ios: buttonIosCss,
210 md: buttonMdCss
211};
212
213const validateContent = (svgContent) => {
214 const div = document.createElement('div');
215 div.innerHTML = svgContent;
216 // setup this way to ensure it works on our buddy IE
217 for (let i = div.childNodes.length - 1; i >= 0; i--) {
218 if (div.childNodes[i].nodeName.toLowerCase() !== 'svg') {
219 div.removeChild(div.childNodes[i]);
220 }
221 }
222 // must only have 1 root element
223 const svgElm = div.firstElementChild;
224 if (svgElm && svgElm.nodeName.toLowerCase() === 'svg') {
225 const svgClass = svgElm.getAttribute('class') || '';
226 svgElm.setAttribute('class', (svgClass + ' s-ion-icon').trim());
227 // root element must be an svg
228 // lets double check we've got valid elements
229 // do not allow scripts
230 if (isValid(svgElm)) {
231 return div.innerHTML;
232 }
233 }
234 return '';
235};
236const isValid = (elm) => {
237 if (elm.nodeType === 1) {
238 if (elm.nodeName.toLowerCase() === 'script') {
239 return false;
240 }
241 for (let i = 0; i < elm.attributes.length; i++) {
242 const val = elm.attributes[i].value;
243 if (isStr(val) && val.toLowerCase().indexOf('on') === 0) {
244 return false;
245 }
246 }
247 for (let i = 0; i < elm.childNodes.length; i++) {
248 if (!isValid(elm.childNodes[i])) {
249 return false;
250 }
251 }
252 }
253 return true;
254};
255
256const ioniconContent = new Map();
257const requests = new Map();
258const getSvgContent = (url, sanitize) => {
259 // see if we already have a request for this url
260 let req = requests.get(url);
261 if (!req) {
262 if (typeof fetch !== 'undefined' && typeof document !== 'undefined') {
263 // we don't already have a request
264 req = fetch(url).then((rsp) => {
265 if (rsp.ok) {
266 return rsp.text().then((svgContent) => {
267 if (svgContent && sanitize !== false) {
268 svgContent = validateContent(svgContent);
269 }
270 ioniconContent.set(url, svgContent || '');
271 });
272 }
273 ioniconContent.set(url, '');
274 });
275 // cache for the same requests
276 requests.set(url, req);
277 }
278 else {
279 // set to empty for ssr scenarios and resolve promise
280 ioniconContent.set(url, '');
281 return Promise.resolve();
282 }
283 }
284 return req;
285};
286
287const iconCss = ":host{display:inline-block;width:1em;height:1em;contain:strict;fill:currentColor;-webkit-box-sizing:content-box !important;box-sizing:content-box !important}:host .ionicon{stroke:currentColor}.ionicon-fill-none{fill:none}.ionicon-stroke-width{stroke-width:32px;stroke-width:var(--ionicon-stroke-width, 32px)}.icon-inner,.ionicon,svg{display:block;height:100%;width:100%}:host(.flip-rtl) .icon-inner{-webkit-transform:scaleX(-1);transform:scaleX(-1)}:host(.icon-small){font-size:18px !important}:host(.icon-large){font-size:32px !important}:host(.ion-color){color:var(--ion-color-base) !important}:host(.ion-color-primary){--ion-color-base:var(--ion-color-primary, #3880ff)}:host(.ion-color-secondary){--ion-color-base:var(--ion-color-secondary, #0cd1e8)}:host(.ion-color-tertiary){--ion-color-base:var(--ion-color-tertiary, #f4a942)}:host(.ion-color-success){--ion-color-base:var(--ion-color-success, #10dc60)}:host(.ion-color-warning){--ion-color-base:var(--ion-color-warning, #ffce00)}:host(.ion-color-danger){--ion-color-base:var(--ion-color-danger, #f14141)}:host(.ion-color-light){--ion-color-base:var(--ion-color-light, #f4f5f8)}:host(.ion-color-medium){--ion-color-base:var(--ion-color-medium, #989aa2)}:host(.ion-color-dark){--ion-color-base:var(--ion-color-dark, #222428)}";
288
289const Icon = class {
290 constructor(hostRef) {
291 index.registerInstance(this, hostRef);
292 this.iconName = null;
293 this.isVisible = false;
294 /**
295 * The mode determines which platform styles to use.
296 */
297 this.mode = getIonMode();
298 /**
299 * If enabled, ion-icon will be loaded lazily when it's visible in the viewport.
300 * Default, `false`.
301 */
302 this.lazy = false;
303 /**
304 * When set to `false`, SVG content that is HTTP fetched will not be checked
305 * if the response SVG content has any `<script>` elements, or any attributes
306 * that start with `on`, such as `onclick`.
307 * @default true
308 */
309 this.sanitize = true;
310 }
311 connectedCallback() {
312 // purposely do not return the promise here because loading
313 // the svg file should not hold up loading the app
314 // only load the svg if it's visible
315 this.waitUntilVisible(this.el, '50px', () => {
316 this.isVisible = true;
317 this.loadIcon();
318 });
319 }
320 disconnectedCallback() {
321 if (this.io) {
322 this.io.disconnect();
323 this.io = undefined;
324 }
325 }
326 waitUntilVisible(el, rootMargin, cb) {
327 if (this.lazy && typeof window !== 'undefined' && window.IntersectionObserver) {
328 const io = (this.io = new window.IntersectionObserver((data) => {
329 if (data[0].isIntersecting) {
330 io.disconnect();
331 this.io = undefined;
332 cb();
333 }
334 }, { rootMargin }));
335 io.observe(el);
336 }
337 else {
338 // browser doesn't support IntersectionObserver
339 // so just fallback to always show it
340 cb();
341 }
342 }
343 loadIcon() {
344 if (this.isVisible) {
345 const url = getUrl(this);
346 if (url) {
347 if (ioniconContent.has(url)) {
348 // sync if it's already loaded
349 this.svgContent = ioniconContent.get(url);
350 }
351 else {
352 // async if it hasn't been loaded
353 getSvgContent(url, this.sanitize).then(() => (this.svgContent = ioniconContent.get(url)));
354 }
355 }
356 }
357 const label = this.iconName = getName(this.name, this.icon, this.mode, this.ios, this.md);
358 if (!this.ariaLabel && this.ariaHidden !== 'true') {
359 // user did not provide a label
360 // come up with the label based on the icon name
361 if (label) {
362 this.ariaLabel = label.replace(/\-/g, ' ');
363 }
364 }
365 }
366 render() {
367 const { iconName } = this;
368 const mode = this.mode || 'md';
369 const flipRtl = this.flipRtl ||
370 (iconName &&
371 (iconName.indexOf('arrow') > -1 || iconName.indexOf('chevron') > -1) &&
372 this.flipRtl !== false);
373 return (index.h(index.Host, { role: "img", class: Object.assign(Object.assign({ [mode]: true }, createColorClasses(this.color)), { [`icon-${this.size}`]: !!this.size, 'flip-rtl': !!flipRtl && this.el.ownerDocument.dir === 'rtl' }) }, this.svgContent ? (index.h("div", { class: "icon-inner", innerHTML: this.svgContent })) : (index.h("div", { class: "icon-inner" }))));
374 }
375 static get assetsDirs() { return ["svg"]; }
376 get el() { return index.getElement(this); }
377 static get watchers() { return {
378 "name": ["loadIcon"],
379 "src": ["loadIcon"],
380 "icon": ["loadIcon"]
381 }; }
382};
383const getIonMode = () => (typeof document !== 'undefined' && document.documentElement.getAttribute('mode')) || 'md';
384const createColorClasses = (color) => {
385 return color
386 ? {
387 'ion-color': true,
388 [`ion-color-${color}`]: true,
389 }
390 : null;
391};
392Icon.style = iconCss;
393
394exports.ion_button = Button;
395exports.ion_icon = Icon;