You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

40946 lines
1.1 MiB

/*!
* Quasar Framework v2.12.2
* (c) 2015-present Razvan Stoenescu
* Released under the MIT License.
*/
(function (global, factory) {
typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory(require('vue')) :
typeof define === 'function' && define.amd ? define(['vue'], factory) :
(global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Quasar = factory(global.Vue));
})(this, (function (vue) { 'use strict';
function injectProp (target, propName, get, set) {
Object.defineProperty(target, propName, {
get,
set,
enumerable: true
});
return target
}
function injectMultipleProps (target, props) {
for (const key in props) {
injectProp(target, key, props[ key ]);
}
return target
}
/* eslint-disable no-useless-escape */
/**
* __ QUASAR_SSR __ -> runs on SSR on client or server
* __ QUASAR_SSR_SERVER __ -> runs on SSR on server
* __ QUASAR_SSR_CLIENT __ -> runs on SSR on client
* __ QUASAR_SSR_PWA __ -> built with SSR+PWA; may run on SSR on client or on PWA client
* (needs runtime detection)
*/
const isRuntimeSsrPreHydration = vue.ref(
false
);
let iosCorrection;
function getMatch (userAgent, platformMatch) {
const match = /(edg|edge|edga|edgios)\/([\w.]+)/.exec(userAgent)
|| /(opr)[\/]([\w.]+)/.exec(userAgent)
|| /(vivaldi)[\/]([\w.]+)/.exec(userAgent)
|| /(chrome|crios)[\/]([\w.]+)/.exec(userAgent)
|| /(version)(applewebkit)[\/]([\w.]+).*(safari)[\/]([\w.]+)/.exec(userAgent)
|| /(webkit)[\/]([\w.]+).*(version)[\/]([\w.]+).*(safari)[\/]([\w.]+)/.exec(userAgent)
|| /(firefox|fxios)[\/]([\w.]+)/.exec(userAgent)
|| /(webkit)[\/]([\w.]+)/.exec(userAgent)
|| /(opera)(?:.*version|)[\/]([\w.]+)/.exec(userAgent)
|| [];
return {
browser: match[ 5 ] || match[ 3 ] || match[ 1 ] || '',
version: match[ 2 ] || match[ 4 ] || '0',
versionNumber: match[ 4 ] || match[ 2 ] || '0',
platform: platformMatch[ 0 ] || ''
}
}
function getPlatformMatch (userAgent) {
return /(ipad)/.exec(userAgent)
|| /(ipod)/.exec(userAgent)
|| /(windows phone)/.exec(userAgent)
|| /(iphone)/.exec(userAgent)
|| /(kindle)/.exec(userAgent)
|| /(silk)/.exec(userAgent)
|| /(android)/.exec(userAgent)
|| /(win)/.exec(userAgent)
|| /(mac)/.exec(userAgent)
|| /(linux)/.exec(userAgent)
|| /(cros)/.exec(userAgent)
// TODO: Remove BlackBerry detection. BlackBerry OS, BlackBerry 10, and BlackBerry PlayBook OS
// is officially dead as of January 4, 2022 (https://www.blackberry.com/us/en/support/devices/end-of-life)
|| /(playbook)/.exec(userAgent)
|| /(bb)/.exec(userAgent)
|| /(blackberry)/.exec(userAgent)
|| []
}
const hasTouch = 'ontouchstart' in window || window.navigator.maxTouchPoints > 0;
function applyIosCorrection (is) {
iosCorrection = { is: { ...is } };
delete is.mac;
delete is.desktop;
const platform = Math.min(window.innerHeight, window.innerWidth) > 414
? 'ipad'
: 'iphone';
Object.assign(is, {
mobile: true,
ios: true,
platform,
[ platform ]: true
});
}
function getPlatform (UA) {
const
userAgent = UA.toLowerCase(),
platformMatch = getPlatformMatch(userAgent),
matched = getMatch(userAgent, platformMatch),
browser = {};
if (matched.browser) {
browser[ matched.browser ] = true;
browser.version = matched.version;
browser.versionNumber = parseInt(matched.versionNumber, 10);
}
if (matched.platform) {
browser[ matched.platform ] = true;
}
const knownMobiles = browser.android
|| browser.ios
|| browser.bb
|| browser.blackberry
|| browser.ipad
|| browser.iphone
|| browser.ipod
|| browser.kindle
|| browser.playbook
|| browser.silk
|| browser[ 'windows phone' ];
// These are all considered mobile platforms, meaning they run a mobile browser
if (knownMobiles === true || userAgent.indexOf('mobile') > -1) {
browser.mobile = true;
if (browser.edga || browser.edgios) {
browser.edge = true;
matched.browser = 'edge';
}
else if (browser.crios) {
browser.chrome = true;
matched.browser = 'chrome';
}
else if (browser.fxios) {
browser.firefox = true;
matched.browser = 'firefox';
}
}
// If it's not mobile we should consider it's desktop platform, meaning it runs a desktop browser
// It's a workaround for anonymized user agents
// (browser.cros || browser.mac || browser.linux || browser.win)
else {
browser.desktop = true;
}
// Set iOS if on iPod, iPad or iPhone
if (browser.ipod || browser.ipad || browser.iphone) {
browser.ios = true;
}
if (browser[ 'windows phone' ]) {
browser.winphone = true;
delete browser[ 'windows phone' ];
}
// TODO: The assumption about WebKit based browsers below is not completely accurate.
// Google released Blink(a fork of WebKit) engine on April 3, 2013, which is really different than WebKit today.
// Today, one might want to check for WebKit to deal with its bugs, which is used on all browsers on iOS, and Safari browser on all platforms.
// Chrome, Opera 15+, Vivaldi and Safari are webkit based browsers
if (
browser.chrome
|| browser.opr
|| browser.safari
|| browser.vivaldi
// we expect unknown, non iOS mobile browsers to be webkit based
|| (
browser.mobile === true
&& browser.ios !== true
&& knownMobiles !== true
)
) {
browser.webkit = true;
}
// TODO: (Qv3) rename the terms 'edge' to 'edge legacy'(or remove it) then 'edge chromium' to 'edge' to match with the known up-to-date terms
// Microsoft Edge is the new Chromium-based browser. Microsoft Edge Legacy is the old EdgeHTML-based browser (EOL: March 9, 2021).
if (browser.edg) {
matched.browser = 'edgechromium';
browser.edgeChromium = true;
}
// Blackberry browsers are marked as Safari on BlackBerry
if ((browser.safari && browser.blackberry) || browser.bb) {
matched.browser = 'blackberry';
browser.blackberry = true;
}
// Playbook browsers are marked as Safari on Playbook
if (browser.safari && browser.playbook) {
matched.browser = 'playbook';
browser.playbook = true;
}
// Opera 15+ are identified as opr
if (browser.opr) {
matched.browser = 'opera';
browser.opera = true;
}
// Stock Android browsers are marked as Safari on Android.
if (browser.safari && browser.android) {
matched.browser = 'android';
browser.android = true;
}
// Kindle browsers are marked as Safari on Kindle
if (browser.safari && browser.kindle) {
matched.browser = 'kindle';
browser.kindle = true;
}
// Kindle Silk browsers are marked as Safari on Kindle
if (browser.safari && browser.silk) {
matched.browser = 'silk';
browser.silk = true;
}
if (browser.vivaldi) {
matched.browser = 'vivaldi';
browser.vivaldi = true;
}
// Assign the name and platform variable
browser.name = matched.browser;
browser.platform = matched.platform;
{
if (userAgent.indexOf('electron') > -1) {
browser.electron = true;
}
else if (document.location.href.indexOf('-extension://') > -1) {
browser.bex = true;
}
else {
if (window.Capacitor !== void 0) {
browser.capacitor = true;
browser.nativeMobile = true;
browser.nativeMobileWrapper = 'capacitor';
}
else if (window._cordovaNative !== void 0 || window.cordova !== void 0) {
browser.cordova = true;
browser.nativeMobile = true;
browser.nativeMobileWrapper = 'cordova';
}
if (
hasTouch === true
&& browser.mac === true
&& (
(browser.desktop === true && browser.safari === true)
|| (
browser.nativeMobile === true
&& browser.android !== true
&& browser.ios !== true
&& browser.ipad !== true
)
)
) {
/*
* Correction needed for iOS since the default
* setting on iPad is to request desktop view; if we have
* touch support and the user agent says it's a
* desktop, we infer that it's an iPhone/iPad with desktop view
* so we must fix the false positives
*/
applyIosCorrection(browser);
}
}
}
return browser
}
const userAgent = navigator.userAgent || navigator.vendor || window.opera;
const ssrClient = {
has: {
touch: false,
webStorage: false
},
within: { iframe: false }
};
// We export "client" for hydration error-free parts,
// like touch directives who do not (and must NOT) wait
// for the client takeover;
// Do NOT import this directly in your app, unless you really know
// what you are doing.
const client = {
userAgent,
is: getPlatform(userAgent),
has: {
touch: hasTouch
},
within: {
iframe: window.self !== window.top
}
};
const Platform = {
install (opts) {
const { $q } = opts;
if (isRuntimeSsrPreHydration.value === true) {
// takeover should increase accuracy for
// the rest of the props; we also avoid
// hydration errors
opts.onSSRHydrated.push(() => {
Object.assign($q.platform, client);
isRuntimeSsrPreHydration.value = false;
iosCorrection = void 0;
});
// we need to make platform reactive
// for the takeover phase
$q.platform = vue.reactive(this);
}
else {
$q.platform = this;
}
}
};
{
// do not access window.localStorage without
// devland actually using it as this will get
// reported under "Cookies" in Google Chrome
let hasWebStorage;
injectProp(client.has, 'webStorage', () => {
if (hasWebStorage !== void 0) {
return hasWebStorage
}
try {
if (window.localStorage) {
hasWebStorage = true;
return true
}
}
catch (e) {}
hasWebStorage = false;
return false
});
client.is.ios === true
&& window.navigator.vendor.toLowerCase().indexOf('apple') === -1;
if (isRuntimeSsrPreHydration.value === true) {
// must match with server-side before
// client taking over in order to prevent
// hydration errors
Object.assign(Platform, client, iosCorrection, ssrClient);
}
else {
Object.assign(Platform, client);
}
}
var defineReactivePlugin = (state, plugin) => {
const reactiveState = vue.reactive(state);
for (const name in state) {
injectProp(
plugin,
name,
() => reactiveState[ name ],
val => { reactiveState[ name ] = val; }
);
}
return plugin
};
const listenOpts = {
hasPassive: false,
passiveCapture: true,
notPassiveCapture: true
};
try {
const opts = Object.defineProperty({}, 'passive', {
get () {
Object.assign(listenOpts, {
hasPassive: true,
passive: { passive: true },
notPassive: { passive: false },
passiveCapture: { passive: true, capture: true },
notPassiveCapture: { passive: false, capture: true }
});
}
});
window.addEventListener('qtest', null, opts);
window.removeEventListener('qtest', null, opts);
}
catch (e) {}
function noop () {}
function leftClick (e) {
return e.button === 0
}
function middleClick (e) {
return e.button === 1
}
function rightClick (e) {
return e.button === 2
}
function position (e) {
if (e.touches && e.touches[ 0 ]) {
e = e.touches[ 0 ];
}
else if (e.changedTouches && e.changedTouches[ 0 ]) {
e = e.changedTouches[ 0 ];
}
else if (e.targetTouches && e.targetTouches[ 0 ]) {
e = e.targetTouches[ 0 ];
}
return {
top: e.clientY,
left: e.clientX
}
}
function getEventPath (e) {
if (e.path) {
return e.path
}
if (e.composedPath) {
return e.composedPath()
}
const path = [];
let el = e.target;
while (el) {
path.push(el);
if (el.tagName === 'HTML') {
path.push(document);
path.push(window);
return path
}
el = el.parentElement;
}
}
// Reasonable defaults
const
LINE_HEIGHT = 40,
PAGE_HEIGHT = 800;
function getMouseWheelDistance (e) {
let x = e.deltaX, y = e.deltaY;
if ((x || y) && e.deltaMode) {
const multiplier = e.deltaMode === 1 ? LINE_HEIGHT : PAGE_HEIGHT;
x *= multiplier;
y *= multiplier;
}
if (e.shiftKey && !x) {
[ y, x ] = [ x, y ];
}
return { x, y }
}
function stop (e) {
e.stopPropagation();
}
function prevent (e) {
e.cancelable !== false && e.preventDefault();
}
function stopAndPrevent (e) {
e.cancelable !== false && e.preventDefault();
e.stopPropagation();
}
function preventDraggable (el, status) {
if (el === void 0 || (status === true && el.__dragPrevented === true)) {
return
}
const fn = status === true
? el => {
el.__dragPrevented = true;
el.addEventListener('dragstart', prevent, listenOpts.notPassiveCapture);
}
: el => {
delete el.__dragPrevented;
el.removeEventListener('dragstart', prevent, listenOpts.notPassiveCapture);
};
el.querySelectorAll('a, img').forEach(fn);
}
function addEvt (ctx, targetName, events) {
const name = `__q_${ targetName }_evt`;
ctx[ name ] = ctx[ name ] !== void 0
? ctx[ name ].concat(events)
: events;
events.forEach(evt => {
evt[ 0 ].addEventListener(evt[ 1 ], ctx[ evt[ 2 ] ], listenOpts[ evt[ 3 ] ]);
});
}
function cleanEvt (ctx, targetName) {
const name = `__q_${ targetName }_evt`;
if (ctx[ name ] !== void 0) {
ctx[ name ].forEach(evt => {
evt[ 0 ].removeEventListener(evt[ 1 ], ctx[ evt[ 2 ] ], listenOpts[ evt[ 3 ] ]);
});
ctx[ name ] = void 0;
}
}
/*
* also update /types/utils/event.d.ts
*/
var event = {
listenOpts,
leftClick,
middleClick,
rightClick,
position,
getEventPath,
getMouseWheelDistance,
stop,
prevent,
stopAndPrevent,
preventDraggable
};
function debounce (fn, wait = 250, immediate) {
let timer = null;
function debounced (/* ...args */) {
const args = arguments;
const later = () => {
timer = null;
if (immediate !== true) {
fn.apply(this, args);
}
};
if (timer !== null) {
clearTimeout(timer);
}
else if (immediate === true) {
fn.apply(this, args);
}
timer = setTimeout(later, wait);
}
debounced.cancel = () => {
timer !== null && clearTimeout(timer);
};
return debounced
}
const SIZE_LIST = [ 'sm', 'md', 'lg', 'xl' ];
const { passive: passive$4 } = listenOpts;
var Screen = defineReactivePlugin({
width: 0,
height: 0,
name: 'xs',
sizes: {
sm: 600,
md: 1024,
lg: 1440,
xl: 1920
},
lt: {
sm: true,
md: true,
lg: true,
xl: true
},
gt: {
xs: false,
sm: false,
md: false,
lg: false
},
xs: true,
sm: false,
md: false,
lg: false,
xl: false
}, {
setSizes: noop,
setDebounce: noop,
install ({ $q, onSSRHydrated }) {
$q.screen = this;
if (this.__installed === true) {
if ($q.config.screen !== void 0) {
if ($q.config.screen.bodyClasses === false) {
document.body.classList.remove(`screen--${ this.name }`);
}
else {
this.__update(true);
}
}
return
}
const { visualViewport } = window;
const target = visualViewport || window;
const scrollingElement = document.scrollingElement || document.documentElement;
const getSize = visualViewport === void 0 || client.is.mobile === true
? () => [
Math.max(window.innerWidth, scrollingElement.clientWidth),
Math.max(window.innerHeight, scrollingElement.clientHeight)
]
: () => [
visualViewport.width * visualViewport.scale + window.innerWidth - scrollingElement.clientWidth,
visualViewport.height * visualViewport.scale + window.innerHeight - scrollingElement.clientHeight
];
const classes = $q.config.screen !== void 0 && $q.config.screen.bodyClasses === true;
this.__update = force => {
const [ w, h ] = getSize();
if (h !== this.height) {
this.height = h;
}
if (w !== this.width) {
this.width = w;
}
else if (force !== true) {
return
}
let s = this.sizes;
this.gt.xs = w >= s.sm;
this.gt.sm = w >= s.md;
this.gt.md = w >= s.lg;
this.gt.lg = w >= s.xl;
this.lt.sm = w < s.sm;
this.lt.md = w < s.md;
this.lt.lg = w < s.lg;
this.lt.xl = w < s.xl;
this.xs = this.lt.sm;
this.sm = this.gt.xs === true && this.lt.md === true;
this.md = this.gt.sm === true && this.lt.lg === true;
this.lg = this.gt.md === true && this.lt.xl === true;
this.xl = this.gt.lg;
s = (this.xs === true && 'xs')
|| (this.sm === true && 'sm')
|| (this.md === true && 'md')
|| (this.lg === true && 'lg')
|| 'xl';
if (s !== this.name) {
if (classes === true) {
document.body.classList.remove(`screen--${ this.name }`);
document.body.classList.add(`screen--${ s }`);
}
this.name = s;
}
};
let updateEvt, updateSizes = {}, updateDebounce = 16;
this.setSizes = sizes => {
SIZE_LIST.forEach(name => {
if (sizes[ name ] !== void 0) {
updateSizes[ name ] = sizes[ name ];
}
});
};
this.setDebounce = deb => {
updateDebounce = deb;
};
const start = () => {
const style = getComputedStyle(document.body);
// if css props available
if (style.getPropertyValue('--q-size-sm')) {
SIZE_LIST.forEach(name => {
this.sizes[ name ] = parseInt(style.getPropertyValue(`--q-size-${ name }`), 10);
});
}
this.setSizes = sizes => {
SIZE_LIST.forEach(name => {
if (sizes[ name ]) {
this.sizes[ name ] = sizes[ name ];
}
});
this.__update(true);
};
this.setDebounce = delay => {
updateEvt !== void 0 && target.removeEventListener('resize', updateEvt, passive$4);
updateEvt = delay > 0
? debounce(this.__update, delay)
: this.__update;
target.addEventListener('resize', updateEvt, passive$4);
};
this.setDebounce(updateDebounce);
if (Object.keys(updateSizes).length !== 0) {
this.setSizes(updateSizes);
updateSizes = void 0; // free up memory
}
else {
this.__update();
}
// due to optimizations, this would be left out otherwise
classes === true && this.name === 'xs'
&& document.body.classList.add('screen--xs');
};
if (isRuntimeSsrPreHydration.value === true) {
onSSRHydrated.push(start);
}
else {
start();
}
}
});
const Plugin$9 = defineReactivePlugin({
isActive: false,
mode: false
}, {
__media: void 0,
set (val) {
Plugin$9.mode = val;
if (val === 'auto') {
if (Plugin$9.__media === void 0) {
Plugin$9.__media = window.matchMedia('(prefers-color-scheme: dark)');
Plugin$9.__updateMedia = () => { Plugin$9.set('auto'); };
Plugin$9.__media.addListener(Plugin$9.__updateMedia);
}
val = Plugin$9.__media.matches;
}
else if (Plugin$9.__media !== void 0) {
Plugin$9.__media.removeListener(Plugin$9.__updateMedia);
Plugin$9.__media = void 0;
}
Plugin$9.isActive = val === true;
document.body.classList.remove(`body--${ val === true ? 'light' : 'dark' }`);
document.body.classList.add(`body--${ val === true ? 'dark' : 'light' }`);
},
toggle () {
{
Plugin$9.set(Plugin$9.isActive === false);
}
},
install ({ $q, onSSRHydrated, ssrContext }) {
const { dark } = $q.config;
$q.dark = this;
if (this.__installed === true && dark === void 0) {
return
}
this.isActive = dark === true;
const initialVal = dark !== void 0 ? dark : false;
if (isRuntimeSsrPreHydration.value === true) {
const ssrSet = val => {
this.__fromSSR = val;
};
const originalSet = this.set;
this.set = ssrSet;
ssrSet(initialVal);
onSSRHydrated.push(() => {
this.set = originalSet;
this.set(this.__fromSSR);
});
}
else {
this.set(initialVal);
}
}
});
const getTrue = () => true;
function filterInvalidPath (path) {
return typeof path === 'string'
&& path !== ''
&& path !== '/'
&& path !== '#/'
}
function normalizeExitPath (path) {
path.startsWith('#') === true && (path = path.substring(1));
path.startsWith('/') === false && (path = '/' + path);
path.endsWith('/') === true && (path = path.substring(0, path.length - 1));
return '#' + path
}
function getShouldExitFn (cfg) {
if (cfg.backButtonExit === false) {
return () => false
}
if (cfg.backButtonExit === '*') {
return getTrue
}
// Add default root path
const exitPaths = [ '#/' ];
// Add custom exit paths
Array.isArray(cfg.backButtonExit) === true && exitPaths.push(
...cfg.backButtonExit.filter(filterInvalidPath).map(normalizeExitPath)
);
return () => exitPaths.includes(window.location.hash)
}
var History = {
__history: [],
add: noop,
remove: noop,
install ({ $q }) {
if (this.__installed === true) { return }
const { cordova, capacitor } = client.is;
if (cordova !== true && capacitor !== true) {
return
}
const qConf = $q.config[ cordova === true ? 'cordova' : 'capacitor' ];
if (qConf !== void 0 && qConf.backButton === false) {
return
}
// if the '@capacitor/app' plugin is not installed
// then we got nothing to do
if (
// if we're on Capacitor mode
capacitor === true
// and it's also not in Capacitor's main instance
&& (window.Capacitor === void 0 || window.Capacitor.Plugins.App === void 0)
) {
return
}
this.add = entry => {
if (entry.condition === void 0) {
entry.condition = getTrue;
}
this.__history.push(entry);
};
this.remove = entry => {
const index = this.__history.indexOf(entry);
if (index >= 0) {
this.__history.splice(index, 1);
}
};
const shouldExit = getShouldExitFn(
Object.assign(
{ backButtonExit: true },
qConf
)
);
const backHandler = () => {
if (this.__history.length) {
const entry = this.__history[ this.__history.length - 1 ];
if (entry.condition() === true) {
this.__history.pop();
entry.handler();
}
}
else if (shouldExit() === true) {
navigator.app.exitApp();
}
else {
window.history.back();
}
};
if (cordova === true) {
document.addEventListener('deviceready', () => {
document.addEventListener('backbutton', backHandler, false);
});
}
else {
window.Capacitor.Plugins.App.addListener('backButton', backHandler);
}
}
};
var defaultLang = {
isoName: 'en-US',
nativeName: 'English (US)',
label: {
clear: 'Clear',
ok: 'OK',
cancel: 'Cancel',
close: 'Close',
set: 'Set',
select: 'Select',
reset: 'Reset',
remove: 'Remove',
update: 'Update',
create: 'Create',
search: 'Search',
filter: 'Filter',
refresh: 'Refresh',
expand: label => (label ? `Expand "${ label }"` : 'Expand'),
collapse: label => (label ? `Collapse "${ label }"` : 'Collapse')
},
date: {
days: 'Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday'.split('_'),
daysShort: 'Sun_Mon_Tue_Wed_Thu_Fri_Sat'.split('_'),
months: 'January_February_March_April_May_June_July_August_September_October_November_December'.split('_'),
monthsShort: 'Jan_Feb_Mar_Apr_May_Jun_Jul_Aug_Sep_Oct_Nov_Dec'.split('_'),
firstDayOfWeek: 0, // 0-6, 0 - Sunday, 1 Monday, ...
format24h: false,
pluralDay: 'days'
},
table: {
noData: 'No data available',
noResults: 'No matching records found',
loading: 'Loading...',
selectedRecords: rows => (
rows === 1
? '1 record selected.'
: (rows === 0 ? 'No' : rows) + ' records selected.'
),
recordsPerPage: 'Records per page:',
allRows: 'All',
pagination: (start, end, total) => start + '-' + end + ' of ' + total,
columns: 'Columns'
},
editor: {
url: 'URL',
bold: 'Bold',
italic: 'Italic',
strikethrough: 'Strikethrough',
underline: 'Underline',
unorderedList: 'Unordered List',
orderedList: 'Ordered List',
subscript: 'Subscript',
superscript: 'Superscript',
hyperlink: 'Hyperlink',
toggleFullscreen: 'Toggle Fullscreen',
quote: 'Quote',
left: 'Left align',
center: 'Center align',
right: 'Right align',
justify: 'Justify align',
print: 'Print',
outdent: 'Decrease indentation',
indent: 'Increase indentation',
removeFormat: 'Remove formatting',
formatting: 'Formatting',
fontSize: 'Font Size',
align: 'Align',
hr: 'Insert Horizontal Rule',
undo: 'Undo',
redo: 'Redo',
heading1: 'Heading 1',
heading2: 'Heading 2',
heading3: 'Heading 3',
heading4: 'Heading 4',
heading5: 'Heading 5',
heading6: 'Heading 6',
paragraph: 'Paragraph',
code: 'Code',
size1: 'Very small',
size2: 'A bit small',
size3: 'Normal',
size4: 'Medium-large',
size5: 'Big',
size6: 'Very big',
size7: 'Maximum',
defaultFont: 'Default Font',
viewSource: 'View Source'
},
tree: {
noNodes: 'No nodes available',
noResults: 'No matching nodes found'
}
};
function getLocale () {
const val = Array.isArray(navigator.languages) === true && navigator.languages.length !== 0
? navigator.languages[ 0 ]
: navigator.language;
if (typeof val === 'string') {
return val.split(/[-_]/).map((v, i) => (
i === 0
? v.toLowerCase()
: (
i > 1 || v.length < 4
? v.toUpperCase()
: (v[ 0 ].toUpperCase() + v.slice(1).toLowerCase())
)
)).join('-')
}
}
const Plugin$8 = defineReactivePlugin({
__langPack: {}
}, {
getLocale,
set (langObject = defaultLang, ssrContext) {
const lang = {
...langObject,
rtl: langObject.rtl === true,
getLocale
};
{
lang.set = Plugin$8.set;
if (Plugin$8.__langConfig === void 0 || Plugin$8.__langConfig.noHtmlAttrs !== true) {
const el = document.documentElement;
el.setAttribute('dir', lang.rtl === true ? 'rtl' : 'ltr');
el.setAttribute('lang', lang.isoName);
}
Object.assign(Plugin$8.__langPack, lang);
Plugin$8.props = lang;
Plugin$8.isoName = lang.isoName;
Plugin$8.nativeName = lang.nativeName;
}
},
install ({ $q, lang, ssrContext }) {
{
$q.lang = Plugin$8.__langPack;
Plugin$8.__langConfig = $q.config.lang;
if (this.__installed === true) {
lang !== void 0 && this.set(lang);
}
else {
this.set(lang || defaultLang);
}
}
}
});
function setCssVar (propName, value, element = document.body) {
if (typeof propName !== 'string') {
throw new TypeError('Expected a string as propName')
}
if (typeof value !== 'string') {
throw new TypeError('Expected a string as value')
}
if (!(element instanceof Element)) {
throw new TypeError('Expected a DOM element')
}
element.style.setProperty(`--q-${ propName }`, value);
}
let lastKeyCompositionStatus = false;
function onKeyDownComposition (evt) {
lastKeyCompositionStatus = evt.isComposing === true;
}
function shouldIgnoreKey (evt) {
return lastKeyCompositionStatus === true
|| evt !== Object(evt)
|| evt.isComposing === true
|| evt.qKeyEvent === true
}
function isKeyCode (evt, keyCodes) {
return shouldIgnoreKey(evt) === true
? false
: [].concat(keyCodes).includes(evt.keyCode)
}
function getMobilePlatform (is) {
if (is.ios === true) return 'ios'
if (is.android === true) return 'android'
}
function getBodyClasses ({ is, has, within }, cfg) {
const cls = [
is.desktop === true ? 'desktop' : 'mobile',
`${ has.touch === false ? 'no-' : '' }touch`
];
if (is.mobile === true) {
const mobile = getMobilePlatform(is);
mobile !== void 0 && cls.push('platform-' + mobile);
}
if (is.nativeMobile === true) {
const type = is.nativeMobileWrapper;
cls.push(type);
cls.push('native-mobile');
if (
is.ios === true
&& (cfg[ type ] === void 0 || cfg[ type ].iosStatusBarPadding !== false)
) {
cls.push('q-ios-padding');
}
}
else if (is.electron === true) {
cls.push('electron');
}
else if (is.bex === true) {
cls.push('bex');
}
within.iframe === true && cls.push('within-iframe');
return cls
}
function applyClientSsrCorrections () {
const { is } = client;
const classes = document.body.className;
const classList = new Set(classes.replace(/ {2}/g, ' ').split(' '));
if (iosCorrection !== void 0) {
classList.delete('desktop');
classList.add('platform-ios');
classList.add('mobile');
}
// else: is it SSG?
else if (is.nativeMobile !== true && is.electron !== true && is.bex !== true) {
if (is.desktop === true) {
classList.delete('mobile');
classList.delete('platform-ios');
classList.delete('platform-android');
classList.add('desktop');
}
else if (is.mobile === true) {
classList.delete('desktop');
classList.add('mobile');
const mobile = getMobilePlatform(is);
if (mobile !== void 0) {
classList.add(`platform-${ mobile }`);
classList.delete(`platform-${ mobile === 'ios' ? 'android' : 'ios' }`);
}
else {
classList.delete('platform-ios');
classList.delete('platform-android');
}
}
}
if (client.has.touch === true) {
classList.delete('no-touch');
classList.add('touch');
}
if (client.within.iframe === true) {
classList.add('within-iframe');
}
const newCls = Array.from(classList).join(' ');
if (classes !== newCls) {
document.body.className = newCls;
}
}
function setColors (brand) {
for (const color in brand) {
setCssVar(color, brand[ color ]);
}
}
var Body = {
install (opts) {
if (this.__installed === true) { return }
if (isRuntimeSsrPreHydration.value === true) {
applyClientSsrCorrections();
}
else {
const { $q } = opts;
$q.config.brand !== void 0 && setColors($q.config.brand);
const cls = getBodyClasses(client, $q.config);
document.body.classList.add.apply(document.body.classList, cls);
}
if (client.is.ios === true) {
// needed for iOS button active state
document.body.addEventListener('touchstart', noop);
}
window.addEventListener('keydown', onKeyDownComposition, true);
}
};
var materialIcons = {
name: 'material-icons',
type: {
positive: 'check_circle',
negative: 'warning',
info: 'info',
warning: 'priority_high'
},
arrow: {
up: 'arrow_upward',
right: 'arrow_forward',
down: 'arrow_downward',
left: 'arrow_back',
dropdown: 'arrow_drop_down'
},
chevron: {
left: 'chevron_left',
right: 'chevron_right'
},
colorPicker: {
spectrum: 'gradient',
tune: 'tune',
palette: 'style'
},
pullToRefresh: {
icon: 'refresh'
},
carousel: {
left: 'chevron_left',
right: 'chevron_right',
up: 'keyboard_arrow_up',
down: 'keyboard_arrow_down',
navigationIcon: 'lens'
},
chip: {
remove: 'cancel',
selected: 'check'
},
datetime: {
arrowLeft: 'chevron_left',
arrowRight: 'chevron_right',
now: 'access_time',
today: 'today'
},
editor: {
bold: 'format_bold',
italic: 'format_italic',
strikethrough: 'strikethrough_s',
underline: 'format_underlined',
unorderedList: 'format_list_bulleted',
orderedList: 'format_list_numbered',
subscript: 'vertical_align_bottom',
superscript: 'vertical_align_top',
hyperlink: 'link',
toggleFullscreen: 'fullscreen',
quote: 'format_quote',
left: 'format_align_left',
center: 'format_align_center',
right: 'format_align_right',
justify: 'format_align_justify',
print: 'print',
outdent: 'format_indent_decrease',
indent: 'format_indent_increase',
removeFormat: 'format_clear',
formatting: 'text_format',
fontSize: 'format_size',
align: 'format_align_left',
hr: 'remove',
undo: 'undo',
redo: 'redo',
heading: 'format_size',
code: 'code',
size: 'format_size',
font: 'font_download',
viewSource: 'code'
},
expansionItem: {
icon: 'keyboard_arrow_down',
denseIcon: 'arrow_drop_down'
},
fab: {
icon: 'add',
activeIcon: 'close'
},
field: {
clear: 'cancel',
error: 'error'
},
pagination: {
first: 'first_page',
prev: 'keyboard_arrow_left',
next: 'keyboard_arrow_right',
last: 'last_page'
},
rating: {
icon: 'grade'
},
stepper: {
done: 'check',
active: 'edit',
error: 'warning'
},
tabs: {
left: 'chevron_left',
right: 'chevron_right',
up: 'keyboard_arrow_up',
down: 'keyboard_arrow_down'
},
table: {
arrowUp: 'arrow_upward',
warning: 'warning',
firstPage: 'first_page',
prevPage: 'chevron_left',
nextPage: 'chevron_right',
lastPage: 'last_page'
},
tree: {
icon: 'play_arrow'
},
uploader: {
done: 'done',
clear: 'clear',
add: 'add_box',
upload: 'cloud_upload',
removeQueue: 'clear_all',
removeUploaded: 'done_all'
}
};
const Plugin$7 = defineReactivePlugin({
iconMapFn: null,
__icons: {}
}, {
set (setObject, ssrContext) {
const def = { ...setObject, rtl: setObject.rtl === true };
{
def.set = Plugin$7.set;
Object.assign(Plugin$7.__icons, def);
}
},
install ({ $q, iconSet, ssrContext }) {
{
if ($q.config.iconMapFn !== void 0) {
this.iconMapFn = $q.config.iconMapFn;
}
$q.iconSet = this.__icons;
injectProp($q, 'iconMapFn', () => this.iconMapFn, val => { this.iconMapFn = val; });
if (this.__installed === true) {
iconSet !== void 0 && this.set(iconSet);
}
else {
this.set(iconSet || materialIcons);
}
}
}
});
const quasarKey = '_q_';
const timelineKey = '_q_t_';
const stepperKey = '_q_s_';
const layoutKey = '_q_l_';
const pageContainerKey = '_q_pc_';
const fabKey = '_q_f_';
const formKey = '_q_fo_';
const tabsKey = '_q_tabs_';
const uploaderKey = '_q_u_';
const emptyRenderFn = () => {};
const globalConfig = {};
let globalConfigIsFrozen = false;
function freezeGlobalConfig () {
globalConfigIsFrozen = true;
}
function isDeepEqual (a, b) {
if (a === b) {
return true
}
if (a !== null && b !== null && typeof a === 'object' && typeof b === 'object') {
if (a.constructor !== b.constructor) {
return false
}
let length, i;
if (a.constructor === Array) {
length = a.length;
if (length !== b.length) {
return false
}
for (i = length; i-- !== 0;) {
if (isDeepEqual(a[ i ], b[ i ]) !== true) {
return false
}
}
return true
}
if (a.constructor === Map) {
if (a.size !== b.size) {
return false
}
let iter = a.entries();
i = iter.next();
while (i.done !== true) {
if (b.has(i.value[ 0 ]) !== true) {
return false
}
i = iter.next();
}
iter = a.entries();
i = iter.next();
while (i.done !== true) {
if (isDeepEqual(i.value[ 1 ], b.get(i.value[ 0 ])) !== true) {
return false
}
i = iter.next();
}
return true
}
if (a.constructor === Set) {
if (a.size !== b.size) {
return false
}
const iter = a.entries();
i = iter.next();
while (i.done !== true) {
if (b.has(i.value[ 0 ]) !== true) {
return false
}
i = iter.next();
}
return true
}
if (a.buffer != null && a.buffer.constructor === ArrayBuffer) {
length = a.length;
if (length !== b.length) {
return false
}
for (i = length; i-- !== 0;) {
if (a[ i ] !== b[ i ]) {
return false
}
}
return true
}
if (a.constructor === RegExp) {
return a.source === b.source && a.flags === b.flags
}
if (a.valueOf !== Object.prototype.valueOf) {
return a.valueOf() === b.valueOf()
}
if (a.toString !== Object.prototype.toString) {
return a.toString() === b.toString()
}
const keys = Object.keys(a).filter(key => a[ key ] !== void 0);
length = keys.length;
if (length !== Object.keys(b).filter(key => b[ key ] !== void 0).length) {
return false
}
for (i = length; i-- !== 0;) {
const key = keys[ i ];
if (isDeepEqual(a[ key ], b[ key ]) !== true) {
return false
}
}
return true
}
// true if both NaN, false otherwise
return a !== a && b !== b // eslint-disable-line no-self-compare
}
// not perfect, but what we ARE interested is for Arrays not to slip in
// as spread operator will mess things up in various areas
function isObject (v) {
return v !== null && typeof v === 'object' && Array.isArray(v) !== true
}
function isDate (v) {
return Object.prototype.toString.call(v) === '[object Date]'
}
function isRegexp (v) {
return Object.prototype.toString.call(v) === '[object RegExp]'
}
function isNumber (v) {
return typeof v === 'number' && isFinite(v)
}
var is = {
deepEqual: isDeepEqual,
object: isObject,
date: isDate,
regexp: isRegexp,
number: isNumber
};
const autoInstalledPlugins = [
Platform,
Body,
Plugin$9,
Screen,
History,
Plugin$8,
Plugin$7
];
function createChildApp (appCfg, parentApp) {
const app = vue.createApp(appCfg);
app.config.globalProperties = parentApp.config.globalProperties;
const { reload, ...appContext } = parentApp._context;
Object.assign(app._context, appContext);
return app
}
function installPlugins (pluginOpts, pluginList) {
pluginList.forEach(Plugin => {
Plugin.install(pluginOpts);
Plugin.__installed = true;
});
}
function prepareApp (app, uiOpts, pluginOpts) {
app.config.globalProperties.$q = pluginOpts.$q;
app.provide(quasarKey, pluginOpts.$q);
installPlugins(pluginOpts, autoInstalledPlugins);
uiOpts.components !== void 0 && Object.values(uiOpts.components).forEach(c => {
if (isObject(c) === true && c.name !== void 0) {
app.component(c.name, c);
}
});
uiOpts.directives !== void 0 && Object.values(uiOpts.directives).forEach(d => {
if (isObject(d) === true && d.name !== void 0) {
app.directive(d.name, d);
}
});
uiOpts.plugins !== void 0 && installPlugins(
pluginOpts,
Object.values(uiOpts.plugins).filter(
p => typeof p.install === 'function' && autoInstalledPlugins.includes(p) === false
)
);
if (isRuntimeSsrPreHydration.value === true) {
pluginOpts.$q.onSSRHydrated = () => {
pluginOpts.onSSRHydrated.forEach(fn => { fn(); });
pluginOpts.$q.onSSRHydrated = () => {};
};
}
}
var installQuasar = function (parentApp, opts = {}) {
const $q = { version: '2.12.2' };
if (globalConfigIsFrozen === false) {
if (opts.config !== void 0) {
Object.assign(globalConfig, opts.config);
}
$q.config = { ...globalConfig };
freezeGlobalConfig();
}
else {
$q.config = opts.config || {};
}
prepareApp(parentApp, opts, {
parentApp,
$q,
lang: opts.lang,
iconSet: opts.iconSet,
onSSRHydrated: []
});
};
const createComponent = raw => vue.markRaw(vue.defineComponent(raw));
const createDirective = raw => vue.markRaw(raw);
const units = [ 'B', 'KB', 'MB', 'GB', 'TB', 'PB' ];
function humanStorageSize (bytes) {
let u = 0;
while (parseInt(bytes, 10) >= 1024 && u < units.length - 1) {
bytes /= 1024;
++u;
}
return `${ bytes.toFixed(1) }${ units[ u ] }`
}
function capitalize (str) {
return str.charAt(0).toUpperCase() + str.slice(1)
}
function between (v, min, max) {
return max <= min
? min
: Math.min(max, Math.max(min, v))
}
function normalizeToInterval (v, min, max) {
if (max <= min) {
return min
}
const size = (max - min + 1);
let index = min + (v - min) % size;
if (index < min) {
index = size + index;
}
return index === 0 ? 0 : index // fix for (-a % a) => -0
}
function pad (v, length = 2, char = '0') {
if (v === void 0 || v === null) {
return v
}
const val = '' + v;
return val.length >= length
? val
: new Array(length - val.length + 1).join(char) + val
}
var format = {
humanStorageSize,
capitalize,
between,
normalizeToInterval,
pad
};
const
xhr = XMLHttpRequest,
open = xhr.prototype.open,
positionValues = [ 'top', 'right', 'bottom', 'left' ];
let stack = [];
let highjackCount = 0;
function translate ({ p, pos, active, horiz, reverse, dir }) {
let x = 1, y = 1;
if (horiz === true) {
if (reverse === true) { x = -1; }
if (pos === 'bottom') { y = -1; }
return { transform: `translate3d(${ x * (p - 100) }%,${ active ? 0 : y * -200 }%,0)` }
}
if (reverse === true) { y = -1; }
if (pos === 'right') { x = -1; }
return { transform: `translate3d(${ active ? 0 : dir * x * -200 }%,${ y * (p - 100) }%,0)` }
}
function inc (p, amount) {
if (typeof amount !== 'number') {
if (p < 25) {
amount = Math.random() * 3 + 3;
}
else if (p < 65) {
amount = Math.random() * 3;
}
else if (p < 85) {
amount = Math.random() * 2;
}
else if (p < 99) {
amount = 0.6;
}
else {
amount = 0;
}
}
return between(p + amount, 0, 100)
}
function highjackAjax (stackEntry) {
highjackCount++;
stack.push(stackEntry);
if (highjackCount > 1) { return }
xhr.prototype.open = function (_, url) {
const stopStack = [];
const loadStart = () => {
stack.forEach(entry => {
if (
entry.hijackFilter.value === null
|| (entry.hijackFilter.value(url) === true)
) {
entry.start();
stopStack.push(entry.stop);
}
});
};
const loadEnd = () => {
stopStack.forEach(stop => { stop(); });
};
this.addEventListener('loadstart', loadStart, { once: true });
this.addEventListener('loadend', loadEnd, { once: true });
open.apply(this, arguments);
};
}
function restoreAjax (start) {
stack = stack.filter(entry => entry.start !== start);
highjackCount = Math.max(0, highjackCount - 1);
if (highjackCount === 0) {
xhr.prototype.open = open;
}
}
var QAjaxBar = createComponent({
name: 'QAjaxBar',
props: {
position: {
type: String,
default: 'top',
validator: val => positionValues.includes(val)
},
size: {
type: String,
default: '2px'
},
color: String,
skipHijack: Boolean,
reverse: Boolean,
hijackFilter: Function
},
emits: [ 'start', 'stop' ],
setup (props, { emit }) {
const { proxy } = vue.getCurrentInstance();
const progress = vue.ref(0);
const onScreen = vue.ref(false);
const animate = vue.ref(true);
let sessions = 0, timer = null, speed;
const classes = vue.computed(() =>
`q-loading-bar q-loading-bar--${ props.position }`
+ (props.color !== void 0 ? ` bg-${ props.color }` : '')
+ (animate.value === true ? '' : ' no-transition')
);
const horizontal = vue.computed(() => props.position === 'top' || props.position === 'bottom');
const sizeProp = vue.computed(() => (horizontal.value === true ? 'height' : 'width'));
const style = vue.computed(() => {
const active = onScreen.value;
const obj = translate({
p: progress.value,
pos: props.position,
active,
horiz: horizontal.value,
reverse: proxy.$q.lang.rtl === true && [ 'top', 'bottom' ].includes(props.position)
? props.reverse === false
: props.reverse,
dir: proxy.$q.lang.rtl === true ? -1 : 1
});
obj[ sizeProp.value ] = props.size;
obj.opacity = active ? 1 : 0;
return obj
});
const attributes = vue.computed(() => (
onScreen.value === true
? {
role: 'progressbar',
'aria-valuemin': 0,
'aria-valuemax': 100,
'aria-valuenow': progress.value
}
: { 'aria-hidden': 'true' }
));
function start (newSpeed = 300) {
const oldSpeed = speed;
speed = Math.max(0, newSpeed) || 0;
sessions++;
if (sessions > 1) {
if (oldSpeed === 0 && newSpeed > 0) {
planNextStep();
}
else if (timer !== null && oldSpeed > 0 && newSpeed <= 0) {
clearTimeout(timer);
timer = null;
}
return sessions
}
timer !== null && clearTimeout(timer);
emit('start');
progress.value = 0;
timer = setTimeout(() => {
timer = null;
animate.value = true;
newSpeed > 0 && planNextStep();
}, onScreen.value === true ? 500 : 1);
if (onScreen.value !== true) {
onScreen.value = true;
animate.value = false;
}
return sessions
}
function increment (amount) {
if (sessions > 0) {
progress.value = inc(progress.value, amount);
}
return sessions
}
function stop () {
sessions = Math.max(0, sessions - 1);
if (sessions > 0) {
return sessions
}
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
emit('stop');
const end = () => {
animate.value = true;
progress.value = 100;
timer = setTimeout(() => {
timer = null;
onScreen.value = false;
}, 1000);
};
if (progress.value === 0) {
timer = setTimeout(end, 1);
}
else {
end();
}
return sessions
}
function planNextStep () {
if (progress.value < 100) {
timer = setTimeout(() => {
timer = null;
increment();
planNextStep();
}, speed);
}
}
let hijacked;
vue.onMounted(() => {
if (props.skipHijack !== true) {
hijacked = true;
highjackAjax({
start,
stop,
hijackFilter: vue.computed(() => props.hijackFilter || null)
});
}
});
vue.onBeforeUnmount(() => {
timer !== null && clearTimeout(timer);
hijacked === true && restoreAjax(start);
});
// expose public methods
Object.assign(proxy, { start, stop, increment });
return () => vue.h('div', {
class: classes.value,
style: style.value,
...attributes.value
})
}
});
const useSizeDefaults = {
xs: 18,
sm: 24,
md: 32,
lg: 38,
xl: 46
};
const useSizeProps = {
size: String
};
function useSize (props, sizes = useSizeDefaults) {
// return sizeStyle
return vue.computed(() => (
props.size !== void 0
? { fontSize: props.size in sizes ? `${ sizes[ props.size ] }px` : props.size }
: null
))
}
function hSlot (slot, otherwise) {
return slot !== void 0
? slot() || otherwise
: otherwise
}
function hUniqueSlot (slot, otherwise) {
if (slot !== void 0) {
const vnode = slot();
if (vnode !== void 0 && vnode !== null) {
return vnode.slice()
}
}
return otherwise
}
/**
* Source definitely exists,
* so it's merged with the possible slot
*/
function hMergeSlot (slot, source) {
return slot !== void 0
? source.concat(slot())
: source
}
/**
* Merge with possible slot,
* even if source might not exist
*/
function hMergeSlotSafely (slot, source) {
if (slot === void 0) {
return source
}
return source !== void 0
? source.concat(slot())
: slot()
}
/*
* (String) key - unique vnode key
* (Boolean) condition - should change ONLY when adding/removing directive
*/
function hDir (
tag,
data,
children,
key,
condition,
getDirsFn
) {
data.key = key + condition;
const vnode = vue.h(tag, data, children);
return condition === true
? vue.withDirectives(vnode, getDirsFn())
: vnode
}
const defaultViewBox = '0 0 24 24';
const sameFn = i => i;
const ionFn = i => `ionicons ${ i }`;
const libMap = {
'mdi-': i => `mdi ${ i }`,
'icon-': sameFn, // fontawesome equiv
'bt-': i => `bt ${ i }`,
'eva-': i => `eva ${ i }`,
'ion-md': ionFn,
'ion-ios': ionFn,
'ion-logo': ionFn,
'iconfont ': sameFn,
'ti-': i => `themify-icon ${ i }`,
'bi-': i => `bootstrap-icons ${ i }`
};
const matMap = {
o_: '-outlined',
r_: '-round',
s_: '-sharp'
};
const symMap = {
sym_o_: '-outlined',
sym_r_: '-rounded',
sym_s_: '-sharp'
};
const libRE = new RegExp('^(' + Object.keys(libMap).join('|') + ')');
const matRE = new RegExp('^(' + Object.keys(matMap).join('|') + ')');
const symRE = new RegExp('^(' + Object.keys(symMap).join('|') + ')');
const mRE = /^[Mm]\s?[-+]?\.?\d/;
const imgRE = /^img:/;
const svgUseRE = /^svguse:/;
const ionRE = /^ion-/;
const faRE = /^(fa-(sharp|solid|regular|light|brands|duotone|thin)|[lf]a[srlbdk]?) /;
var QIcon = createComponent({
name: 'QIcon',
props: {
...useSizeProps,
tag: {
type: String,
default: 'i'
},
name: String,
color: String,
left: Boolean,
right: Boolean
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const sizeStyle = useSize(props);
const classes = vue.computed(() =>
'q-icon'
+ (props.left === true ? ' on-left' : '') // TODO Qv3: drop this
+ (props.right === true ? ' on-right' : '')
+ (props.color !== void 0 ? ` text-${ props.color }` : '')
);
const type = vue.computed(() => {
let cls;
let icon = props.name;
if (icon === 'none' || !icon) {
return { none: true }
}
if ($q.iconMapFn !== null) {
const res = $q.iconMapFn(icon);
if (res !== void 0) {
if (res.icon !== void 0) {
icon = res.icon;
if (icon === 'none' || !icon) {
return { none: true }
}
}
else {
return {
cls: res.cls,
content: res.content !== void 0
? res.content
: ' '
}
}
}
}
if (mRE.test(icon) === true) {
const [ def, viewBox = defaultViewBox ] = icon.split('|');
return {
svg: true,
viewBox,
nodes: def.split('&&').map(path => {
const [ d, style, transform ] = path.split('@@');
return vue.h('path', { style, d, transform })
})
}
}
if (imgRE.test(icon) === true) {
return {
img: true,
src: icon.substring(4)
}
}
if (svgUseRE.test(icon) === true) {
const [ def, viewBox = defaultViewBox ] = icon.split('|');
return {
svguse: true,
src: def.substring(7),
viewBox
}
}
let content = ' ';
const matches = icon.match(libRE);
if (matches !== null) {
cls = libMap[ matches[ 1 ] ](icon);
}
else if (faRE.test(icon) === true) {
cls = icon;
}
else if (ionRE.test(icon) === true) {
cls = `ionicons ion-${ $q.platform.is.ios === true ? 'ios' : 'md' }${ icon.substring(3) }`;
}
else if (symRE.test(icon) === true) {
// "notranslate" class is for Google Translate
// to avoid tampering with Material Symbols ligature font
//
// Caution: To be able to add suffix to the class name,
// keep the 'material-symbols' at the end of the string.
cls = 'notranslate material-symbols';
const matches = icon.match(symRE);
if (matches !== null) {
icon = icon.substring(6);
cls += symMap[ matches[ 1 ] ];
}
content = icon;
}
else {
// "notranslate" class is for Google Translate
// to avoid tampering with Material Icons ligature font
//
// Caution: To be able to add suffix to the class name,
// keep the 'material-icons' at the end of the string.
cls = 'notranslate material-icons';
const matches = icon.match(matRE);
if (matches !== null) {
icon = icon.substring(2);
cls += matMap[ matches[ 1 ] ];
}
content = icon;
}
return {
cls,
content
}
});
return () => {
const data = {
class: classes.value,
style: sizeStyle.value,
'aria-hidden': 'true',
role: 'presentation'
};
if (type.value.none === true) {
return vue.h(props.tag, data, hSlot(slots.default))
}
if (type.value.img === true) {
return vue.h('span', data, hMergeSlot(slots.default, [
vue.h('img', { src: type.value.src })
]))
}
if (type.value.svg === true) {
return vue.h('span', data, hMergeSlot(slots.default, [
vue.h('svg', {
viewBox: type.value.viewBox || '0 0 24 24'
}, type.value.nodes)
]))
}
if (type.value.svguse === true) {
return vue.h('span', data, hMergeSlot(slots.default, [
vue.h('svg', {
viewBox: type.value.viewBox
}, [
vue.h('use', { 'xlink:href': type.value.src })
])
]))
}
if (type.value.cls !== void 0) {
data.class += ' ' + type.value.cls;
}
return vue.h(props.tag, data, hMergeSlot(slots.default, [
type.value.content
]))
}
}
});
var QAvatar = createComponent({
name: 'QAvatar',
props: {
...useSizeProps,
fontSize: String,
color: String,
textColor: String,
icon: String,
square: Boolean,
rounded: Boolean
},
setup (props, { slots }) {
const sizeStyle = useSize(props);
const classes = vue.computed(() =>
'q-avatar'
+ (props.color ? ` bg-${ props.color }` : '')
+ (props.textColor ? ` text-${ props.textColor } q-chip--colored` : '')
+ (
props.square === true
? ' q-avatar--square'
: (props.rounded === true ? ' rounded-borders' : '')
)
);
const contentStyle = vue.computed(() => (
props.fontSize
? { fontSize: props.fontSize }
: null
));
return () => {
const icon = props.icon !== void 0
? [ vue.h(QIcon, { name: props.icon }) ]
: void 0;
return vue.h('div', {
class: classes.value,
style: sizeStyle.value
}, [
vue.h('div', {
class: 'q-avatar__content row flex-center overflow-hidden',
style: contentStyle.value
}, hMergeSlotSafely(slots.default, icon))
])
}
}
});
const alignValues$3 = [ 'top', 'middle', 'bottom' ];
var QBadge = createComponent({
name: 'QBadge',
props: {
color: String,
textColor: String,
floating: Boolean,
transparent: Boolean,
multiLine: Boolean,
outline: Boolean,
rounded: Boolean,
label: [ Number, String ],
align: {
type: String,
validator: v => alignValues$3.includes(v)
}
},
setup (props, { slots }) {
const style = vue.computed(() => {
return props.align !== void 0
? { verticalAlign: props.align }
: null
});
const classes = vue.computed(() => {
const text = props.outline === true
? props.color || props.textColor
: props.textColor;
return 'q-badge flex inline items-center no-wrap'
+ ` q-badge--${ props.multiLine === true ? 'multi' : 'single' }-line`
+ (props.outline === true
? ' q-badge--outline'
: (props.color !== void 0 ? ` bg-${ props.color }` : '')
)
+ (text !== void 0 ? ` text-${ text }` : '')
+ (props.floating === true ? ' q-badge--floating' : '')
+ (props.rounded === true ? ' q-badge--rounded' : '')
+ (props.transparent === true ? ' q-badge--transparent' : '')
});
return () => vue.h('div', {
class: classes.value,
style: style.value,
role: 'status',
'aria-label': props.label
}, hMergeSlot(slots.default, props.label !== void 0 ? [ props.label ] : []))
}
});
const useDarkProps = {
dark: {
type: Boolean,
default: null
}
};
function useDark (props, $q) {
// return isDark
return vue.computed(() => (
props.dark === null
? $q.dark.isActive
: props.dark
))
}
var QBanner = createComponent({
name: 'QBanner',
props: {
...useDarkProps,
inlineActions: Boolean,
dense: Boolean,
rounded: Boolean
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const classes = vue.computed(() =>
'q-banner row items-center'
+ (props.dense === true ? ' q-banner--dense' : '')
+ (isDark.value === true ? ' q-banner--dark q-dark' : '')
+ (props.rounded === true ? ' rounded-borders' : '')
);
const actionClass = vue.computed(() =>
'q-banner__actions row items-center justify-end'
+ ` col-${ props.inlineActions === true ? 'auto' : 'all' }`
);
return () => {
const child = [
vue.h('div', {
class: 'q-banner__avatar col-auto row items-center self-start'
}, hSlot(slots.avatar)),
vue.h('div', {
class: 'q-banner__content col text-body2'
}, hSlot(slots.default))
];
const actions = hSlot(slots.action);
actions !== void 0 && child.push(
vue.h('div', { class: actionClass.value }, actions)
);
return vue.h('div', {
class: classes.value
+ (props.inlineActions === false && actions !== void 0 ? ' q-banner--top-padding' : ''),
role: 'alert'
}, child)
}
}
});
var QBar = createComponent({
name: 'QBar',
props: {
...useDarkProps,
dense: Boolean
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const classes = vue.computed(() =>
'q-bar row no-wrap items-center'
+ ` q-bar--${ props.dense === true ? 'dense' : 'standard' } `
+ ` q-bar--${ isDark.value === true ? 'dark' : 'light' }`
);
return () => vue.h('div', {
class: classes.value,
role: 'toolbar'
}, hSlot(slots.default))
}
});
const alignMap = {
left: 'start',
center: 'center',
right: 'end',
between: 'between',
around: 'around',
evenly: 'evenly',
stretch: 'stretch'
};
const alignValues$2 = Object.keys(alignMap);
const useAlignProps = {
align: {
type: String,
validator: v => alignValues$2.includes(v)
}
};
function useAlign (props) {
// return alignClass
return vue.computed(() => {
const align = props.align === void 0
? props.vertical === true ? 'stretch' : 'left'
: props.align;
return `${ props.vertical === true ? 'items' : 'justify' }-${ alignMap[ align ] }`
})
}
// copied to docs too
function getParentProxy (proxy) {
if (Object(proxy.$parent) === proxy.$parent) {
return proxy.$parent
}
let { parent } = proxy.$;
while (Object(parent) === parent) {
if (Object(parent.proxy) === parent.proxy) {
return parent.proxy
}
parent = parent.parent;
}
}
function fillNormalizedVNodes (children, vnode) {
if (typeof vnode.type === 'symbol') {
if (Array.isArray(vnode.children) === true) {
vnode.children.forEach(child => {
fillNormalizedVNodes(children, child);
});
}
}
else {
children.add(vnode);
}
}
// vnodes from rendered in advanced slots
function getNormalizedVNodes (vnodes) {
const children = new Set();
vnodes.forEach(vnode => {
fillNormalizedVNodes(children, vnode);
});
return Array.from(children)
}
function vmHasRouter (vm) {
return vm.appContext.config.globalProperties.$router !== void 0
}
function vmIsDestroyed (vm) {
return vm.isUnmounted === true || vm.isDeactivated === true
}
const disabledValues = [ '', true ];
var QBreadcrumbs = createComponent({
name: 'QBreadcrumbs',
props: {
...useAlignProps,
separator: {
type: String,
default: '/'
},
separatorColor: String,
activeColor: {
type: String,
default: 'primary'
},
gutter: {
type: String,
validator: v => [ 'none', 'xs', 'sm', 'md', 'lg', 'xl' ].includes(v),
default: 'sm'
}
},
setup (props, { slots }) {
const alignClass = useAlign(props);
const classes = vue.computed(() =>
`flex items-center ${ alignClass.value }${ props.gutter === 'none' ? '' : ` q-gutter-${ props.gutter }` }`
);
const sepClass = vue.computed(() => (props.separatorColor ? ` text-${ props.separatorColor }` : ''));
const activeClass = vue.computed(() => ` text-${ props.activeColor }`);
return () => {
const vnodes = getNormalizedVNodes(
hSlot(slots.default)
);
if (vnodes.length === 0) { return }
let els = 1;
const
child = [],
len = vnodes.filter(c => c.type !== void 0 && c.type.name === 'QBreadcrumbsEl').length,
separator = slots.separator !== void 0
? slots.separator
: () => props.separator;
vnodes.forEach(comp => {
if (comp.type !== void 0 && comp.type.name === 'QBreadcrumbsEl') {
const middle = els < len;
const disabled = comp.props !== null && disabledValues.includes(comp.props.disable);
const cls = (middle === true ? '' : ' q-breadcrumbs--last')
+ (disabled !== true && middle === true ? activeClass.value : '');
els++;
child.push(
vue.h('div', {
class: `flex items-center${ cls }`
}, [ comp ])
);
if (middle === true) {
child.push(
vue.h('div', {
class: 'q-breadcrumbs__separator' + sepClass.value
}, separator())
);
}
}
else {
child.push(comp);
}
});
return vue.h('div', {
class: 'q-breadcrumbs'
}, [
vue.h('div', { class: classes.value }, child)
])
}
}
});
/*
* Inspired by RouterLink from Vue Router
* --> API should match!
*/
// Get the original path value of a record by following its aliasOf
function getOriginalPath (record) {
return record
? (
record.aliasOf
? record.aliasOf.path
: record.path
) : ''
}
function isSameRouteRecord (a, b) {
// since the original record has an undefined value for aliasOf
// but all aliases point to the original record, this will always compare
// the original record
return (a.aliasOf || a) === (b.aliasOf || b)
}
function includesParams (outer, inner) {
for (const key in inner) {
const
innerValue = inner[ key ],
outerValue = outer[ key ];
if (typeof innerValue === 'string') {
if (innerValue !== outerValue) {
return false
}
}
else if (
Array.isArray(outerValue) === false
|| outerValue.length !== innerValue.length
|| innerValue.some((value, i) => value !== outerValue[ i ])
) {
return false
}
}
return true
}
function isEquivalentArray (a, b) {
return Array.isArray(b) === true
? a.length === b.length && a.every((value, i) => value === b[ i ])
: a.length === 1 && a[ 0 ] === b
}
function isSameRouteLocationParamsValue (a, b) {
return Array.isArray(a) === true
? isEquivalentArray(a, b)
: (
Array.isArray(b) === true
? isEquivalentArray(b, a)
: a === b
)
}
function isSameRouteLocationParams (a, b) {
if (Object.keys(a).length !== Object.keys(b).length) {
return false
}
for (const key in a) {
if (isSameRouteLocationParamsValue(a[ key ], b[ key ]) === false) {
return false
}
}
return true
}
const useRouterLinkProps = {
// router-link
to: [ String, Object ],
replace: Boolean,
exact: Boolean,
activeClass: {
type: String,
default: 'q-router-link--active'
},
exactActiveClass: {
type: String,
default: 'q-router-link--exact-active'
},
// regular <a> link
href: String,
target: String,
// state
disable: Boolean
};
// external props: type, tag
function useRouterLink ({ fallbackTag, useDisableForRouterLinkProps = true } = {}) {
const vm = vue.getCurrentInstance();
const { props, proxy, emit } = vm;
const hasRouter = vmHasRouter(vm);
const hasHrefLink = vue.computed(() => props.disable !== true && props.href !== void 0);
// for perf reasons, we use minimum amount of runtime work
const hasRouterLinkProps = useDisableForRouterLinkProps === true
? vue.computed(() =>
hasRouter === true
&& props.disable !== true
&& hasHrefLink.value !== true
&& props.to !== void 0 && props.to !== null && props.to !== ''
)
: vue.computed(() =>
hasRouter === true
&& hasHrefLink.value !== true
&& props.to !== void 0 && props.to !== null && props.to !== ''
);
const resolvedLink = vue.computed(() => (
hasRouterLinkProps.value === true
? getLink(props.to)
: null
));
const hasRouterLink = vue.computed(() => resolvedLink.value !== null);
const hasLink = vue.computed(() => hasHrefLink.value === true || hasRouterLink.value === true);
const linkTag = vue.computed(() => (
props.type === 'a' || hasLink.value === true
? 'a'
: (props.tag || fallbackTag || 'div')
));
const linkAttrs = vue.computed(() => (
hasHrefLink.value === true
? {
href: props.href,
target: props.target
}
: (
hasRouterLink.value === true
? {
href: resolvedLink.value.href,
target: props.target
}
: {}
)
));
const linkActiveIndex = vue.computed(() => {
if (hasRouterLink.value === false) {
return -1
}
const
{ matched } = resolvedLink.value,
{ length } = matched,
routeMatched = matched[ length - 1 ];
if (routeMatched === void 0) {
return -1
}
const currentMatched = proxy.$route.matched;
if (currentMatched.length === 0) {
return -1
}
const index = currentMatched.findIndex(
isSameRouteRecord.bind(null, routeMatched)
);
if (index > -1) {
return index
}
// possible parent record
const parentRecordPath = getOriginalPath(matched[ length - 2 ]);
return (
// we are dealing with nested routes
length > 1
// if the parent and matched route have the same path, this link is
// referring to the empty child. Or we currently are on a different
// child of the same parent
&& getOriginalPath(routeMatched) === parentRecordPath
// avoid comparing the child with its parent
&& currentMatched[ currentMatched.length - 1 ].path !== parentRecordPath
? currentMatched.findIndex(
isSameRouteRecord.bind(null, matched[ length - 2 ])
)
: index
)
});
const linkIsActive = vue.computed(() =>
hasRouterLink.value === true
&& linkActiveIndex.value !== -1
&& includesParams(proxy.$route.params, resolvedLink.value.params)
);
const linkIsExactActive = vue.computed(() =>
linkIsActive.value === true
&& linkActiveIndex.value === proxy.$route.matched.length - 1
&& isSameRouteLocationParams(proxy.$route.params, resolvedLink.value.params)
);
const linkClass = vue.computed(() => (
hasRouterLink.value === true
? (
linkIsExactActive.value === true
? ` ${ props.exactActiveClass } ${ props.activeClass }`
: (
props.exact === true
? ''
: (linkIsActive.value === true ? ` ${ props.activeClass }` : '')
)
)
: ''
));
function getLink (to) {
try { return proxy.$router.resolve(to) }
catch (_) {}
return null
}
/**
* @returns Promise<RouterError | false | undefined>
*/
function navigateToRouterLink (
e,
{ returnRouterError, to = props.to, replace = props.replace } = {}
) {
if (props.disable === true) {
// ensure native navigation is prevented in all cases,
// like when useDisableForRouterLinkProps === false (QRouteTab)
e.preventDefault();
return Promise.resolve(false)
}
if (
// don't redirect with control keys;
// should match RouterLink from Vue Router
e.metaKey || e.altKey || e.ctrlKey || e.shiftKey
// don't redirect on right click
|| (e.button !== void 0 && e.button !== 0)
// don't redirect if it should open in a new window
|| props.target === '_blank'
) {
return Promise.resolve(false)
}
// hinder the native navigation
e.preventDefault();
// then() can also return a "soft" router error (Vue Router behavior)
const promise = proxy.$router[ replace === true ? 'replace' : 'push' ](to);
return returnRouterError === true
? promise
// else catching hard errors and also "soft" ones - then(err => ...)
: promise.then(() => {}).catch(() => {})
}
// warning! ensure that the component using it has 'click' included in its 'emits' definition prop
function navigateOnClick (e) {
if (hasRouterLink.value === true) {
const go = opts => navigateToRouterLink(e, opts);
emit('click', e, go);
e.defaultPrevented !== true && go();
}
else {
emit('click', e);
}
}
return {
hasRouterLink,
hasHrefLink,
hasLink,
linkTag,
resolvedLink,
linkIsActive,
linkIsExactActive,
linkClass,
linkAttrs,
getLink,
navigateToRouterLink,
navigateOnClick
}
}
var QBreadcrumbsEl = createComponent({
name: 'QBreadcrumbsEl',
props: {
...useRouterLinkProps,
label: String,
icon: String,
tag: {
type: String,
default: 'span'
}
},
emits: [ 'click' ],
setup (props, { slots }) {
const { linkTag, linkAttrs, linkClass, navigateOnClick } = useRouterLink();
const data = vue.computed(() => {
return {
class: 'q-breadcrumbs__el q-link '
+ 'flex inline items-center relative-position '
+ (props.disable !== true ? 'q-link--focusable' + linkClass.value : 'q-breadcrumbs__el--disable'),
...linkAttrs.value,
onClick: navigateOnClick
}
});
const iconClass = vue.computed(() =>
'q-breadcrumbs__el-icon'
+ (props.label !== void 0 ? ' q-breadcrumbs__el-icon--with-label' : '')
);
return () => {
const child = [];
props.icon !== void 0 && child.push(
vue.h(QIcon, {
class: iconClass.value,
name: props.icon
})
);
props.label !== void 0 && child.push(props.label);
return vue.h(
linkTag.value,
{ ...data.value },
hMergeSlot(slots.default, child)
)
}
}
});
const useSpinnerProps = {
size: {
type: [ Number, String ],
default: '1em'
},
color: String
};
function useSpinner (props) {
return {
cSize: vue.computed(() => (
props.size in useSizeDefaults
? `${ useSizeDefaults[ props.size ] }px`
: props.size
)),
classes: vue.computed(() =>
'q-spinner' + (props.color ? ` text-${ props.color }` : '')
)
}
}
var QSpinner = createComponent({
name: 'QSpinner',
props: {
...useSpinnerProps,
thickness: {
type: Number,
default: 5
}
},
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value + ' q-spinner-mat',
width: cSize.value,
height: cSize.value,
viewBox: '25 25 50 50'
}, [
vue.h('circle', {
class: 'path',
cx: '50',
cy: '50',
r: '20',
fill: 'none',
stroke: 'currentColor',
'stroke-width': props.thickness,
'stroke-miterlimit': '10'
})
])
}
});
function offset (el) {
if (el === window) {
return { top: 0, left: 0 }
}
const { top, left } = el.getBoundingClientRect();
return { top, left }
}
function style (el, property) {
return window.getComputedStyle(el).getPropertyValue(property)
}
function height (el) {
return el === window
? window.innerHeight
: el.getBoundingClientRect().height
}
function width$1 (el) {
return el === window
? window.innerWidth
: el.getBoundingClientRect().width
}
function css (element, css) {
const style = element.style;
for (const prop in css) {
style[ prop ] = css[ prop ];
}
}
function cssBatch (elements, style) {
elements.forEach(el => css(el, style));
}
function ready (fn) {
if (typeof fn !== 'function') {
return
}
if (document.readyState !== 'loading') {
return fn()
}
document.addEventListener('DOMContentLoaded', fn, false);
}
// internal
function getElement$1 (el) {
if (el === void 0 || el === null) {
return void 0
}
if (typeof el === 'string') {
try {
return document.querySelector(el) || void 0
}
catch (err) {
return void 0
}
}
const target = vue.unref(el);
if (target) {
return target.$el || target
}
}
// internal
function childHasFocus (el, focusedEl) {
if (el === void 0 || el === null || el.contains(focusedEl) === true) {
return true
}
for (let next = el.nextElementSibling; next !== null; next = next.nextElementSibling) {
if (next.contains(focusedEl)) {
return true
}
}
return false
}
var dom = {
offset,
style,
height,
width: width$1,
css,
cssBatch,
ready
};
function throttle (fn, limit = 250) {
let wait = false, result;
return function (/* ...args */) {
if (wait === false) {
wait = true;
setTimeout(() => { wait = false; }, limit);
result = fn.apply(this, arguments);
}
return result
}
}
function showRipple (evt, el, ctx, forceCenter) {
ctx.modifiers.stop === true && stop(evt);
const color = ctx.modifiers.color;
let center = ctx.modifiers.center;
center = center === true || forceCenter === true;
const
node = document.createElement('span'),
innerNode = document.createElement('span'),
pos = position(evt),
{ left, top, width, height } = el.getBoundingClientRect(),
diameter = Math.sqrt(width * width + height * height),
radius = diameter / 2,
centerX = `${ (width - diameter) / 2 }px`,
x = center ? centerX : `${ pos.left - left - radius }px`,
centerY = `${ (height - diameter) / 2 }px`,
y = center ? centerY : `${ pos.top - top - radius }px`;
innerNode.className = 'q-ripple__inner';
css(innerNode, {
height: `${ diameter }px`,
width: `${ diameter }px`,
transform: `translate3d(${ x },${ y },0) scale3d(.2,.2,1)`,
opacity: 0
});
node.className = `q-ripple${ color ? ' text-' + color : '' }`;
node.setAttribute('dir', 'ltr');
node.appendChild(innerNode);
el.appendChild(node);
const abort = () => {
node.remove();
clearTimeout(timer);
};
ctx.abort.push(abort);
let timer = setTimeout(() => {
innerNode.classList.add('q-ripple__inner--enter');
innerNode.style.transform = `translate3d(${ centerX },${ centerY },0) scale3d(1,1,1)`;
innerNode.style.opacity = 0.2;
timer = setTimeout(() => {
innerNode.classList.remove('q-ripple__inner--enter');
innerNode.classList.add('q-ripple__inner--leave');
innerNode.style.opacity = 0;
timer = setTimeout(() => {
node.remove();
ctx.abort.splice(ctx.abort.indexOf(abort), 1);
}, 275);
}, 250);
}, 50);
}
function updateModifiers$1 (ctx, { modifiers, value, arg }) {
const cfg = Object.assign({}, ctx.cfg.ripple, modifiers, value);
ctx.modifiers = {
early: cfg.early === true,
stop: cfg.stop === true,
center: cfg.center === true,
color: cfg.color || arg,
keyCodes: [].concat(cfg.keyCodes || 13)
};
}
var Ripple = createDirective({
name: 'ripple',
beforeMount (el, binding) {
const cfg = binding.instance.$.appContext.config.globalProperties.$q.config || {};
if (cfg.ripple === false) {
return
}
const ctx = {
cfg,
enabled: binding.value !== false,
modifiers: {},
abort: [],
start (evt) {
if (
ctx.enabled === true
&& evt.qSkipRipple !== true
&& evt.type === (ctx.modifiers.early === true ? 'pointerdown' : 'click')
) {
showRipple(evt, el, ctx, evt.qKeyEvent === true);
}
},
keystart: throttle(evt => {
if (
ctx.enabled === true
&& evt.qSkipRipple !== true
&& isKeyCode(evt, ctx.modifiers.keyCodes) === true
&& evt.type === `key${ ctx.modifiers.early === true ? 'down' : 'up' }`
) {
showRipple(evt, el, ctx, true);
}
}, 300)
};
updateModifiers$1(ctx, binding);
el.__qripple = ctx;
addEvt(ctx, 'main', [
[ el, 'pointerdown', 'start', 'passive' ],
[ el, 'click', 'start', 'passive' ],
[ el, 'keydown', 'keystart', 'passive' ],
[ el, 'keyup', 'keystart', 'passive' ]
]);
},
updated (el, binding) {
if (binding.oldValue !== binding.value) {
const ctx = el.__qripple;
if (ctx !== void 0) {
ctx.enabled = binding.value !== false;
if (ctx.enabled === true && Object(binding.value) === binding.value) {
updateModifiers$1(ctx, binding);
}
}
}
},
beforeUnmount (el) {
const ctx = el.__qripple;
if (ctx !== void 0) {
ctx.abort.forEach(fn => { fn(); });
cleanEvt(ctx, 'main');
delete el._qripple;
}
}
}
);
const btnPadding = {
none: 0,
xs: 4,
sm: 8,
md: 16,
lg: 24,
xl: 32
};
const defaultSizes$2 = {
xs: 8,
sm: 10,
md: 14,
lg: 20,
xl: 24
};
const formTypes = [ 'button', 'submit', 'reset' ];
const mediaTypeRE = /[^\s]\/[^\s]/;
const btnDesignOptions = [ 'flat', 'outline', 'push', 'unelevated' ];
const getBtnDesign = (props, defaultValue) => {
if (props.flat === true) return 'flat'
if (props.outline === true) return 'outline'
if (props.push === true) return 'push'
if (props.unelevated === true) return 'unelevated'
return defaultValue
};
const getBtnDesignAttr = props => {
const design = getBtnDesign(props);
return design !== void 0
? { [ design ]: true }
: {}
};
const useBtnProps = {
...useSizeProps,
...useRouterLinkProps,
type: {
type: String,
default: 'button'
},
label: [ Number, String ],
icon: String,
iconRight: String,
...btnDesignOptions.reduce(
(acc, val) => (acc[ val ] = Boolean) && acc,
{}
),
square: Boolean,
round: Boolean,
rounded: Boolean,
glossy: Boolean,
size: String,
fab: Boolean,
fabMini: Boolean,
padding: String,
color: String,
textColor: String,
noCaps: Boolean,
noWrap: Boolean,
dense: Boolean,
tabindex: [ Number, String ],
ripple: {
type: [ Boolean, Object ],
default: true
},
align: {
...useAlignProps.align,
default: 'center'
},
stack: Boolean,
stretch: Boolean,
loading: {
type: Boolean,
default: null
},
disable: Boolean
};
function useBtn (props) {
const sizeStyle = useSize(props, defaultSizes$2);
const alignClass = useAlign(props);
const { hasRouterLink, hasLink, linkTag, linkAttrs, navigateOnClick } = useRouterLink({
fallbackTag: 'button'
});
const style = vue.computed(() => {
const obj = props.fab === false && props.fabMini === false
? sizeStyle.value
: {};
return props.padding !== void 0
? Object.assign({}, obj, {
padding: props.padding
.split(/\s+/)
.map(v => (v in btnPadding ? btnPadding[ v ] + 'px' : v))
.join(' '),
minWidth: '0',
minHeight: '0'
})
: obj
});
const isRounded = vue.computed(() =>
props.rounded === true || props.fab === true || props.fabMini === true
);
const isActionable = vue.computed(() =>
props.disable !== true && props.loading !== true
);
const tabIndex = vue.computed(() => (
isActionable.value === true ? props.tabindex || 0 : -1
));
const design = vue.computed(() => getBtnDesign(props, 'standard'));
const attributes = vue.computed(() => {
const acc = { tabindex: tabIndex.value };
if (hasLink.value === true) {
Object.assign(acc, linkAttrs.value);
}
else if (formTypes.includes(props.type) === true) {
acc.type = props.type;
}
if (linkTag.value === 'a') {
if (props.disable === true) {
acc[ 'aria-disabled' ] = 'true';
}
else if (acc.href === void 0) {
acc.role = 'button';
}
if (hasRouterLink.value !== true && mediaTypeRE.test(props.type) === true) {
acc.type = props.type;
}
}
else if (props.disable === true) {
acc.disabled = '';
acc[ 'aria-disabled' ] = 'true';
}
if (props.loading === true && props.percentage !== void 0) {
Object.assign(acc, {
role: 'progressbar',
'aria-valuemin': 0,
'aria-valuemax': 100,
'aria-valuenow': props.percentage
});
}
return acc
});
const classes = vue.computed(() => {
let colors;
if (props.color !== void 0) {
if (props.flat === true || props.outline === true) {
colors = `text-${ props.textColor || props.color }`;
}
else {
colors = `bg-${ props.color } text-${ props.textColor || 'white' }`;
}
}
else if (props.textColor) {
colors = `text-${ props.textColor }`;
}
const shape = props.round === true
? 'round'
: `rectangle${ isRounded.value === true ? ' q-btn--rounded' : (props.square === true ? ' q-btn--square' : '') }`;
return `q-btn--${ design.value } q-btn--${ shape }`
+ (colors !== void 0 ? ' ' + colors : '')
+ (isActionable.value === true ? ' q-btn--actionable q-focusable q-hoverable' : (props.disable === true ? ' disabled' : ''))
+ (props.fab === true ? ' q-btn--fab' : (props.fabMini === true ? ' q-btn--fab-mini' : ''))
+ (props.noCaps === true ? ' q-btn--no-uppercase' : '')
+ (props.dense === true ? ' q-btn--dense' : '')
+ (props.stretch === true ? ' no-border-radius self-stretch' : '')
+ (props.glossy === true ? ' glossy' : '')
+ (props.square ? ' q-btn--square' : '')
});
const innerClasses = vue.computed(() =>
alignClass.value + (props.stack === true ? ' column' : ' row')
+ (props.noWrap === true ? ' no-wrap text-no-wrap' : '')
+ (props.loading === true ? ' q-btn__content--hidden' : '')
);
return {
classes,
style,
innerClasses,
attributes,
hasLink,
linkTag,
navigateOnClick,
isActionable
}
}
const { passiveCapture } = listenOpts;
let
touchTarget = null,
keyboardTarget = null,
mouseTarget = null;
var QBtn = createComponent({
name: 'QBtn',
props: {
...useBtnProps,
percentage: Number,
darkPercentage: Boolean,
onTouchstart: [ Function, Array ]
},
emits: [ 'click', 'keydown', 'mousedown', 'keyup' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const {
classes, style, innerClasses,
attributes,
hasLink, linkTag, navigateOnClick,
isActionable
} = useBtn(props);
const rootRef = vue.ref(null);
const blurTargetRef = vue.ref(null);
let localTouchTargetEl = null, avoidMouseRipple, mouseTimer = null;
const hasLabel = vue.computed(() =>
props.label !== void 0 && props.label !== null && props.label !== ''
);
const ripple = vue.computed(() => (
props.disable === true || props.ripple === false
? false
: {
keyCodes: hasLink.value === true ? [ 13, 32 ] : [ 13 ],
...(props.ripple === true ? {} : props.ripple)
}
));
const rippleProps = vue.computed(() => ({ center: props.round }));
const percentageStyle = vue.computed(() => {
const val = Math.max(0, Math.min(100, props.percentage));
return val > 0
? { transition: 'transform 0.6s', transform: `translateX(${ val - 100 }%)` }
: {}
});
const onEvents = vue.computed(() => {
if (props.loading === true) {
return {
onMousedown: onLoadingEvt,
onTouchstart: onLoadingEvt,
onClick: onLoadingEvt,
onKeydown: onLoadingEvt,
onKeyup: onLoadingEvt
}
}
if (isActionable.value === true) {
const acc = {
onClick,
onKeydown,
onMousedown
};
if (proxy.$q.platform.has.touch === true) {
const suffix = props.onTouchstart !== void 0
? ''
: 'Passive';
acc[ `onTouchstart${ suffix }` ] = onTouchstart;
}
return acc
}
return {
// needed; especially for disabled <a> tags
onClick: stopAndPrevent
}
});
const nodeProps = vue.computed(() => ({
ref: rootRef,
class: 'q-btn q-btn-item non-selectable no-outline ' + classes.value,
style: style.value,
...attributes.value,
...onEvents.value
}));
function onClick (e) {
// is it already destroyed?
if (rootRef.value === null) { return }
if (e !== void 0) {
if (e.defaultPrevented === true) {
return
}
const el = document.activeElement;
// focus button if it came from ENTER on form
// prevent the new submit (already done)
if (
props.type === 'submit'
&& el !== document.body
&& rootRef.value.contains(el) === false
// required for iOS and desktop Safari
&& el.contains(rootRef.value) === false
) {
rootRef.value.focus();
const onClickCleanup = () => {
document.removeEventListener('keydown', stopAndPrevent, true);
document.removeEventListener('keyup', onClickCleanup, passiveCapture);
rootRef.value !== null && rootRef.value.removeEventListener('blur', onClickCleanup, passiveCapture);
};
document.addEventListener('keydown', stopAndPrevent, true);
document.addEventListener('keyup', onClickCleanup, passiveCapture);
rootRef.value.addEventListener('blur', onClickCleanup, passiveCapture);
}
}
navigateOnClick(e);
}
function onKeydown (e) {
// is it already destroyed?
if (rootRef.value === null) { return }
emit('keydown', e);
if (isKeyCode(e, [ 13, 32 ]) === true && keyboardTarget !== rootRef.value) {
keyboardTarget !== null && cleanup();
if (e.defaultPrevented !== true) {
// focus external button if the focus helper was focused before
rootRef.value.focus();
keyboardTarget = rootRef.value;
rootRef.value.classList.add('q-btn--active');
document.addEventListener('keyup', onPressEnd, true);
rootRef.value.addEventListener('blur', onPressEnd, passiveCapture);
}
stopAndPrevent(e);
}
}
function onTouchstart (e) {
// is it already destroyed?
if (rootRef.value === null) { return }
emit('touchstart', e);
if (e.defaultPrevented === true) { return }
if (touchTarget !== rootRef.value) {
touchTarget !== null && cleanup();
touchTarget = rootRef.value;
localTouchTargetEl = e.target;
localTouchTargetEl.addEventListener('touchcancel', onPressEnd, passiveCapture);
localTouchTargetEl.addEventListener('touchend', onPressEnd, passiveCapture);
}
// avoid duplicated mousedown event
// triggering another early ripple
avoidMouseRipple = true;
mouseTimer !== null && clearTimeout(mouseTimer);
mouseTimer = setTimeout(() => {
mouseTimer = null;
avoidMouseRipple = false;
}, 200);
}
function onMousedown (e) {
// is it already destroyed?
if (rootRef.value === null) { return }
e.qSkipRipple = avoidMouseRipple === true;
emit('mousedown', e);
if (e.defaultPrevented !== true && mouseTarget !== rootRef.value) {
mouseTarget !== null && cleanup();
mouseTarget = rootRef.value;
rootRef.value.classList.add('q-btn--active');
document.addEventListener('mouseup', onPressEnd, passiveCapture);
}
}
function onPressEnd (e) {
// is it already destroyed?
if (rootRef.value === null) { return }
// needed for IE (because it emits blur when focusing button from focus helper)
if (e !== void 0 && e.type === 'blur' && document.activeElement === rootRef.value) {
return
}
if (e !== void 0 && e.type === 'keyup') {
if (keyboardTarget === rootRef.value && isKeyCode(e, [ 13, 32 ]) === true) {
// for click trigger
const evt = new MouseEvent('click', e);
evt.qKeyEvent = true;
e.defaultPrevented === true && prevent(evt);
e.cancelBubble === true && stop(evt);
rootRef.value.dispatchEvent(evt);
stopAndPrevent(e);
// for ripple
e.qKeyEvent = true;
}
emit('keyup', e);
}
cleanup();
}
function cleanup (destroying) {
const blurTarget = blurTargetRef.value;
if (
destroying !== true
&& (touchTarget === rootRef.value || mouseTarget === rootRef.value)
&& blurTarget !== null
&& blurTarget !== document.activeElement
) {
blurTarget.setAttribute('tabindex', -1);
blurTarget.focus();
}
if (touchTarget === rootRef.value) {
if (localTouchTargetEl !== null) {
localTouchTargetEl.removeEventListener('touchcancel', onPressEnd, passiveCapture);
localTouchTargetEl.removeEventListener('touchend', onPressEnd, passiveCapture);
}
touchTarget = localTouchTargetEl = null;
}
if (mouseTarget === rootRef.value) {
document.removeEventListener('mouseup', onPressEnd, passiveCapture);
mouseTarget = null;
}
if (keyboardTarget === rootRef.value) {
document.removeEventListener('keyup', onPressEnd, true);
rootRef.value !== null && rootRef.value.removeEventListener('blur', onPressEnd, passiveCapture);
keyboardTarget = null;
}
rootRef.value !== null && rootRef.value.classList.remove('q-btn--active');
}
function onLoadingEvt (evt) {
stopAndPrevent(evt);
evt.qSkipRipple = true;
}
vue.onBeforeUnmount(() => {
cleanup(true);
});
// expose public methods
Object.assign(proxy, { click: onClick });
return () => {
let inner = [];
props.icon !== void 0 && inner.push(
vue.h(QIcon, {
name: props.icon,
left: props.stack === false && hasLabel.value === true,
role: 'img',
'aria-hidden': 'true'
})
);
hasLabel.value === true && inner.push(
vue.h('span', { class: 'block' }, [ props.label ])
);
inner = hMergeSlot(slots.default, inner);
if (props.iconRight !== void 0 && props.round === false) {
inner.push(
vue.h(QIcon, {
name: props.iconRight,
right: props.stack === false && hasLabel.value === true,
role: 'img',
'aria-hidden': 'true'
})
);
}
const child = [
vue.h('span', {
class: 'q-focus-helper',
ref: blurTargetRef
})
];
if (props.loading === true && props.percentage !== void 0) {
child.push(
vue.h('span', {
class: 'q-btn__progress absolute-full overflow-hidden' + (props.darkPercentage === true ? ' q-btn__progress--dark' : '')
}, [
vue.h('span', {
class: 'q-btn__progress-indicator fit block',
style: percentageStyle.value
})
])
);
}
child.push(
vue.h('span', {
class: 'q-btn__content text-center col items-center q-anchor--skip ' + innerClasses.value
}, inner)
);
props.loading !== null && child.push(
vue.h(vue.Transition, {
name: 'q-transition--fade'
}, () => (
props.loading === true
? [
vue.h('span', {
key: 'loading',
class: 'absolute-full flex flex-center'
}, slots.loading !== void 0 ? slots.loading() : [ vue.h(QSpinner) ])
]
: null
))
);
return vue.withDirectives(
vue.h(
linkTag.value,
nodeProps.value,
child
),
[ [
Ripple,
ripple.value,
void 0,
rippleProps.value
] ]
)
}
}
});
var QBtnGroup = createComponent({
name: 'QBtnGroup',
props: {
unelevated: Boolean,
outline: Boolean,
flat: Boolean,
rounded: Boolean,
square: Boolean,
push: Boolean,
stretch: Boolean,
glossy: Boolean,
spread: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() => {
const cls = [ 'unelevated', 'outline', 'flat', 'rounded', 'square', 'push', 'stretch', 'glossy' ]
.filter(t => props[ t ] === true)
.map(t => `q-btn-group--${ t }`).join(' ');
return `q-btn-group row no-wrap${ cls.length !== 0 ? ' ' + cls : '' }`
+ (props.spread === true ? ' q-btn-group--spread' : ' inline')
});
return () => vue.h('div', { class: classes.value }, hSlot(slots.default))
}
});
function clearSelection () {
if (window.getSelection !== void 0) {
const selection = window.getSelection();
if (selection.empty !== void 0) {
selection.empty();
}
else if (selection.removeAllRanges !== void 0) {
selection.removeAllRanges();
Platform.is.mobile !== true && selection.addRange(document.createRange());
}
}
else if (document.selection !== void 0) {
document.selection.empty();
}
}
const useAnchorProps = {
target: {
default: true
},
noParentEvent: Boolean,
contextMenu: Boolean
};
function useAnchor ({
showing,
avoidEmit, // required for QPopupProxy (true)
configureAnchorEl // optional
}) {
const { props, proxy, emit } = vue.getCurrentInstance();
const anchorEl = vue.ref(null);
let touchTimer = null;
function canShow (evt) {
// abort with no parent configured or on multi-touch
return anchorEl.value === null
? false
: (evt === void 0 || evt.touches === void 0 || evt.touches.length <= 1)
}
const anchorEvents = {};
if (configureAnchorEl === void 0) {
// default configureAnchorEl is designed for
// QMenu & QPopupProxy (which is why it's handled here)
Object.assign(anchorEvents, {
hide (evt) {
proxy.hide(evt);
},
toggle (evt) {
proxy.toggle(evt);
evt.qAnchorHandled = true;
},
toggleKey (evt) {
isKeyCode(evt, 13) === true && anchorEvents.toggle(evt);
},
contextClick (evt) {
proxy.hide(evt);
prevent(evt);
vue.nextTick(() => {
proxy.show(evt);
evt.qAnchorHandled = true;
});
},
prevent,
mobileTouch (evt) {
anchorEvents.mobileCleanup(evt);
if (canShow(evt) !== true) {
return
}
proxy.hide(evt);
anchorEl.value.classList.add('non-selectable');
const target = evt.target;
addEvt(anchorEvents, 'anchor', [
[ target, 'touchmove', 'mobileCleanup', 'passive' ],
[ target, 'touchend', 'mobileCleanup', 'passive' ],
[ target, 'touchcancel', 'mobileCleanup', 'passive' ],
[ anchorEl.value, 'contextmenu', 'prevent', 'notPassive' ]
]);
touchTimer = setTimeout(() => {
touchTimer = null;
proxy.show(evt);
evt.qAnchorHandled = true;
}, 300);
},
mobileCleanup (evt) {
anchorEl.value.classList.remove('non-selectable');
if (touchTimer !== null) {
clearTimeout(touchTimer);
touchTimer = null;
}
if (showing.value === true && evt !== void 0) {
clearSelection();
}
}
});
configureAnchorEl = function (context = props.contextMenu) {
if (props.noParentEvent === true || anchorEl.value === null) { return }
let evts;
if (context === true) {
if (proxy.$q.platform.is.mobile === true) {
evts = [
[ anchorEl.value, 'touchstart', 'mobileTouch', 'passive' ]
];
}
else {
evts = [
[ anchorEl.value, 'mousedown', 'hide', 'passive' ],
[ anchorEl.value, 'contextmenu', 'contextClick', 'notPassive' ]
];
}
}
else {
evts = [
[ anchorEl.value, 'click', 'toggle', 'passive' ],
[ anchorEl.value, 'keyup', 'toggleKey', 'passive' ]
];
}
addEvt(anchorEvents, 'anchor', evts);
};
}
function unconfigureAnchorEl () {
cleanEvt(anchorEvents, 'anchor');
}
function setAnchorEl (el) {
anchorEl.value = el;
while (anchorEl.value.classList.contains('q-anchor--skip')) {
anchorEl.value = anchorEl.value.parentNode;
}
configureAnchorEl();
}
function pickAnchorEl () {
if (props.target === false || props.target === '' || proxy.$el.parentNode === null) {
anchorEl.value = null;
}
else if (props.target === true) {
setAnchorEl(proxy.$el.parentNode);
}
else {
let el = props.target;
if (typeof props.target === 'string') {
try {
el = document.querySelector(props.target);
}
catch (err) {
el = void 0;
}
}
if (el !== void 0 && el !== null) {
anchorEl.value = el.$el || el;
configureAnchorEl();
}
else {
anchorEl.value = null;
console.error(`Anchor: target "${ props.target }" not found`);
}
}
}
vue.watch(() => props.contextMenu, val => {
if (anchorEl.value !== null) {
unconfigureAnchorEl();
configureAnchorEl(val);
}
});
vue.watch(() => props.target, () => {
if (anchorEl.value !== null) {
unconfigureAnchorEl();
}
pickAnchorEl();
});
vue.watch(() => props.noParentEvent, val => {
if (anchorEl.value !== null) {
if (val === true) {
unconfigureAnchorEl();
}
else {
configureAnchorEl();
}
}
});
vue.onMounted(() => {
pickAnchorEl();
if (avoidEmit !== true && props.modelValue === true && anchorEl.value === null) {
emit('update:modelValue', false);
}
});
vue.onBeforeUnmount(() => {
touchTimer !== null && clearTimeout(touchTimer);
unconfigureAnchorEl();
});
return {
anchorEl,
canShow,
anchorEvents
}
}
function useScrollTarget (
props,
configureScrollTarget
) {
const localScrollTarget = vue.ref(null);
let scrollFn;
function changeScrollEvent (scrollTarget, fn) {
const fnProp = `${ fn !== void 0 ? 'add' : 'remove' }EventListener`;
const fnHandler = fn !== void 0 ? fn : scrollFn;
if (scrollTarget !== window) {
scrollTarget[ fnProp ]('scroll', fnHandler, listenOpts.passive);
}
window[ fnProp ]('scroll', fnHandler, listenOpts.passive);
scrollFn = fn;
}
function unconfigureScrollTarget () {
if (localScrollTarget.value !== null) {
changeScrollEvent(localScrollTarget.value);
localScrollTarget.value = null;
}
}
const noParentEventWatcher = vue.watch(() => props.noParentEvent, () => {
if (localScrollTarget.value !== null) {
unconfigureScrollTarget();
configureScrollTarget();
}
});
vue.onBeforeUnmount(noParentEventWatcher);
return {
localScrollTarget,
unconfigureScrollTarget,
changeScrollEvent
}
}
const useModelToggleProps = {
modelValue: {
type: Boolean,
default: null
},
'onUpdate:modelValue': [ Function, Array ]
};
const useModelToggleEmits = [
'beforeShow', 'show', 'beforeHide', 'hide'
];
// handleShow/handleHide -> removeTick(), self (& emit show)
function useModelToggle ({
showing,
canShow, // optional
hideOnRouteChange, // optional
handleShow, // optional
handleHide, // optional
processOnMount // optional
}) {
const vm = vue.getCurrentInstance();
const { props, emit, proxy } = vm;
let payload;
function toggle (evt) {
if (showing.value === true) {
hide(evt);
}
else {
show(evt);
}
}
function show (evt) {
if (
props.disable === true
|| (evt !== void 0 && evt.qAnchorHandled === true)
|| (canShow !== void 0 && canShow(evt) !== true)
) {
return
}
const listener = props[ 'onUpdate:modelValue' ] !== void 0;
if (listener === true && false !== true) {
emit('update:modelValue', true);
payload = evt;
vue.nextTick(() => {
if (payload === evt) {
payload = void 0;
}
});
}
if (props.modelValue === null || listener === false || false) {
processShow(evt);
}
}
function processShow (evt) {
if (showing.value === true) {
return
}
showing.value = true;
emit('beforeShow', evt);
if (handleShow !== void 0) {
handleShow(evt);
}
else {
emit('show', evt);
}
}
function hide (evt) {
if (props.disable === true) {
return
}
const listener = props[ 'onUpdate:modelValue' ] !== void 0;
if (listener === true && false !== true) {
emit('update:modelValue', false);
payload = evt;
vue.nextTick(() => {
if (payload === evt) {
payload = void 0;
}
});
}
if (props.modelValue === null || listener === false || false) {
processHide(evt);
}
}
function processHide (evt) {
if (showing.value === false) {
return
}
showing.value = false;
emit('beforeHide', evt);
if (handleHide !== void 0) {
handleHide(evt);
}
else {
emit('hide', evt);
}
}
function processModelChange (val) {
if (props.disable === true && val === true) {
if (props[ 'onUpdate:modelValue' ] !== void 0) {
emit('update:modelValue', false);
}
}
else if ((val === true) !== showing.value) {
const fn = val === true ? processShow : processHide;
fn(payload);
}
}
vue.watch(() => props.modelValue, processModelChange);
if (hideOnRouteChange !== void 0 && vmHasRouter(vm) === true) {
vue.watch(() => proxy.$route.fullPath, () => {
if (hideOnRouteChange.value === true && showing.value === true) {
hide();
}
});
}
processOnMount === true && vue.onMounted(() => {
processModelChange(props.modelValue);
});
// expose public methods
const publicMethods = { show, hide, toggle };
Object.assign(proxy, publicMethods);
return publicMethods
}
let queue = [];
let waitFlags = [];
function clearFlag (flag) {
waitFlags = waitFlags.filter(entry => entry !== flag);
}
function addFocusWaitFlag (flag) {
clearFlag(flag);
waitFlags.push(flag);
}
function removeFocusWaitFlag (flag) {
clearFlag(flag);
if (waitFlags.length === 0 && queue.length !== 0) {
// only call last focus handler (can't focus multiple things at once)
queue[ queue.length - 1 ]();
queue = [];
}
}
function addFocusFn (fn) {
if (waitFlags.length === 0) {
fn();
}
else {
queue.push(fn);
}
}
function removeFocusFn (fn) {
queue = queue.filter(entry => entry !== fn);
}
const nodesList = [];
const portalTypeList = [];
let portalIndex = 1;
let target = document.body;
function createGlobalNode (id, portalType) {
const el = document.createElement('div');
el.id = portalType !== void 0
? `q-portal--${ portalType }--${ portalIndex++ }`
: id;
if (globalConfig.globalNodes !== void 0) {
const cls = globalConfig.globalNodes.class;
if (cls !== void 0) {
el.className = cls;
}
}
target.appendChild(el);
nodesList.push(el);
portalTypeList.push(portalType);
return el
}
function removeGlobalNode (el) {
const nodeIndex = nodesList.indexOf(el);
nodesList.splice(nodeIndex, 1);
portalTypeList.splice(nodeIndex, 1);
el.remove();
}
function changeGlobalNodesTarget (newTarget) {
if (newTarget === target) {
return
}
target = newTarget;
if (
target === document.body
// or we have less than 2 dialogs:
|| portalTypeList.reduce((acc, type) => (type === 'dialog' ? acc + 1 : acc), 0) < 2
) {
nodesList.forEach(node => {
if (node.contains(target) === false) {
target.appendChild(node);
}
});
return
}
const lastDialogIndex = portalTypeList.lastIndexOf('dialog');
for (let i = 0; i < nodesList.length; i++) {
const el = nodesList[ i ];
if (
(i === lastDialogIndex || portalTypeList[ i ] !== 'dialog')
&& el.contains(target) === false
) {
target.appendChild(el);
}
}
}
const portalProxyList = [];
function getPortalProxy (el) {
return portalProxyList.find(proxy =>
proxy.contentEl !== null
&& proxy.contentEl.contains(el)
)
}
function closePortalMenus (proxy, evt) {
do {
if (proxy.$options.name === 'QMenu') {
proxy.hide(evt);
// is this a point of separation?
if (proxy.$props.separateClosePopup === true) {
return getParentProxy(proxy)
}
}
else if (proxy.__qPortal === true) {
// treat it as point of separation if parent is QPopupProxy
// (so mobile matches desktop behavior)
// and hide it too
const parent = getParentProxy(proxy);
if (parent !== void 0 && parent.$options.name === 'QPopupProxy') {
proxy.hide(evt);
return parent
}
else {
return proxy
}
}
proxy = getParentProxy(proxy);
} while (proxy !== void 0 && proxy !== null)
}
function closePortals (proxy, evt, depth) {
while (depth !== 0 && proxy !== void 0 && proxy !== null) {
if (proxy.__qPortal === true) {
depth--;
if (proxy.$options.name === 'QMenu') {
proxy = closePortalMenus(proxy, evt);
continue
}
proxy.hide(evt);
}
proxy = getParentProxy(proxy);
}
}
function isOnGlobalDialog (vm) {
vm = vm.parent;
while (vm !== void 0 && vm !== null) {
if (vm.type.name === 'QGlobalDialog') {
return true
}
if (vm.type.name === 'QDialog' || vm.type.name === 'QMenu') {
return false
}
vm = vm.parent;
}
return false
}
// Warning!
// You MUST specify "inheritAttrs: false" in your component
function usePortal (vm, innerRef, renderPortalContent, type) {
// showing, including while in show/hide transition
const portalIsActive = vue.ref(false);
// showing & not in any show/hide transition
const portalIsAccessible = vue.ref(false);
let portalEl = null;
const focusObj = {};
const onGlobalDialog = type === 'dialog' && isOnGlobalDialog(vm);
function showPortal (isReady) {
if (isReady === true) {
removeFocusWaitFlag(focusObj);
portalIsAccessible.value = true;
return
}
portalIsAccessible.value = false;
if (portalIsActive.value === false) {
if (onGlobalDialog === false && portalEl === null) {
portalEl = createGlobalNode(false, type);
}
portalIsActive.value = true;
// register portal
portalProxyList.push(vm.proxy);
addFocusWaitFlag(focusObj);
}
}
function hidePortal (isReady) {
portalIsAccessible.value = false;
if (isReady !== true) { return }
removeFocusWaitFlag(focusObj);
portalIsActive.value = false;
// unregister portal
const index = portalProxyList.indexOf(vm.proxy);
if (index !== -1) {
portalProxyList.splice(index, 1);
}
if (portalEl !== null) {
removeGlobalNode(portalEl);
portalEl = null;
}
}
vue.onUnmounted(() => { hidePortal(true); });
// needed for portal vm detection
vm.proxy.__qPortal = true;
// public way of accessing the rendered content
injectProp(vm.proxy, 'contentEl', () => innerRef.value);
return {
showPortal,
hidePortal,
portalIsActive,
portalIsAccessible,
renderPortal: () => (
onGlobalDialog === true
? renderPortalContent()
: (
portalIsActive.value === true
? [ vue.h(vue.Teleport, { to: portalEl }, renderPortalContent()) ]
: void 0
)
)
}
}
const useTransitionProps = {
transitionShow: {
type: String,
default: 'fade'
},
transitionHide: {
type: String,
default: 'fade'
},
transitionDuration: {
type: [ String, Number ],
default: 300
}
};
function useTransition (props, defaultShowFn = () => {}, defaultHideFn = () => {}) {
return {
transitionProps: vue.computed(() => {
const show = `q-transition--${ props.transitionShow || defaultShowFn() }`;
const hide = `q-transition--${ props.transitionHide || defaultHideFn() }`;
return {
appear: true,
enterFromClass: `${ show }-enter-from`,
enterActiveClass: `${ show }-enter-active`,
enterToClass: `${ show }-enter-to`,
leaveFromClass: `${ hide }-leave-from`,
leaveActiveClass: `${ hide }-leave-active`,
leaveToClass: `${ hide }-leave-to`
}
}),
transitionStyle: vue.computed(() => `--q-transition-duration: ${ props.transitionDuration }ms`)
}
}
/*
* Usage:
* registerTick(fn)
* removeTick()
*/
function useTick () {
let tickFn;
const vm = vue.getCurrentInstance();
function removeTick () {
tickFn = void 0;
}
vue.onDeactivated(removeTick);
vue.onBeforeUnmount(removeTick);
return {
removeTick,
registerTick (fn) {
tickFn = fn;
vue.nextTick(() => {
if (tickFn === fn) {
// we also check if VM is destroyed, since if it
// got to trigger one nextTick() we cannot stop it
vmIsDestroyed(vm) === false && tickFn();
tickFn = void 0;
}
});
}
}
}
/*
* Usage:
* registerTimeout(fn[, delay])
* removeTimeout()
*/
function useTimeout () {
let timer = null;
const vm = vue.getCurrentInstance();
function removeTimeout () {
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
}
vue.onDeactivated(removeTimeout);
vue.onBeforeUnmount(removeTimeout);
return {
removeTimeout,
registerTimeout (fn, delay) {
removeTimeout();
if (vmIsDestroyed(vm) === false) {
timer = setTimeout(fn, delay);
}
}
}
}
const scrollTargets = [ null, document, document.body, document.scrollingElement, document.documentElement ];
function getScrollTarget (el, targetEl) {
let target = getElement$1(targetEl);
if (target === void 0) {
if (el === void 0 || el === null) {
return window
}
target = el.closest('.scroll,.scroll-y,.overflow-auto');
}
return scrollTargets.includes(target)
? window
: target
}
function getScrollHeight (el) {
return (el === window ? document.body : el).scrollHeight
}
function getScrollWidth (el) {
return (el === window ? document.body : el).scrollWidth
}
function getVerticalScrollPosition (scrollTarget) {
return scrollTarget === window
? window.pageYOffset || window.scrollY || document.body.scrollTop || 0
: scrollTarget.scrollTop
}
function getHorizontalScrollPosition (scrollTarget) {
return scrollTarget === window
? window.pageXOffset || window.scrollX || document.body.scrollLeft || 0
: scrollTarget.scrollLeft
}
function animVerticalScrollTo (el, to, duration = 0 /* , prevTime */) {
const prevTime = arguments[ 3 ] === void 0 ? performance.now() : arguments[ 3 ];
const pos = getVerticalScrollPosition(el);
if (duration <= 0) {
if (pos !== to) {
setScroll$1(el, to);
}
return
}
requestAnimationFrame(nowTime => {
const frameTime = nowTime - prevTime;
const newPos = pos + (to - pos) / Math.max(frameTime, duration) * frameTime;
setScroll$1(el, newPos);
if (newPos !== to) {
animVerticalScrollTo(el, to, duration - frameTime, nowTime);
}
});
}
function animHorizontalScrollTo (el, to, duration = 0 /* , prevTime */) {
const prevTime = arguments[ 3 ] === void 0 ? performance.now() : arguments[ 3 ];
const pos = getHorizontalScrollPosition(el);
if (duration <= 0) {
if (pos !== to) {
setHorizontalScroll(el, to);
}
return
}
requestAnimationFrame(nowTime => {
const frameTime = nowTime - prevTime;
const newPos = pos + (to - pos) / Math.max(frameTime, duration) * frameTime;
setHorizontalScroll(el, newPos);
if (newPos !== to) {
animHorizontalScrollTo(el, to, duration - frameTime, nowTime);
}
});
}
function setScroll$1 (scrollTarget, offset) {
if (scrollTarget === window) {
window.scrollTo(window.pageXOffset || window.scrollX || document.body.scrollLeft || 0, offset);
return
}
scrollTarget.scrollTop = offset;
}
function setHorizontalScroll (scrollTarget, offset) {
if (scrollTarget === window) {
window.scrollTo(offset, window.pageYOffset || window.scrollY || document.body.scrollTop || 0);
return
}
scrollTarget.scrollLeft = offset;
}
function setVerticalScrollPosition (scrollTarget, offset, duration) {
if (duration) {
animVerticalScrollTo(scrollTarget, offset, duration);
return
}
setScroll$1(scrollTarget, offset);
}
function setHorizontalScrollPosition (scrollTarget, offset, duration) {
if (duration) {
animHorizontalScrollTo(scrollTarget, offset, duration);
return
}
setHorizontalScroll(scrollTarget, offset);
}
let size;
function getScrollbarWidth () {
if (size !== undefined) {
return size
}
const
inner = document.createElement('p'),
outer = document.createElement('div');
css(inner, {
width: '100%',
height: '200px'
});
css(outer, {
position: 'absolute',
top: '0px',
left: '0px',
visibility: 'hidden',
width: '200px',
height: '150px',
overflow: 'hidden'
});
outer.appendChild(inner);
document.body.appendChild(outer);
const w1 = inner.offsetWidth;
outer.style.overflow = 'scroll';
let w2 = inner.offsetWidth;
if (w1 === w2) {
w2 = outer.clientWidth;
}
outer.remove();
size = w1 - w2;
return size
}
function hasScrollbar (el, onY = true) {
if (!el || el.nodeType !== Node.ELEMENT_NODE) {
return false
}
return onY
? (
el.scrollHeight > el.clientHeight && (
el.classList.contains('scroll')
|| el.classList.contains('overflow-auto')
|| [ 'auto', 'scroll' ].includes(window.getComputedStyle(el)[ 'overflow-y' ])
)
)
: (
el.scrollWidth > el.clientWidth && (
el.classList.contains('scroll')
|| el.classList.contains('overflow-auto')
|| [ 'auto', 'scroll' ].includes(window.getComputedStyle(el)[ 'overflow-x' ])
)
)
}
var scroll = {
getScrollTarget,
getScrollHeight,
getScrollWidth,
getVerticalScrollPosition,
getHorizontalScrollPosition,
animVerticalScrollTo,
animHorizontalScrollTo,
setVerticalScrollPosition,
setHorizontalScrollPosition,
getScrollbarWidth,
hasScrollbar
};
const handlers$1 = [];
let escDown;
function onKeydown (evt) {
escDown = evt.keyCode === 27;
}
function onBlur () {
if (escDown === true) {
escDown = false;
}
}
function onKeyup (evt) {
if (escDown === true) {
escDown = false;
if (isKeyCode(evt, 27) === true) {
handlers$1[ handlers$1.length - 1 ](evt);
}
}
}
function update$4 (action) {
window[ action ]('keydown', onKeydown);
window[ action ]('blur', onBlur);
window[ action ]('keyup', onKeyup);
escDown = false;
}
function addEscapeKey (fn) {
if (client.is.desktop === true) {
handlers$1.push(fn);
if (handlers$1.length === 1) {
update$4('addEventListener');
}
}
}
function removeEscapeKey (fn) {
const index = handlers$1.indexOf(fn);
if (index > -1) {
handlers$1.splice(index, 1);
if (handlers$1.length === 0) {
update$4('removeEventListener');
}
}
}
const handlers = [];
function trigger$1 (e) {
handlers[ handlers.length - 1 ](e);
}
function addFocusout (fn) {
if (client.is.desktop === true) {
handlers.push(fn);
if (handlers.length === 1) {
document.body.addEventListener('focusin', trigger$1);
}
}
}
function removeFocusout (fn) {
const index = handlers.indexOf(fn);
if (index > -1) {
handlers.splice(index, 1);
if (handlers.length === 0) {
document.body.removeEventListener('focusin', trigger$1);
}
}
}
const
{ notPassiveCapture } = listenOpts,
registeredList = [];
function globalHandler (evt) {
const target = evt.target;
if (
target === void 0
|| target.nodeType === 8
|| target.classList.contains('no-pointer-events') === true
) {
return
}
// check last portal vm if it's
// a QDialog and not in seamless mode
let portalIndex = portalProxyList.length - 1;
while (portalIndex >= 0) {
const proxy = portalProxyList[ portalIndex ].$;
// skip QTooltip portals
if (proxy.type.name === 'QTooltip') {
portalIndex--;
continue
}
if (proxy.type.name !== 'QDialog') {
break
}
if (proxy.props.seamless !== true) {
return
}
portalIndex--;
}
for (let i = registeredList.length - 1; i >= 0; i--) {
const state = registeredList[ i ];
if (
(
state.anchorEl.value === null
|| state.anchorEl.value.contains(target) === false
)
&& (
target === document.body
|| (
state.innerRef.value !== null
&& state.innerRef.value.contains(target) === false
)
)
) {
// mark the event as being processed by clickOutside
// used to prevent refocus after menu close
evt.qClickOutside = true;
state.onClickOutside(evt);
}
else {
return
}
}
}
function addClickOutside (clickOutsideProps) {
registeredList.push(clickOutsideProps);
if (registeredList.length === 1) {
document.addEventListener('mousedown', globalHandler, notPassiveCapture);
document.addEventListener('touchstart', globalHandler, notPassiveCapture);
}
}
function removeClickOutside (clickOutsideProps) {
const index = registeredList.findIndex(h => h === clickOutsideProps);
if (index > -1) {
registeredList.splice(index, 1);
if (registeredList.length === 0) {
document.removeEventListener('mousedown', globalHandler, notPassiveCapture);
document.removeEventListener('touchstart', globalHandler, notPassiveCapture);
}
}
}
let vpLeft, vpTop;
function validatePosition (pos) {
const parts = pos.split(' ');
if (parts.length !== 2) {
return false
}
if ([ 'top', 'center', 'bottom' ].includes(parts[ 0 ]) !== true) {
console.error('Anchor/Self position must start with one of top/center/bottom');
return false
}
if ([ 'left', 'middle', 'right', 'start', 'end' ].includes(parts[ 1 ]) !== true) {
console.error('Anchor/Self position must end with one of left/middle/right/start/end');
return false
}
return true
}
function validateOffset (val) {
if (!val) { return true }
if (val.length !== 2) { return false }
if (typeof val[ 0 ] !== 'number' || typeof val[ 1 ] !== 'number') {
return false
}
return true
}
const horizontalPos = {
'start#ltr': 'left',
'start#rtl': 'right',
'end#ltr': 'right',
'end#rtl': 'left'
}
;[ 'left', 'middle', 'right' ].forEach(pos => {
horizontalPos[ `${ pos }#ltr` ] = pos;
horizontalPos[ `${ pos }#rtl` ] = pos;
});
function parsePosition (pos, rtl) {
const parts = pos.split(' ');
return {
vertical: parts[ 0 ],
horizontal: horizontalPos[ `${ parts[ 1 ] }#${ rtl === true ? 'rtl' : 'ltr' }` ]
}
}
function getAnchorProps (el, offset) {
let { top, left, right, bottom, width, height } = el.getBoundingClientRect();
if (offset !== void 0) {
top -= offset[ 1 ];
left -= offset[ 0 ];
bottom += offset[ 1 ];
right += offset[ 0 ];
width += offset[ 0 ];
height += offset[ 1 ];
}
return {
top, bottom, height,
left, right, width,
middle: left + (right - left) / 2,
center: top + (bottom - top) / 2
}
}
function getAbsoluteAnchorProps (el, absoluteOffset, offset) {
let { top, left } = el.getBoundingClientRect();
top += absoluteOffset.top;
left += absoluteOffset.left;
if (offset !== void 0) {
top += offset[ 1 ];
left += offset[ 0 ];
}
return {
top, bottom: top + 1, height: 1,
left, right: left + 1, width: 1,
middle: left,
center: top
}
}
function getTargetProps (width, height) {
return {
top: 0,
center: height / 2,
bottom: height,
left: 0,
middle: width / 2,
right: width
}
}
function getTopLeftProps (anchorProps, targetProps, anchorOrigin, selfOrigin) {
return {
top: anchorProps[ anchorOrigin.vertical ] - targetProps[ selfOrigin.vertical ],
left: anchorProps[ anchorOrigin.horizontal ] - targetProps[ selfOrigin.horizontal ]
}
}
function setPosition (cfg, retryNumber = 0) {
if (
cfg.targetEl === null
|| cfg.anchorEl === null
|| retryNumber > 5 // we should try only a few times
) {
return
}
// some browsers report zero height or width because
// we are trying too early to get these dimensions
if (cfg.targetEl.offsetHeight === 0 || cfg.targetEl.offsetWidth === 0) {
setTimeout(() => {
setPosition(cfg, retryNumber + 1);
}, 10);
return
}
const {
targetEl,
offset,
anchorEl,
anchorOrigin,
selfOrigin,
absoluteOffset,
fit,
cover,
maxHeight,
maxWidth
} = cfg;
if (client.is.ios === true && window.visualViewport !== void 0) {
// uses the q-position-engine CSS class
const el = document.body.style;
const { offsetLeft: left, offsetTop: top } = window.visualViewport;
if (left !== vpLeft) {
el.setProperty('--q-pe-left', left + 'px');
vpLeft = left;
}
if (top !== vpTop) {
el.setProperty('--q-pe-top', top + 'px');
vpTop = top;
}
}
// scroll position might change
// if max-height/-width changes, so we
// need to restore it after we calculate
// the new positioning
const { scrollLeft, scrollTop } = targetEl;
const anchorProps = absoluteOffset === void 0
? getAnchorProps(anchorEl, cover === true ? [ 0, 0 ] : offset)
: getAbsoluteAnchorProps(anchorEl, absoluteOffset, offset);
// we "reset" the critical CSS properties
// so we can take an accurate measurement
Object.assign(targetEl.style, {
top: 0,
left: 0,
minWidth: null,
minHeight: null,
maxWidth: maxWidth || '100vw',
maxHeight: maxHeight || '100vh',
visibility: 'visible'
});
const { offsetWidth: origElWidth, offsetHeight: origElHeight } = targetEl;
const { elWidth, elHeight } = fit === true || cover === true
? { elWidth: Math.max(anchorProps.width, origElWidth), elHeight: cover === true ? Math.max(anchorProps.height, origElHeight) : origElHeight }
: { elWidth: origElWidth, elHeight: origElHeight };
let elStyle = { maxWidth, maxHeight };
if (fit === true || cover === true) {
elStyle.minWidth = anchorProps.width + 'px';
if (cover === true) {
elStyle.minHeight = anchorProps.height + 'px';
}
}
Object.assign(targetEl.style, elStyle);
const targetProps = getTargetProps(elWidth, elHeight);
let props = getTopLeftProps(anchorProps, targetProps, anchorOrigin, selfOrigin);
if (absoluteOffset === void 0 || offset === void 0) {
applyBoundaries(props, anchorProps, targetProps, anchorOrigin, selfOrigin);
}
else { // we have touch position or context menu with offset
const { top, left } = props; // cache initial values
// apply initial boundaries
applyBoundaries(props, anchorProps, targetProps, anchorOrigin, selfOrigin);
let hasChanged = false;
// did it flip vertically?
if (props.top !== top) {
hasChanged = true;
const offsetY = 2 * offset[ 1 ];
anchorProps.center = anchorProps.top -= offsetY;
anchorProps.bottom -= offsetY + 2;
}
// did it flip horizontally?
if (props.left !== left) {
hasChanged = true;
const offsetX = 2 * offset[ 0 ];
anchorProps.middle = anchorProps.left -= offsetX;
anchorProps.right -= offsetX + 2;
}
if (hasChanged === true) {
// re-calculate props with the new anchor
props = getTopLeftProps(anchorProps, targetProps, anchorOrigin, selfOrigin);
// and re-apply boundaries
applyBoundaries(props, anchorProps, targetProps, anchorOrigin, selfOrigin);
}
}
elStyle = {
top: props.top + 'px',
left: props.left + 'px'
};
if (props.maxHeight !== void 0) {
elStyle.maxHeight = props.maxHeight + 'px';
if (anchorProps.height > props.maxHeight) {
elStyle.minHeight = elStyle.maxHeight;
}
}
if (props.maxWidth !== void 0) {
elStyle.maxWidth = props.maxWidth + 'px';
if (anchorProps.width > props.maxWidth) {
elStyle.minWidth = elStyle.maxWidth;
}
}
Object.assign(targetEl.style, elStyle);
// restore scroll position
if (targetEl.scrollTop !== scrollTop) {
targetEl.scrollTop = scrollTop;
}
if (targetEl.scrollLeft !== scrollLeft) {
targetEl.scrollLeft = scrollLeft;
}
}
function applyBoundaries (props, anchorProps, targetProps, anchorOrigin, selfOrigin) {
const
currentHeight = targetProps.bottom,
currentWidth = targetProps.right,
margin = getScrollbarWidth(),
innerHeight = window.innerHeight - margin,
innerWidth = document.body.clientWidth;
if (props.top < 0 || props.top + currentHeight > innerHeight) {
if (selfOrigin.vertical === 'center') {
props.top = anchorProps[ anchorOrigin.vertical ] > innerHeight / 2
? Math.max(0, innerHeight - currentHeight)
: 0;
props.maxHeight = Math.min(currentHeight, innerHeight);
}
else if (anchorProps[ anchorOrigin.vertical ] > innerHeight / 2) {
const anchorY = Math.min(
innerHeight,
anchorOrigin.vertical === 'center'
? anchorProps.center
: (anchorOrigin.vertical === selfOrigin.vertical ? anchorProps.bottom : anchorProps.top)
);
props.maxHeight = Math.min(currentHeight, anchorY);
props.top = Math.max(0, anchorY - currentHeight);
}
else {
props.top = Math.max(0, anchorOrigin.vertical === 'center'
? anchorProps.center
: (anchorOrigin.vertical === selfOrigin.vertical ? anchorProps.top : anchorProps.bottom)
);
props.maxHeight = Math.min(currentHeight, innerHeight - props.top);
}
}
if (props.left < 0 || props.left + currentWidth > innerWidth) {
props.maxWidth = Math.min(currentWidth, innerWidth);
if (selfOrigin.horizontal === 'middle') {
props.left = anchorProps[ anchorOrigin.horizontal ] > innerWidth / 2
? Math.max(0, innerWidth - currentWidth)
: 0;
}
else if (anchorProps[ anchorOrigin.horizontal ] > innerWidth / 2) {
const anchorX = Math.min(
innerWidth,
anchorOrigin.horizontal === 'middle'
? anchorProps.middle
: (anchorOrigin.horizontal === selfOrigin.horizontal ? anchorProps.right : anchorProps.left)
);
props.maxWidth = Math.min(currentWidth, anchorX);
props.left = Math.max(0, anchorX - props.maxWidth);
}
else {
props.left = Math.max(0, anchorOrigin.horizontal === 'middle'
? anchorProps.middle
: (anchorOrigin.horizontal === selfOrigin.horizontal ? anchorProps.left : anchorProps.right)
);
props.maxWidth = Math.min(currentWidth, innerWidth - props.left);
}
}
}
var QMenu = createComponent({
name: 'QMenu',
inheritAttrs: false,
props: {
...useAnchorProps,
...useModelToggleProps,
...useDarkProps,
...useTransitionProps,
persistent: Boolean,
autoClose: Boolean,
separateClosePopup: Boolean,
noRouteDismiss: Boolean,
noRefocus: Boolean,
noFocus: Boolean,
fit: Boolean,
cover: Boolean,
square: Boolean,
anchor: {
type: String,
validator: validatePosition
},
self: {
type: String,
validator: validatePosition
},
offset: {
type: Array,
validator: validateOffset
},
scrollTarget: {
default: void 0
},
touchPosition: Boolean,
maxHeight: {
type: String,
default: null
},
maxWidth: {
type: String,
default: null
}
},
emits: [
...useModelToggleEmits,
'click', 'escapeKey'
],
setup (props, { slots, emit, attrs }) {
let refocusTarget = null, absoluteOffset, unwatchPosition, avoidAutoClose;
const vm = vue.getCurrentInstance();
const { proxy } = vm;
const { $q } = proxy;
const innerRef = vue.ref(null);
const showing = vue.ref(false);
const hideOnRouteChange = vue.computed(() =>
props.persistent !== true
&& props.noRouteDismiss !== true
);
const isDark = useDark(props, $q);
const { registerTick, removeTick } = useTick();
const { registerTimeout } = useTimeout();
const { transitionProps, transitionStyle } = useTransition(props);
const { localScrollTarget, changeScrollEvent, unconfigureScrollTarget } = useScrollTarget(props, configureScrollTarget);
const { anchorEl, canShow } = useAnchor({ showing });
const { hide } = useModelToggle({
showing, canShow, handleShow, handleHide,
hideOnRouteChange,
processOnMount: true
});
const { showPortal, hidePortal, renderPortal } = usePortal(vm, innerRef, renderPortalContent, 'menu');
const clickOutsideProps = {
anchorEl,
innerRef,
onClickOutside (e) {
if (props.persistent !== true && showing.value === true) {
hide(e);
if (
// always prevent touch event
e.type === 'touchstart'
// prevent click if it's on a dialog backdrop
|| e.target.classList.contains('q-dialog__backdrop')
) {
stopAndPrevent(e);
}
return true
}
}
};
const anchorOrigin = vue.computed(() =>
parsePosition(
props.anchor || (
props.cover === true ? 'center middle' : 'bottom start'
),
$q.lang.rtl
)
);
const selfOrigin = vue.computed(() => (
props.cover === true
? anchorOrigin.value
: parsePosition(props.self || 'top start', $q.lang.rtl)
));
const menuClass = vue.computed(() =>
(props.square === true ? ' q-menu--square' : '')
+ (isDark.value === true ? ' q-menu--dark q-dark' : '')
);
const onEvents = vue.computed(() => (
props.autoClose === true
? { onClick: onAutoClose }
: {}
));
const handlesFocus = vue.computed(() =>
showing.value === true && props.persistent !== true
);
vue.watch(handlesFocus, val => {
if (val === true) {
addEscapeKey(onEscapeKey);
addClickOutside(clickOutsideProps);
}
else {
removeEscapeKey(onEscapeKey);
removeClickOutside(clickOutsideProps);
}
});
function focus () {
addFocusFn(() => {
let node = innerRef.value;
if (node && node.contains(document.activeElement) !== true) {
node = node.querySelector('[autofocus][tabindex], [data-autofocus][tabindex]')
|| node.querySelector('[autofocus] [tabindex], [data-autofocus] [tabindex]')
|| node.querySelector('[autofocus], [data-autofocus]')
|| node;
node.focus({ preventScroll: true });
}
});
}
function handleShow (evt) {
refocusTarget = props.noRefocus === false
? document.activeElement
: null;
addFocusout(onFocusout);
showPortal();
configureScrollTarget();
absoluteOffset = void 0;
if (evt !== void 0 && (props.touchPosition || props.contextMenu)) {
const pos = position(evt);
if (pos.left !== void 0) {
const { top, left } = anchorEl.value.getBoundingClientRect();
absoluteOffset = { left: pos.left - left, top: pos.top - top };
}
}
if (unwatchPosition === void 0) {
unwatchPosition = vue.watch(
() => $q.screen.width + '|' + $q.screen.height + '|' + props.self + '|' + props.anchor + '|' + $q.lang.rtl,
updatePosition
);
}
if (props.noFocus !== true) {
document.activeElement.blur();
}
// should removeTick() if this gets removed
registerTick(() => {
updatePosition();
props.noFocus !== true && focus();
});
// should removeTimeout() if this gets removed
registerTimeout(() => {
// required in order to avoid the "double-tap needed" issue
if ($q.platform.is.ios === true) {
// if auto-close, then this click should
// not close the menu
avoidAutoClose = props.autoClose;
innerRef.value.click();
}
updatePosition();
showPortal(true); // done showing portal
emit('show', evt);
}, props.transitionDuration);
}
function handleHide (evt) {
removeTick();
hidePortal();
anchorCleanup(true);
if (
refocusTarget !== null
&& (
// menu was hidden from code or ESC plugin
evt === void 0
// menu was not closed from a mouse or touch clickOutside
|| evt.qClickOutside !== true
)
) {
((evt && evt.type.indexOf('key') === 0
? refocusTarget.closest('[tabindex]:not([tabindex^="-"])')
: void 0
) || refocusTarget).focus();
refocusTarget = null;
}
// should removeTimeout() if this gets removed
registerTimeout(() => {
hidePortal(true); // done hiding, now destroy
emit('hide', evt);
}, props.transitionDuration);
}
function anchorCleanup (hiding) {
absoluteOffset = void 0;
if (unwatchPosition !== void 0) {
unwatchPosition();
unwatchPosition = void 0;
}
if (hiding === true || showing.value === true) {
removeFocusout(onFocusout);
unconfigureScrollTarget();
removeClickOutside(clickOutsideProps);
removeEscapeKey(onEscapeKey);
}
if (hiding !== true) {
refocusTarget = null;
}
}
function configureScrollTarget () {
if (anchorEl.value !== null || props.scrollTarget !== void 0) {
localScrollTarget.value = getScrollTarget(anchorEl.value, props.scrollTarget);
changeScrollEvent(localScrollTarget.value, updatePosition);
}
}
function onAutoClose (e) {
// if auto-close, then the ios double-tap fix which
// issues a click should not close the menu
if (avoidAutoClose !== true) {
closePortalMenus(proxy, e);
emit('click', e);
}
else {
avoidAutoClose = false;
}
}
function onFocusout (evt) {
// the focus is not in a vue child component
if (
handlesFocus.value === true
&& props.noFocus !== true
&& childHasFocus(innerRef.value, evt.target) !== true
) {
focus();
}
}
function onEscapeKey (evt) {
emit('escapeKey');
hide(evt);
}
function updatePosition () {
setPosition({
targetEl: innerRef.value,
offset: props.offset,
anchorEl: anchorEl.value,
anchorOrigin: anchorOrigin.value,
selfOrigin: selfOrigin.value,
absoluteOffset,
fit: props.fit,
cover: props.cover,
maxHeight: props.maxHeight,
maxWidth: props.maxWidth
});
}
function renderPortalContent () {
return vue.h(
vue.Transition,
transitionProps.value,
() => (
showing.value === true
? vue.h('div', {
role: 'menu',
...attrs,
ref: innerRef,
tabindex: -1,
class: [
'q-menu q-position-engine scroll' + menuClass.value,
attrs.class
],
style: [
attrs.style,
transitionStyle.value
],
...onEvents.value
}, hSlot(slots.default))
: null
)
)
}
vue.onBeforeUnmount(anchorCleanup);
// expose public methods
Object.assign(proxy, { focus, updatePosition });
return renderPortal
}
});
/**
* Based on the work of https://github.com/jchook/uuid-random
*/
let
buf,
bufIdx = 0;
const hexBytes = new Array(256);
// Pre-calculate toString(16) for speed
for (let i = 0; i < 256; i++) {
hexBytes[ i ] = (i + 0x100).toString(16).substring(1);
}
// Use best available PRNG
const randomBytes = (() => {
// Node & Browser support
const lib = typeof crypto !== 'undefined'
? crypto
: (
typeof window !== 'undefined'
? window.crypto || window.msCrypto
: void 0
);
if (lib !== void 0) {
if (lib.randomBytes !== void 0) {
return lib.randomBytes
}
if (lib.getRandomValues !== void 0) {
return n => {
const bytes = new Uint8Array(n);
lib.getRandomValues(bytes);
return bytes
}
}
}
return n => {
const r = [];
for (let i = n; i > 0; i--) {
r.push(Math.floor(Math.random() * 256));
}
return r
}
})();
// Buffer random numbers for speed
// Reduce memory usage by decreasing this number (min 16)
// or improve speed by increasing this number (try 16384)
const BUFFER_SIZE = 4096;
function uid$3 () {
// Buffer some random bytes for speed
if (buf === void 0 || (bufIdx + 16 > BUFFER_SIZE)) {
bufIdx = 0;
buf = randomBytes(BUFFER_SIZE);
}
const b = Array.prototype.slice.call(buf, bufIdx, (bufIdx += 16));
b[ 6 ] = (b[ 6 ] & 0x0f) | 0x40;
b[ 8 ] = (b[ 8 ] & 0x3f) | 0x80;
return hexBytes[ b[ 0 ] ] + hexBytes[ b[ 1 ] ]
+ hexBytes[ b[ 2 ] ] + hexBytes[ b[ 3 ] ] + '-'
+ hexBytes[ b[ 4 ] ] + hexBytes[ b[ 5 ] ] + '-'
+ hexBytes[ b[ 6 ] ] + hexBytes[ b[ 7 ] ] + '-'
+ hexBytes[ b[ 8 ] ] + hexBytes[ b[ 9 ] ] + '-'
+ hexBytes[ b[ 10 ] ] + hexBytes[ b[ 11 ] ]
+ hexBytes[ b[ 12 ] ] + hexBytes[ b[ 13 ] ]
+ hexBytes[ b[ 14 ] ] + hexBytes[ b[ 15 ] ]
}
const btnPropsList = Object.keys(useBtnProps);
const passBtnProps = props => btnPropsList.reduce(
(acc, key) => {
const val = props[ key ];
if (val !== void 0) {
acc[ key ] = val;
}
return acc
},
{}
);
var QBtnDropdown = createComponent({
name: 'QBtnDropdown',
props: {
...useBtnProps,
...useTransitionProps,
modelValue: Boolean,
split: Boolean,
dropdownIcon: String,
contentClass: [ Array, String, Object ],
contentStyle: [ Array, String, Object ],
cover: Boolean,
persistent: Boolean,
noRouteDismiss: Boolean,
autoClose: Boolean,
menuAnchor: {
type: String,
default: 'bottom end'
},
menuSelf: {
type: String,
default: 'top end'
},
menuOffset: Array,
disableMainBtn: Boolean,
disableDropdown: Boolean,
noIconAnimation: Boolean,
toggleAriaLabel: String
},
emits: [ 'update:modelValue', 'click', 'beforeShow', 'show', 'beforeHide', 'hide' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const showing = vue.ref(props.modelValue);
const menuRef = vue.ref(null);
const targetUid = uid$3();
const ariaAttrs = vue.computed(() => {
const acc = {
'aria-expanded': showing.value === true ? 'true' : 'false',
'aria-haspopup': 'true',
'aria-controls': targetUid,
'aria-label': props.toggleAriaLabel || proxy.$q.lang.label[ showing.value === true ? 'collapse' : 'expand' ](props.label)
};
if (
props.disable === true
|| (
(props.split === false && props.disableMainBtn === true)
|| props.disableDropdown === true
)
) {
acc[ 'aria-disabled' ] = 'true';
}
return acc
});
const iconClass = vue.computed(() =>
'q-btn-dropdown__arrow'
+ (showing.value === true && props.noIconAnimation === false ? ' rotate-180' : '')
+ (props.split === false ? ' q-btn-dropdown__arrow-container' : '')
);
const btnDesignAttr = vue.computed(() => getBtnDesignAttr(props));
const btnProps = vue.computed(() => passBtnProps(props));
vue.watch(() => props.modelValue, val => {
menuRef.value !== null && menuRef.value[ val ? 'show' : 'hide' ]();
});
vue.watch(() => props.split, hide);
function onBeforeShow (e) {
showing.value = true;
emit('beforeShow', e);
}
function onShow (e) {
emit('show', e);
emit('update:modelValue', true);
}
function onBeforeHide (e) {
showing.value = false;
emit('beforeHide', e);
}
function onHide (e) {
emit('hide', e);
emit('update:modelValue', false);
}
function onClick (e) {
emit('click', e);
}
function onClickHide (e) {
stop(e);
hide();
emit('click', e);
}
function toggle (evt) {
menuRef.value !== null && menuRef.value.toggle(evt);
}
function show (evt) {
menuRef.value !== null && menuRef.value.show(evt);
}
function hide (evt) {
menuRef.value !== null && menuRef.value.hide(evt);
}
// expose public methods
Object.assign(proxy, {
show, hide, toggle
});
vue.onMounted(() => {
props.modelValue === true && show();
});
return () => {
const Arrow = [
vue.h(QIcon, {
class: iconClass.value,
name: props.dropdownIcon || proxy.$q.iconSet.arrow.dropdown
})
];
props.disableDropdown !== true && Arrow.push(
vue.h(QMenu, {
ref: menuRef,
id: targetUid,
class: props.contentClass,
style: props.contentStyle,
cover: props.cover,
fit: true,
persistent: props.persistent,
noRouteDismiss: props.noRouteDismiss,
autoClose: props.autoClose,
anchor: props.menuAnchor,
self: props.menuSelf,
offset: props.menuOffset,
separateClosePopup: true,
transitionShow: props.transitionShow,
transitionHide: props.transitionHide,
transitionDuration: props.transitionDuration,
onBeforeShow,
onShow,
onBeforeHide,
onHide
}, slots.default)
);
if (props.split === false) {
return vue.h(QBtn, {
class: 'q-btn-dropdown q-btn-dropdown--simple',
...btnProps.value,
...ariaAttrs.value,
disable: props.disable === true || props.disableMainBtn === true,
noWrap: true,
round: false,
onClick
}, {
default: () => hSlot(slots.label, []).concat(Arrow),
loading: slots.loading
})
}
return vue.h(QBtnGroup, {
class: 'q-btn-dropdown q-btn-dropdown--split no-wrap q-btn-item',
rounded: props.rounded,
square: props.square,
...btnDesignAttr.value,
glossy: props.glossy,
stretch: props.stretch
}, () => [
vue.h(QBtn, {
class: 'q-btn-dropdown--current',
...btnProps.value,
disable: props.disable === true || props.disableMainBtn === true,
noWrap: true,
round: false,
onClick: onClickHide
}, {
default: slots.label,
loading: slots.loading
}),
vue.h(QBtn, {
class: 'q-btn-dropdown__arrow-container q-anchor--skip',
...ariaAttrs.value,
...btnDesignAttr.value,
disable: props.disable === true || props.disableDropdown === true,
rounded: props.rounded,
color: props.color,
textColor: props.textColor,
dense: props.dense,
size: props.size,
padding: props.padding,
ripple: props.ripple
}, () => Arrow)
])
}
}
});
const useFormProps = {
name: String
};
function useFormAttrs (props) {
return vue.computed(() => ({
type: 'hidden',
name: props.name,
value: props.modelValue
}))
}
function useFormInject (formAttrs = {}) {
return (child, action, className) => {
child[ action ](
vue.h('input', {
class: 'hidden' + (className || ''),
...formAttrs.value
})
);
}
}
function useFormInputNameAttr (props) {
return vue.computed(() => props.name || props.for)
}
var QBtnToggle = createComponent({
name: 'QBtnToggle',
props: {
...useFormProps,
modelValue: {
required: true
},
options: {
type: Array,
required: true,
validator: v => v.every(
opt => ('label' in opt || 'icon' in opt || 'slot' in opt) && 'value' in opt
)
},
// To avoid seeing the active raise shadow through
// the transparent button, give it a color (even white)
color: String,
textColor: String,
toggleColor: {
type: String,
default: 'primary'
},
toggleTextColor: String,
outline: Boolean,
flat: Boolean,
unelevated: Boolean,
rounded: Boolean,
push: Boolean,
glossy: Boolean,
size: String,
padding: String,
noCaps: Boolean,
noWrap: Boolean,
dense: Boolean,
readonly: Boolean,
disable: Boolean,
stack: Boolean,
stretch: Boolean,
spread: Boolean,
clearable: Boolean,
ripple: {
type: [ Boolean, Object ],
default: true
}
},
emits: [ 'update:modelValue', 'clear', 'click' ],
setup (props, { slots, emit }) {
const hasActiveValue = vue.computed(() =>
props.options.find(opt => opt.value === props.modelValue) !== void 0
);
const formAttrs = vue.computed(() => ({
type: 'hidden',
name: props.name,
value: props.modelValue
}));
const injectFormInput = useFormInject(formAttrs);
const btnDesignAttr = vue.computed(() => getBtnDesignAttr(props));
const btnOptionDesign = vue.computed(() => ({
rounded: props.rounded,
dense: props.dense,
...btnDesignAttr.value
}));
const btnOptions = vue.computed(() => props.options.map((item, i) => {
const { attrs, value, slot, ...opt } = item;
return {
slot,
props: {
key: i,
'aria-pressed': value === props.modelValue ? 'true' : 'false',
...attrs,
...opt,
...btnOptionDesign.value,
disable: props.disable === true || opt.disable === true,
// Options that come from the button specific options first, then from general props
color: value === props.modelValue
? mergeOpt(opt, 'toggleColor')
: mergeOpt(opt, 'color'),
textColor: value === props.modelValue
? mergeOpt(opt, 'toggleTextColor')
: mergeOpt(opt, 'textColor'),
noCaps: mergeOpt(opt, 'noCaps') === true,
noWrap: mergeOpt(opt, 'noWrap') === true,
size: mergeOpt(opt, 'size'),
padding: mergeOpt(opt, 'padding'),
ripple: mergeOpt(opt, 'ripple'),
stack: mergeOpt(opt, 'stack') === true,
stretch: mergeOpt(opt, 'stretch') === true,
onClick (e) { set(value, item, e); }
}
}
}));
function set (value, opt, e) {
if (props.readonly !== true) {
if (props.modelValue === value) {
if (props.clearable === true) {
emit('update:modelValue', null, null);
emit('clear');
}
}
else {
emit('update:modelValue', value, opt);
}
emit('click', e);
}
}
function mergeOpt (opt, key) {
return opt[ key ] === void 0 ? props[ key ] : opt[ key ]
}
function getContent () {
const child = btnOptions.value.map(opt => {
return vue.h(QBtn, opt.props, opt.slot !== void 0 ? slots[ opt.slot ] : void 0)
});
if (props.name !== void 0 && props.disable !== true && hasActiveValue.value === true) {
injectFormInput(child, 'push');
}
return hMergeSlot(slots.default, child)
}
return () => vue.h(QBtnGroup, {
class: 'q-btn-toggle',
...btnDesignAttr.value,
rounded: props.rounded,
stretch: props.stretch,
glossy: props.glossy,
spread: props.spread
}, getContent)
}
});
var QCard = createComponent({
name: 'QCard',
props: {
...useDarkProps,
tag: {
type: String,
default: 'div'
},
square: Boolean,
flat: Boolean,
bordered: Boolean
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const classes = vue.computed(() =>
'q-card'
+ (isDark.value === true ? ' q-card--dark q-dark' : '')
+ (props.bordered === true ? ' q-card--bordered' : '')
+ (props.square === true ? ' q-card--square no-border-radius' : '')
+ (props.flat === true ? ' q-card--flat no-shadow' : '')
);
return () => vue.h(props.tag, { class: classes.value }, hSlot(slots.default))
}
});
var QCardSection = createComponent({
name: 'QCardSection',
props: {
tag: {
type: String,
default: 'div'
},
horizontal: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() =>
'q-card__section'
+ ` q-card__section--${ props.horizontal === true ? 'horiz row no-wrap' : 'vert' }`
);
return () => vue.h(props.tag, { class: classes.value }, hSlot(slots.default))
}
});
var QCardActions = createComponent({
name: 'QCardActions',
props: {
...useAlignProps,
vertical: Boolean
},
setup (props, { slots }) {
const alignClass = useAlign(props);
const classes = vue.computed(() =>
`q-card__actions ${ alignClass.value }`
+ ` q-card__actions--${ props.vertical === true ? 'vert column' : 'horiz row' }`
);
return () => vue.h('div', { class: classes.value }, hSlot(slots.default))
}
});
const modifiersAll = {
left: true,
right: true,
up: true,
down: true,
horizontal: true,
vertical: true
};
const directionList = Object.keys(modifiersAll);
modifiersAll.all = true;
function getModifierDirections (mod) {
const dir = {};
for (const direction of directionList) {
if (mod[ direction ] === true) {
dir[ direction ] = true;
}
}
if (Object.keys(dir).length === 0) {
return modifiersAll
}
if (dir.horizontal === true) {
dir.left = dir.right = true;
}
else if (dir.left === true && dir.right === true) {
dir.horizontal = true;
}
if (dir.vertical === true) {
dir.up = dir.down = true;
}
else if (dir.up === true && dir.down === true) {
dir.vertical = true;
}
if (dir.horizontal === true && dir.vertical === true) {
dir.all = true;
}
return dir
}
// This is especially important (not the main reason, but important)
// for TouchSwipe directive running on Firefox
// because text selection on such elements cannot be determined
// without additional work (on top of getSelection() API)
// https://bugzilla.mozilla.org/show_bug.cgi?id=85686
const avoidNodeNamesList = [ 'INPUT', 'TEXTAREA' ];
function shouldStart (evt, ctx) {
return ctx.event === void 0
&& evt.target !== void 0
&& evt.target.draggable !== true
&& typeof ctx.handler === 'function'
&& avoidNodeNamesList.includes(evt.target.nodeName.toUpperCase()) === false
&& (evt.qClonedBy === void 0 || evt.qClonedBy.indexOf(ctx.uid) === -1)
}
function parseArg (arg) {
// delta (min velocity -- dist / time)
// mobile min distance on first move
// desktop min distance until deciding if it's a swipe or not
const data = [ 0.06, 6, 50 ];
if (typeof arg === 'string' && arg.length) {
arg.split(':').forEach((val, index) => {
const v = parseFloat(val);
v && (data[ index ] = v);
});
}
return data
}
var TouchSwipe = createDirective({
name: 'touch-swipe',
beforeMount (el, { value, arg, modifiers }) {
// early return, we don't need to do anything
if (modifiers.mouse !== true && client.has.touch !== true) {
return
}
const mouseCapture = modifiers.mouseCapture === true ? 'Capture' : '';
const ctx = {
handler: value,
sensitivity: parseArg(arg),
direction: getModifierDirections(modifiers),
noop,
mouseStart (evt) {
if (shouldStart(evt, ctx) && leftClick(evt)) {
addEvt(ctx, 'temp', [
[ document, 'mousemove', 'move', `notPassive${ mouseCapture }` ],
[ document, 'mouseup', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt, true);
}
},
touchStart (evt) {
if (shouldStart(evt, ctx)) {
const target = evt.target;
addEvt(ctx, 'temp', [
[ target, 'touchmove', 'move', 'notPassiveCapture' ],
[ target, 'touchcancel', 'end', 'notPassiveCapture' ],
[ target, 'touchend', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt);
}
},
start (evt, mouseEvent) {
client.is.firefox === true && preventDraggable(el, true);
const pos = position(evt);
ctx.event = {
x: pos.left,
y: pos.top,
time: Date.now(),
mouse: mouseEvent === true,
dir: false
};
},
move (evt) {
if (ctx.event === void 0) {
return
}
if (ctx.event.dir !== false) {
stopAndPrevent(evt);
return
}
const time = Date.now() - ctx.event.time;
if (time === 0) {
return
}
const
pos = position(evt),
distX = pos.left - ctx.event.x,
absX = Math.abs(distX),
distY = pos.top - ctx.event.y,
absY = Math.abs(distY);
if (ctx.event.mouse !== true) {
if (absX < ctx.sensitivity[ 1 ] && absY < ctx.sensitivity[ 1 ]) {
ctx.end(evt);
return
}
}
// is user trying to select text?
// if so, then something should be reported here
// (previous selection, if any, was discarded when swipe started)
else if (window.getSelection().toString() !== '') {
ctx.end(evt);
return
}
else if (absX < ctx.sensitivity[ 2 ] && absY < ctx.sensitivity[ 2 ]) {
return
}
const
velX = absX / time,
velY = absY / time;
if (
ctx.direction.vertical === true
&& absX < absY
&& absX < 100
&& velY > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = distY < 0 ? 'up' : 'down';
}
if (
ctx.direction.horizontal === true
&& absX > absY
&& absY < 100
&& velX > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = distX < 0 ? 'left' : 'right';
}
if (
ctx.direction.up === true
&& absX < absY
&& distY < 0
&& absX < 100
&& velY > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = 'up';
}
if (
ctx.direction.down === true
&& absX < absY
&& distY > 0
&& absX < 100
&& velY > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = 'down';
}
if (
ctx.direction.left === true
&& absX > absY
&& distX < 0
&& absY < 100
&& velX > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = 'left';
}
if (
ctx.direction.right === true
&& absX > absY
&& distX > 0
&& absY < 100
&& velX > ctx.sensitivity[ 0 ]
) {
ctx.event.dir = 'right';
}
if (ctx.event.dir !== false) {
stopAndPrevent(evt);
if (ctx.event.mouse === true) {
document.body.classList.add('no-pointer-events--children');
document.body.classList.add('non-selectable');
clearSelection();
ctx.styleCleanup = withDelay => {
ctx.styleCleanup = void 0;
document.body.classList.remove('non-selectable');
const remove = () => {
document.body.classList.remove('no-pointer-events--children');
};
if (withDelay === true) { setTimeout(remove, 50); }
else { remove(); }
};
}
ctx.handler({
evt,
touch: ctx.event.mouse !== true,
mouse: ctx.event.mouse,
direction: ctx.event.dir,
duration: time,
distance: {
x: absX,
y: absY
}
});
}
else {
ctx.end(evt);
}
},
end (evt) {
if (ctx.event === void 0) {
return
}
cleanEvt(ctx, 'temp');
client.is.firefox === true && preventDraggable(el, false);
ctx.styleCleanup !== void 0 && ctx.styleCleanup(true);
evt !== void 0 && ctx.event.dir !== false && stopAndPrevent(evt);
ctx.event = void 0;
}
};
el.__qtouchswipe = ctx;
if (modifiers.mouse === true) {
// account for UMD too where modifiers will be lowercased to work
const capture = modifiers.mouseCapture === true || modifiers.mousecapture === true
? 'Capture'
: '';
addEvt(ctx, 'main', [
[ el, 'mousedown', 'mouseStart', `passive${ capture }` ]
]);
}
client.has.touch === true && addEvt(ctx, 'main', [
[ el, 'touchstart', 'touchStart', `passive${ modifiers.capture === true ? 'Capture' : '' }` ],
[ el, 'touchmove', 'noop', 'notPassiveCapture' ] // cannot be passive (ex: iOS scroll)
]);
},
updated (el, bindings) {
const ctx = el.__qtouchswipe;
if (ctx !== void 0) {
if (bindings.oldValue !== bindings.value) {
typeof bindings.value !== 'function' && ctx.end();
ctx.handler = bindings.value;
}
ctx.direction = getModifierDirections(bindings.modifiers);
}
},
beforeUnmount (el) {
const ctx = el.__qtouchswipe;
if (ctx !== void 0) {
cleanEvt(ctx, 'main');
cleanEvt(ctx, 'temp');
client.is.firefox === true && preventDraggable(el, false);
ctx.styleCleanup !== void 0 && ctx.styleCleanup();
delete el.__qtouchswipe;
}
}
}
);
function useCache () {
const cache = new Map();
return {
getCache: function (key, obj) {
return cache[ key ] === void 0
? (cache[ key ] = obj)
: cache[ key ]
},
getCacheWithFn: function (key, fn) {
return cache[ key ] === void 0
? (cache[ key ] = fn())
: cache[ key ]
}
}
}
const usePanelChildProps = {
name: { required: true },
disable: Boolean
};
const PanelWrapper$1 = {
setup (_, { slots }) {
return () => vue.h('div', {
class: 'q-panel scroll',
role: 'tabpanel'
}, hSlot(slots.default))
}
};
const usePanelProps = {
modelValue: {
required: true
},
animated: Boolean,
infinite: Boolean,
swipeable: Boolean,
vertical: Boolean,
transitionPrev: String,
transitionNext: String,
transitionDuration: {
type: [ String, Number ],
default: 300
},
keepAlive: Boolean,
keepAliveInclude: [ String, Array, RegExp ],
keepAliveExclude: [ String, Array, RegExp ],
keepAliveMax: Number
};
const usePanelEmits = [ 'update:modelValue', 'beforeTransition', 'transition' ];
function usePanel () {
const { props, emit, proxy } = vue.getCurrentInstance();
const { getCacheWithFn } = useCache();
let panels, forcedPanelTransition;
const panelIndex = vue.ref(null);
const panelTransition = vue.ref(null);
function onSwipe (evt) {
const dir = props.vertical === true ? 'up' : 'left';
goToPanelByOffset((proxy.$q.lang.rtl === true ? -1 : 1) * (evt.direction === dir ? 1 : -1));
}
const panelDirectives = vue.computed(() => {
// if props.swipeable
return [ [
TouchSwipe,
onSwipe,
void 0,
{
horizontal: props.vertical !== true,
vertical: props.vertical,
mouse: true
}
] ]
});
const transitionPrev = vue.computed(() =>
props.transitionPrev || `slide-${ props.vertical === true ? 'down' : 'right' }`
);
const transitionNext = vue.computed(() =>
props.transitionNext || `slide-${ props.vertical === true ? 'up' : 'left' }`
);
const transitionStyle = vue.computed(
() => `--q-transition-duration: ${ props.transitionDuration }ms`
);
const contentKey = vue.computed(() => (
typeof props.modelValue === 'string' || typeof props.modelValue === 'number'
? props.modelValue
: String(props.modelValue)
));
const keepAliveProps = vue.computed(() => ({
include: props.keepAliveInclude,
exclude: props.keepAliveExclude,
max: props.keepAliveMax
}));
const needsUniqueKeepAliveWrapper = vue.computed(() =>
props.keepAliveInclude !== void 0
|| props.keepAliveExclude !== void 0
);
vue.watch(() => props.modelValue, (newVal, oldVal) => {
const index = isValidPanelName(newVal) === true
? getPanelIndex(newVal)
: -1;
if (forcedPanelTransition !== true) {
updatePanelTransition(
index === -1 ? 0 : (index < getPanelIndex(oldVal) ? -1 : 1)
);
}
if (panelIndex.value !== index) {
panelIndex.value = index;
emit('beforeTransition', newVal, oldVal);
vue.nextTick(() => {
emit('transition', newVal, oldVal);
});
}
});
function nextPanel () { goToPanelByOffset(1); }
function previousPanel () { goToPanelByOffset(-1); }
function goToPanel (name) {
emit('update:modelValue', name);
}
function isValidPanelName (name) {
return name !== void 0 && name !== null && name !== ''
}
function getPanelIndex (name) {
return panels.findIndex(panel => {
return panel.props.name === name
&& panel.props.disable !== ''
&& panel.props.disable !== true
})
}
function getEnabledPanels () {
return panels.filter(panel => {
return panel.props.disable !== ''
&& panel.props.disable !== true
})
}
function updatePanelTransition (direction) {
const val = direction !== 0 && props.animated === true && panelIndex.value !== -1
? 'q-transition--' + (direction === -1 ? transitionPrev.value : transitionNext.value)
: null;
if (panelTransition.value !== val) {
panelTransition.value = val;
}
}
function goToPanelByOffset (direction, startIndex = panelIndex.value) {
let index = startIndex + direction;
while (index > -1 && index < panels.length) {
const opt = panels[ index ];
if (
opt !== void 0
&& opt.props.disable !== ''
&& opt.props.disable !== true
) {
updatePanelTransition(direction);
forcedPanelTransition = true;
emit('update:modelValue', opt.props.name);
setTimeout(() => {
forcedPanelTransition = false;
});
return
}
index += direction;
}
if (props.infinite === true && panels.length !== 0 && startIndex !== -1 && startIndex !== panels.length) {
goToPanelByOffset(direction, direction === -1 ? panels.length : -1);
}
}
function updatePanelIndex () {
const index = getPanelIndex(props.modelValue);
if (panelIndex.value !== index) {
panelIndex.value = index;
}
return true
}
function getPanelContentChild () {
const panel = isValidPanelName(props.modelValue) === true
&& updatePanelIndex()
&& panels[ panelIndex.value ];
return props.keepAlive === true
? [
vue.h(vue.KeepAlive, keepAliveProps.value, [
vue.h(
needsUniqueKeepAliveWrapper.value === true
? getCacheWithFn(contentKey.value, () => ({ ...PanelWrapper$1, name: contentKey.value }))
: PanelWrapper$1,
{ key: contentKey.value, style: transitionStyle.value },
() => panel
)
])
]
: [
vue.h('div', {
class: 'q-panel scroll',
style: transitionStyle.value,
key: contentKey.value,
role: 'tabpanel'
}, [ panel ])
]
}
function getPanelContent () {
if (panels.length === 0) {
return
}
return props.animated === true
? [ vue.h(vue.Transition, { name: panelTransition.value }, getPanelContentChild) ]
: getPanelContentChild()
}
function updatePanelsList (slots) {
panels = getNormalizedVNodes(
hSlot(slots.default, [])
).filter(
panel => panel.props !== null
&& panel.props.slot === void 0
&& isValidPanelName(panel.props.name) === true
);
return panels.length
}
function getPanels () {
return panels
}
// expose public methods
Object.assign(proxy, {
next: nextPanel,
previous: previousPanel,
goTo: goToPanel
});
return {
panelIndex,
panelDirectives,
updatePanelsList,
updatePanelIndex,
getPanelContent,
getEnabledPanels,
getPanels,
isValidPanelName,
keepAliveProps,
needsUniqueKeepAliveWrapper,
goToPanelByOffset,
goToPanel,
nextPanel,
previousPanel
}
}
let counter = 0;
const useFullscreenProps = {
fullscreen: Boolean,
noRouteFullscreenExit: Boolean
};
const useFullscreenEmits = [ 'update:fullscreen', 'fullscreen' ];
function useFullscreen () {
const vm = vue.getCurrentInstance();
const { props, emit, proxy } = vm;
let historyEntry, fullscreenFillerNode, container;
const inFullscreen = vue.ref(false);
vmHasRouter(vm) === true && vue.watch(() => proxy.$route.fullPath, () => {
props.noRouteFullscreenExit !== true && exitFullscreen();
});
vue.watch(() => props.fullscreen, v => {
if (inFullscreen.value !== v) {
toggleFullscreen();
}
});
vue.watch(inFullscreen, v => {
emit('update:fullscreen', v);
emit('fullscreen', v);
});
function toggleFullscreen () {
if (inFullscreen.value === true) {
exitFullscreen();
}
else {
setFullscreen();
}
}
function setFullscreen () {
if (inFullscreen.value === true) {
return
}
inFullscreen.value = true;
container = proxy.$el.parentNode;
container.replaceChild(fullscreenFillerNode, proxy.$el);
document.body.appendChild(proxy.$el);
counter++;
if (counter === 1) {
document.body.classList.add('q-body--fullscreen-mixin');
}
historyEntry = {
handler: exitFullscreen
};
History.add(historyEntry);
}
function exitFullscreen () {
if (inFullscreen.value !== true) {
return
}
if (historyEntry !== void 0) {
History.remove(historyEntry);
historyEntry = void 0;
}
container.replaceChild(proxy.$el, fullscreenFillerNode);
inFullscreen.value = false;
counter = Math.max(0, counter - 1);
if (counter === 0) {
document.body.classList.remove('q-body--fullscreen-mixin');
if (proxy.$el.scrollIntoView !== void 0) {
setTimeout(() => { proxy.$el.scrollIntoView(); });
}
}
}
vue.onBeforeMount(() => {
fullscreenFillerNode = document.createElement('span');
});
vue.onMounted(() => {
props.fullscreen === true && setFullscreen();
});
vue.onBeforeUnmount(exitFullscreen);
// expose public methods
Object.assign(proxy, {
toggleFullscreen,
setFullscreen,
exitFullscreen
});
return {
inFullscreen,
toggleFullscreen
}
}
const navigationPositionOptions = [ 'top', 'right', 'bottom', 'left' ];
const controlTypeOptions = [ 'regular', 'flat', 'outline', 'push', 'unelevated' ];
var QCarousel = createComponent({
name: 'QCarousel',
props: {
...useDarkProps,
...usePanelProps,
...useFullscreenProps,
transitionPrev: { // usePanelParentProps override
type: String,
default: 'fade'
},
transitionNext: { // usePanelParentProps override
type: String,
default: 'fade'
},
height: String,
padding: Boolean,
controlColor: String,
controlTextColor: String,
controlType: {
type: String,
validator: v => controlTypeOptions.includes(v),
default: 'flat'
},
autoplay: [ Number, Boolean ],
arrows: Boolean,
prevIcon: String,
nextIcon: String,
navigation: Boolean,
navigationPosition: {
type: String,
validator: v => navigationPositionOptions.includes(v)
},
navigationIcon: String,
navigationActiveIcon: String,
thumbnails: Boolean
},
emits: [
...useFullscreenEmits,
...usePanelEmits
],
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
let timer = null, panelsLen;
const {
updatePanelsList, getPanelContent,
panelDirectives, goToPanel,
previousPanel, nextPanel, getEnabledPanels,
panelIndex
} = usePanel();
const { inFullscreen } = useFullscreen();
const style = vue.computed(() => (
inFullscreen.value !== true && props.height !== void 0
? { height: props.height }
: {}
));
const direction = vue.computed(() => (props.vertical === true ? 'vertical' : 'horizontal'));
const classes = vue.computed(() =>
`q-carousel q-panel-parent q-carousel--with${ props.padding === true ? '' : 'out' }-padding`
+ (inFullscreen.value === true ? ' fullscreen' : '')
+ (isDark.value === true ? ' q-carousel--dark q-dark' : '')
+ (props.arrows === true ? ` q-carousel--arrows-${ direction.value }` : '')
+ (props.navigation === true ? ` q-carousel--navigation-${ navigationPosition.value }` : '')
);
const arrowIcons = vue.computed(() => {
const ico = [
props.prevIcon || $q.iconSet.carousel[ props.vertical === true ? 'up' : 'left' ],
props.nextIcon || $q.iconSet.carousel[ props.vertical === true ? 'down' : 'right' ]
];
return props.vertical === false && $q.lang.rtl === true
? ico.reverse()
: ico
});
const navIcon = vue.computed(() => props.navigationIcon || $q.iconSet.carousel.navigationIcon);
const navActiveIcon = vue.computed(() => props.navigationActiveIcon || navIcon.value);
const navigationPosition = vue.computed(() => props.navigationPosition
|| (props.vertical === true ? 'right' : 'bottom')
);
const controlProps = vue.computed(() => ({
color: props.controlColor,
textColor: props.controlTextColor,
round: true,
[ props.controlType ]: true,
dense: true
}));
vue.watch(() => props.modelValue, () => {
if (props.autoplay) {
startTimer();
}
});
vue.watch(() => props.autoplay, val => {
if (val) {
startTimer();
}
else if (timer !== null) {
clearTimeout(timer);
timer = null;
}
});
function startTimer () {
const duration = isNumber(props.autoplay) === true
? Math.abs(props.autoplay)
: 5000;
timer !== null && clearTimeout(timer);
timer = setTimeout(() => {
timer = null;
if (duration >= 0) {
nextPanel();
}
else {
previousPanel();
}
}, duration);
}
vue.onMounted(() => {
props.autoplay && startTimer();
});
vue.onBeforeUnmount(() => {
timer !== null && clearTimeout(timer);
});
function getNavigationContainer (type, mapping) {
return vue.h('div', {
class: 'q-carousel__control q-carousel__navigation no-wrap absolute flex'
+ ` q-carousel__navigation--${ type } q-carousel__navigation--${ navigationPosition.value }`
+ (props.controlColor !== void 0 ? ` text-${ props.controlColor }` : '')
}, [
vue.h('div', {
class: 'q-carousel__navigation-inner flex flex-center no-wrap'
}, getEnabledPanels().map(mapping))
])
}
function getContent () {
const node = [];
if (props.navigation === true) {
const fn = slots[ 'navigation-icon' ] !== void 0
? slots[ 'navigation-icon' ]
: opts => vue.h(QBtn, {
key: 'nav' + opts.name,
class: `q-carousel__navigation-icon q-carousel__navigation-icon--${ opts.active === true ? '' : 'in' }active`,
...opts.btnProps,
onClick: opts.onClick
});
const maxIndex = panelsLen - 1;
node.push(
getNavigationContainer('buttons', (panel, index) => {
const name = panel.props.name;
const active = panelIndex.value === index;
return fn({
index,
maxIndex,
name,
active,
btnProps: {
icon: active === true ? navActiveIcon.value : navIcon.value,
size: 'sm',
...controlProps.value
},
onClick: () => { goToPanel(name); }
})
})
);
}
else if (props.thumbnails === true) {
const color = props.controlColor !== void 0
? ` text-${ props.controlColor }`
: '';
node.push(getNavigationContainer('thumbnails', panel => {
const slide = panel.props;
return vue.h('img', {
key: 'tmb#' + slide.name,
class: `q-carousel__thumbnail q-carousel__thumbnail--${ slide.name === props.modelValue ? '' : 'in' }active` + color,
src: slide.imgSrc || slide[ 'img-src' ],
onClick: () => { goToPanel(slide.name); }
})
}));
}
if (props.arrows === true && panelIndex.value >= 0) {
if (props.infinite === true || panelIndex.value > 0) {
node.push(
vue.h('div', {
key: 'prev',
class: `q-carousel__control q-carousel__arrow q-carousel__prev-arrow q-carousel__prev-arrow--${ direction.value } absolute flex flex-center`
}, [
vue.h(QBtn, {
icon: arrowIcons.value[ 0 ],
...controlProps.value,
onClick: previousPanel
})
])
);
}
if (props.infinite === true || panelIndex.value < panelsLen - 1) {
node.push(
vue.h('div', {
key: 'next',
class: 'q-carousel__control q-carousel__arrow q-carousel__next-arrow'
+ ` q-carousel__next-arrow--${ direction.value } absolute flex flex-center`
}, [
vue.h(QBtn, {
icon: arrowIcons.value[ 1 ],
...controlProps.value,
onClick: nextPanel
})
])
);
}
}
return hMergeSlot(slots.control, node)
}
return () => {
panelsLen = updatePanelsList(slots);
return vue.h('div', {
class: classes.value,
style: style.value
}, [
hDir(
'div',
{ class: 'q-carousel__slides-container' },
getPanelContent(),
'sl-cont',
props.swipeable,
() => panelDirectives.value
)
].concat(getContent()))
}
}
});
var QCarouselSlide = createComponent({
name: 'QCarouselSlide',
props: {
...usePanelChildProps,
imgSrc: String
},
setup (props, { slots }) {
const style = vue.computed(() => (
props.imgSrc
? { backgroundImage: `url("${ props.imgSrc }")` }
: {}
));
return () => vue.h('div', {
class: 'q-carousel__slide',
style: style.value
}, hSlot(slots.default))
}
});
var QCarouselControl = createComponent({
name: 'QCarouselControl',
props: {
position: {
type: String,
default: 'bottom-right',
validator: v => [
'top-right', 'top-left',
'bottom-right', 'bottom-left',
'top', 'right', 'bottom', 'left'
].includes(v)
},
offset: {
type: Array,
default: () => [ 18, 18 ],
validator: v => v.length === 2
}
},
setup (props, { slots }) {
const classes = vue.computed(() => `q-carousel__control absolute absolute-${ props.position }`);
const style = vue.computed(() => ({
margin: `${ props.offset[ 1 ] }px ${ props.offset[ 0 ] }px`
}));
return () => vue.h('div', {
class: classes.value,
style: style.value
}, hSlot(slots.default))
}
});
var QChatMessage = createComponent({
name: 'QChatMessage',
props: {
sent: Boolean,
label: String,
bgColor: String,
textColor: String,
name: String,
avatar: String,
text: Array,
stamp: String,
size: String,
labelHtml: Boolean,
nameHtml: Boolean,
textHtml: Boolean,
stampHtml: Boolean
},
setup (props, { slots }) {
const op = vue.computed(() => (props.sent === true ? 'sent' : 'received'));
const textClass = vue.computed(() =>
`q-message-text-content q-message-text-content--${ op.value }`
+ (props.textColor !== void 0 ? ` text-${ props.textColor }` : '')
);
const messageClass = vue.computed(() =>
`q-message-text q-message-text--${ op.value }`
+ (props.bgColor !== void 0 ? ` text-${ props.bgColor }` : '')
);
const containerClass = vue.computed(() =>
'q-message-container row items-end no-wrap'
+ (props.sent === true ? ' reverse' : '')
);
const sizeClass = vue.computed(() => (props.size !== void 0 ? `col-${ props.size }` : ''));
const domProps = vue.computed(() => ({
msg: props.textHtml === true ? 'innerHTML' : 'textContent',
stamp: props.stampHtml === true ? 'innerHTML' : 'textContent',
name: props.nameHtml === true ? 'innerHTML' : 'textContent',
label: props.labelHtml === true ? 'innerHTML' : 'textContent'
}));
function wrapStamp (node) {
if (slots.stamp !== void 0) {
return [ node, vue.h('div', { class: 'q-message-stamp' }, slots.stamp()) ]
}
if (props.stamp) {
return [
node,
vue.h('div', {
class: 'q-message-stamp',
[ domProps.value.stamp ]: props.stamp
})
]
}
return [ node ]
}
function getText (contentList, withSlots) {
const content = withSlots === true
? (contentList.length > 1 ? text => text : text => vue.h('div', [ text ]))
: text => vue.h('div', { [ domProps.value.msg ]: text });
return contentList.map((msg, index) => vue.h('div', {
key: index,
class: messageClass.value
}, [
vue.h('div', { class: textClass.value }, wrapStamp(content(msg)))
]))
}
return () => {
const container = [];
if (slots.avatar !== void 0) {
container.push(slots.avatar());
}
else if (props.avatar !== void 0) {
container.push(
vue.h('img', {
class: `q-message-avatar q-message-avatar--${ op.value }`,
src: props.avatar,
'aria-hidden': 'true'
})
);
}
const msg = [];
if (slots.name !== void 0) {
msg.push(
vue.h('div', { class: `q-message-name q-message-name--${ op.value }` }, slots.name())
);
}
else if (props.name !== void 0) {
msg.push(
vue.h('div', {
class: `q-message-name q-message-name--${ op.value }`,
[ domProps.value.name ]: props.name
})
);
}
if (slots.default !== void 0) {
msg.push(
getText(
getNormalizedVNodes(slots.default()),
true
)
);
}
else if (props.text !== void 0) {
msg.push(getText(props.text));
}
container.push(
vue.h('div', { class: sizeClass.value }, msg)
);
const child = [];
if (slots.label !== void 0) {
child.push(
vue.h('div', { class: 'q-message-label' }, slots.label())
);
}
else if (props.label !== void 0) {
child.push(
vue.h('div', {
class: 'q-message-label',
[ domProps.value.label ]: props.label
})
);
}
child.push(
vue.h('div', { class: containerClass.value }, container)
);
return vue.h('div', {
class: `q-message q-message-${ op.value }`
}, child)
}
}
});
function useRefocusTarget (props, rootRef) {
const refocusRef = vue.ref(null);
const refocusTargetEl = vue.computed(() => {
if (props.disable === true) {
return null
}
return vue.h('span', {
ref: refocusRef,
class: 'no-outline',
tabindex: -1
})
});
function refocusTarget (e) {
const root = rootRef.value;
if (e !== void 0 && e.type.indexOf('key') === 0) {
if (
root !== null
&& document.activeElement !== root
&& root.contains(document.activeElement) === true
) {
root.focus();
}
}
else if (
refocusRef.value !== null
&& (e === void 0 || (root !== null && root.contains(e.target) === true))
) {
refocusRef.value.focus();
}
}
return {
refocusTargetEl,
refocusTarget
}
}
var optionSizes = {
xs: 30,
sm: 35,
md: 40,
lg: 50,
xl: 60
};
const useCheckboxProps = {
...useDarkProps,
...useSizeProps,
...useFormProps,
modelValue: {
required: true,
default: null
},
val: {},
trueValue: { default: true },
falseValue: { default: false },
indeterminateValue: { default: null },
checkedIcon: String,
uncheckedIcon: String,
indeterminateIcon: String,
toggleOrder: {
type: String,
validator: v => v === 'tf' || v === 'ft'
},
toggleIndeterminate: Boolean,
label: String,
leftLabel: Boolean,
color: String,
keepColor: Boolean,
dense: Boolean,
disable: Boolean,
tabindex: [ String, Number ]
};
const useCheckboxEmits = [ 'update:modelValue' ];
function useCheckbox (type, getInner) {
const { props, slots, emit, proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const rootRef = vue.ref(null);
const { refocusTargetEl, refocusTarget } = useRefocusTarget(props, rootRef);
const sizeStyle = useSize(props, optionSizes);
const modelIsArray = vue.computed(() =>
props.val !== void 0 && Array.isArray(props.modelValue)
);
const index = vue.computed(() => {
const val = vue.toRaw(props.val);
return modelIsArray.value === true
? props.modelValue.findIndex(opt => vue.toRaw(opt) === val)
: -1
});
const isTrue = vue.computed(() => (
modelIsArray.value === true
? index.value > -1
: vue.toRaw(props.modelValue) === vue.toRaw(props.trueValue)
));
const isFalse = vue.computed(() => (
modelIsArray.value === true
? index.value === -1
: vue.toRaw(props.modelValue) === vue.toRaw(props.falseValue)
));
const isIndeterminate = vue.computed(() =>
isTrue.value === false && isFalse.value === false
);
const tabindex = vue.computed(() => (
props.disable === true ? -1 : props.tabindex || 0
));
const classes = vue.computed(() =>
`q-${ type } cursor-pointer no-outline row inline no-wrap items-center`
+ (props.disable === true ? ' disabled' : '')
+ (isDark.value === true ? ` q-${ type }--dark` : '')
+ (props.dense === true ? ` q-${ type }--dense` : '')
+ (props.leftLabel === true ? ' reverse' : '')
);
const innerClass = vue.computed(() => {
const state = isTrue.value === true ? 'truthy' : (isFalse.value === true ? 'falsy' : 'indet');
const color = props.color !== void 0 && (
props.keepColor === true
|| (type === 'toggle' ? isTrue.value === true : isFalse.value !== true)
)
? ` text-${ props.color }`
: '';
return `q-${ type }__inner relative-position non-selectable q-${ type }__inner--${ state }${ color }`
});
const formAttrs = vue.computed(() => {
const prop = { type: 'checkbox' };
props.name !== void 0 && Object.assign(prop, {
// see https://vuejs.org/guide/extras/render-function.html#creating-vnodes (.prop)
'.checked': isTrue.value,
'^checked': isTrue.value === true ? 'checked' : void 0,
name: props.name,
value: modelIsArray.value === true
? props.val
: props.trueValue
});
return prop
});
const injectFormInput = useFormInject(formAttrs);
const attributes = vue.computed(() => {
const attrs = {
tabindex: tabindex.value,
role: type === 'toggle' ? 'switch' : 'checkbox',
'aria-label': props.label,
'aria-checked': isIndeterminate.value === true
? 'mixed'
: (isTrue.value === true ? 'true' : 'false')
};
if (props.disable === true) {
attrs[ 'aria-disabled' ] = 'true';
}
return attrs
});
function onClick (e) {
if (e !== void 0) {
stopAndPrevent(e);
refocusTarget(e);
}
if (props.disable !== true) {
emit('update:modelValue', getNextValue(), e);
}
}
function getNextValue () {
if (modelIsArray.value === true) {
if (isTrue.value === true) {
const val = props.modelValue.slice();
val.splice(index.value, 1);
return val
}
return props.modelValue.concat([ props.val ])
}
if (isTrue.value === true) {
if (props.toggleOrder !== 'ft' || props.toggleIndeterminate === false) {
return props.falseValue
}
}
else if (isFalse.value === true) {
if (props.toggleOrder === 'ft' || props.toggleIndeterminate === false) {
return props.trueValue
}
}
else {
return props.toggleOrder !== 'ft'
? props.trueValue
: props.falseValue
}
return props.indeterminateValue
}
function onKeydown (e) {
if (e.keyCode === 13 || e.keyCode === 32) {
stopAndPrevent(e);
}
}
function onKeyup (e) {
if (e.keyCode === 13 || e.keyCode === 32) {
onClick(e);
}
}
const getInnerContent = getInner(isTrue, isIndeterminate);
// expose public methods
Object.assign(proxy, { toggle: onClick });
return () => {
const inner = getInnerContent();
props.disable !== true && injectFormInput(
inner,
'unshift',
` q-${ type }__native absolute q-ma-none q-pa-none`
);
const child = [
vue.h('div', {
class: innerClass.value,
style: sizeStyle.value,
'aria-hidden': 'true'
}, inner)
];
if (refocusTargetEl.value !== null) {
child.push(refocusTargetEl.value);
}
const label = props.label !== void 0
? hMergeSlot(slots.default, [ props.label ])
: hSlot(slots.default);
label !== void 0 && child.push(
vue.h('div', {
class: `q-${ type }__label q-anchor--skip`
}, label)
);
return vue.h('div', {
ref: rootRef,
class: classes.value,
...attributes.value,
onClick,
onKeydown,
onKeyup
}, child)
}
}
const bgNode = vue.h('div', {
key: 'svg',
class: 'q-checkbox__bg absolute'
}, [
vue.h('svg', {
class: 'q-checkbox__svg fit absolute-full',
viewBox: '0 0 24 24'
}, [
vue.h('path', {
class: 'q-checkbox__truthy',
fill: 'none',
d: 'M1.73,12.91 8.1,19.28 22.79,4.59'
}),
vue.h('path', {
class: 'q-checkbox__indet',
d: 'M4,14H20V10H4'
})
])
]);
var QCheckbox = createComponent({
name: 'QCheckbox',
props: useCheckboxProps,
emits: useCheckboxEmits,
setup (props) {
function getInner (isTrue, isIndeterminate) {
const icon = vue.computed(() =>
(isTrue.value === true
? props.checkedIcon
: (isIndeterminate.value === true
? props.indeterminateIcon
: props.uncheckedIcon
)
) || null
);
return () => (
icon.value !== null
? [
vue.h('div', {
key: 'icon',
class: 'q-checkbox__icon-container absolute-full flex flex-center no-wrap'
}, [
vue.h(QIcon, {
class: 'q-checkbox__icon',
name: icon.value
})
])
]
: [ bgNode ]
)
}
return useCheckbox('checkbox', getInner)
}
});
const defaultSizes$1 = {
xs: 8,
sm: 10,
md: 14,
lg: 20,
xl: 24
};
var QChip = createComponent({
name: 'QChip',
props: {
...useDarkProps,
...useSizeProps,
dense: Boolean,
icon: String,
iconRight: String,
iconRemove: String,
iconSelected: String,
label: [ String, Number ],
color: String,
textColor: String,
modelValue: {
type: Boolean,
default: true
},
selected: {
type: Boolean,
default: null
},
square: Boolean,
outline: Boolean,
clickable: Boolean,
removable: Boolean,
removeAriaLabel: String,
tabindex: [ String, Number ],
disable: Boolean,
ripple: {
type: [ Boolean, Object ],
default: true
}
},
emits: [ 'update:modelValue', 'update:selected', 'remove', 'click' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const sizeStyle = useSize(props, defaultSizes$1);
const hasLeftIcon = vue.computed(() => props.selected === true || props.icon !== void 0);
const leftIcon = vue.computed(() => (
props.selected === true
? props.iconSelected || $q.iconSet.chip.selected
: props.icon
));
const removeIcon = vue.computed(() => props.iconRemove || $q.iconSet.chip.remove);
const isClickable = vue.computed(() =>
props.disable === false
&& (props.clickable === true || props.selected !== null)
);
const classes = vue.computed(() => {
const text = props.outline === true
? props.color || props.textColor
: props.textColor;
return 'q-chip row inline no-wrap items-center'
+ (props.outline === false && props.color !== void 0 ? ` bg-${ props.color }` : '')
+ (text ? ` text-${ text } q-chip--colored` : '')
+ (props.disable === true ? ' disabled' : '')
+ (props.dense === true ? ' q-chip--dense' : '')
+ (props.outline === true ? ' q-chip--outline' : '')
+ (props.selected === true ? ' q-chip--selected' : '')
+ (isClickable.value === true ? ' q-chip--clickable cursor-pointer non-selectable q-hoverable' : '')
+ (props.square === true ? ' q-chip--square' : '')
+ (isDark.value === true ? ' q-chip--dark q-dark' : '')
});
const attributes = vue.computed(() => {
const chip = props.disable === true
? { tabindex: -1, 'aria-disabled': 'true' }
: { tabindex: props.tabindex || 0 };
const remove = {
...chip,
role: 'button',
'aria-hidden': 'false',
'aria-label': props.removeAriaLabel || $q.lang.label.remove
};
return { chip, remove }
});
function onKeyup (e) {
e.keyCode === 13 /* ENTER */ && onClick(e);
}
function onClick (e) {
if (!props.disable) {
emit('update:selected', !props.selected);
emit('click', e);
}
}
function onRemove (e) {
if (e.keyCode === void 0 || e.keyCode === 13) {
stopAndPrevent(e);
if (props.disable === false) {
emit('update:modelValue', false);
emit('remove');
}
}
}
function getContent () {
const child = [];
isClickable.value === true && child.push(
vue.h('div', { class: 'q-focus-helper' })
);
hasLeftIcon.value === true && child.push(
vue.h(QIcon, {
class: 'q-chip__icon q-chip__icon--left',
name: leftIcon.value
})
);
const label = props.label !== void 0
? [ vue.h('div', { class: 'ellipsis' }, [ props.label ]) ]
: void 0;
child.push(
vue.h('div', {
class: 'q-chip__content col row no-wrap items-center q-anchor--skip'
}, hMergeSlotSafely(slots.default, label))
);
props.iconRight && child.push(
vue.h(QIcon, {
class: 'q-chip__icon q-chip__icon--right',
name: props.iconRight
})
);
props.removable === true && child.push(
vue.h(QIcon, {
class: 'q-chip__icon q-chip__icon--remove cursor-pointer',
name: removeIcon.value,
...attributes.value.remove,
onClick: onRemove,
onKeyup: onRemove
})
);
return child
}
return () => {
if (props.modelValue === false) { return }
const data = {
class: classes.value,
style: sizeStyle.value
};
isClickable.value === true && Object.assign(
data,
attributes.value.chip,
{ onClick, onKeyup }
);
return hDir(
'div',
data,
getContent(),
'ripple',
props.ripple !== false && props.disable !== true,
() => [ [ Ripple, props.ripple ] ]
)
}
}
});
// also used by QKnob
const useCircularCommonProps = {
...useSizeProps,
min: {
type: Number,
default: 0
},
max: {
type: Number,
default: 100
},
color: String,
centerColor: String,
trackColor: String,
fontSize: String,
rounded: Boolean,
// ratio
thickness: {
type: Number,
default: 0.2,
validator: v => v >= 0 && v <= 1
},
angle: {
type: Number,
default: 0
},
showValue: Boolean,
reverse: Boolean,
instantFeedback: Boolean
};
const
radius = 50,
diameter = 2 * radius,
circumference = diameter * Math.PI,
strokeDashArray = Math.round(circumference * 1000) / 1000;
var QCircularProgress = createComponent({
name: 'QCircularProgress',
props: {
...useCircularCommonProps,
value: {
type: Number,
default: 0
},
animationSpeed: {
type: [ String, Number ],
default: 600
},
indeterminate: Boolean
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const sizeStyle = useSize(props);
const svgStyle = vue.computed(() => {
const angle = ($q.lang.rtl === true ? -1 : 1) * props.angle;
return {
transform: props.reverse !== ($q.lang.rtl === true)
? `scale3d(-1, 1, 1) rotate3d(0, 0, 1, ${ -90 - angle }deg)`
: `rotate3d(0, 0, 1, ${ angle - 90 }deg)`
}
});
const circleStyle = vue.computed(() => (
props.instantFeedback !== true && props.indeterminate !== true
? { transition: `stroke-dashoffset ${ props.animationSpeed }ms ease 0s, stroke ${ props.animationSpeed }ms ease` }
: ''
));
const viewBox = vue.computed(() => diameter / (1 - props.thickness / 2));
const viewBoxAttr = vue.computed(() =>
`${ viewBox.value / 2 } ${ viewBox.value / 2 } ${ viewBox.value } ${ viewBox.value }`
);
const normalized = vue.computed(() => between(props.value, props.min, props.max));
const strokeDashOffset = vue.computed(() => circumference * (
1 - (normalized.value - props.min) / (props.max - props.min)
));
const strokeWidth = vue.computed(() => props.thickness / 2 * viewBox.value);
function getCircle ({ thickness, offset, color, cls, rounded }) {
return vue.h('circle', {
class: 'q-circular-progress__' + cls + (color !== void 0 ? ` text-${ color }` : ''),
style: circleStyle.value,
fill: 'transparent',
stroke: 'currentColor',
'stroke-width': thickness,
'stroke-dasharray': strokeDashArray,
'stroke-dashoffset': offset,
'stroke-linecap': rounded,
cx: viewBox.value,
cy: viewBox.value,
r: radius
})
}
return () => {
const svgChild = [];
props.centerColor !== void 0 && props.centerColor !== 'transparent' && svgChild.push(
vue.h('circle', {
class: `q-circular-progress__center text-${ props.centerColor }`,
fill: 'currentColor',
r: radius - strokeWidth.value / 2,
cx: viewBox.value,
cy: viewBox.value
})
);
props.trackColor !== void 0 && props.trackColor !== 'transparent' && svgChild.push(
getCircle({
cls: 'track',
thickness: strokeWidth.value,
offset: 0,
color: props.trackColor
})
);
svgChild.push(
getCircle({
cls: 'circle',
thickness: strokeWidth.value,
offset: strokeDashOffset.value,
color: props.color,
rounded: props.rounded === true ? 'round' : void 0
})
);
const child = [
vue.h('svg', {
class: 'q-circular-progress__svg',
style: svgStyle.value,
viewBox: viewBoxAttr.value,
'aria-hidden': 'true'
}, svgChild)
];
props.showValue === true && child.push(
vue.h('div', {
class: 'q-circular-progress__text absolute-full row flex-center content-center',
style: { fontSize: props.fontSize }
}, slots.default !== void 0 ? slots.default() : [ vue.h('div', normalized.value) ])
);
return vue.h('div', {
class: `q-circular-progress q-circular-progress--${ props.indeterminate === true ? 'in' : '' }determinate`,
style: sizeStyle.value,
role: 'progressbar',
'aria-valuemin': props.min,
'aria-valuemax': props.max,
'aria-valuenow': props.indeterminate === true ? void 0 : normalized.value
}, hMergeSlotSafely(slots.internal, child)) // "internal" is used by QKnob
}
}
});
function getChanges (evt, ctx, isFinal) {
const pos = position(evt);
let
dir,
distX = pos.left - ctx.event.x,
distY = pos.top - ctx.event.y,
absX = Math.abs(distX),
absY = Math.abs(distY);
const direction = ctx.direction;
if (direction.horizontal === true && direction.vertical !== true) {
dir = distX < 0 ? 'left' : 'right';
}
else if (direction.horizontal !== true && direction.vertical === true) {
dir = distY < 0 ? 'up' : 'down';
}
else if (direction.up === true && distY < 0) {
dir = 'up';
if (absX > absY) {
if (direction.left === true && distX < 0) {
dir = 'left';
}
else if (direction.right === true && distX > 0) {
dir = 'right';
}
}
}
else if (direction.down === true && distY > 0) {
dir = 'down';
if (absX > absY) {
if (direction.left === true && distX < 0) {
dir = 'left';
}
else if (direction.right === true && distX > 0) {
dir = 'right';
}
}
}
else if (direction.left === true && distX < 0) {
dir = 'left';
if (absX < absY) {
if (direction.up === true && distY < 0) {
dir = 'up';
}
else if (direction.down === true && distY > 0) {
dir = 'down';
}
}
}
else if (direction.right === true && distX > 0) {
dir = 'right';
if (absX < absY) {
if (direction.up === true && distY < 0) {
dir = 'up';
}
else if (direction.down === true && distY > 0) {
dir = 'down';
}
}
}
let synthetic = false;
if (dir === void 0 && isFinal === false) {
if (ctx.event.isFirst === true || ctx.event.lastDir === void 0) {
return {}
}
dir = ctx.event.lastDir;
synthetic = true;
if (dir === 'left' || dir === 'right') {
pos.left -= distX;
absX = 0;
distX = 0;
}
else {
pos.top -= distY;
absY = 0;
distY = 0;
}
}
return {
synthetic,
payload: {
evt,
touch: ctx.event.mouse !== true,
mouse: ctx.event.mouse === true,
position: pos,
direction: dir,
isFirst: ctx.event.isFirst,
isFinal: isFinal === true,
duration: Date.now() - ctx.event.time,
distance: {
x: absX,
y: absY
},
offset: {
x: distX,
y: distY
},
delta: {
x: pos.left - ctx.event.lastX,
y: pos.top - ctx.event.lastY
}
}
}
}
let uid$2 = 0;
var TouchPan = createDirective({
name: 'touch-pan',
beforeMount (el, { value, modifiers }) {
// early return, we don't need to do anything
if (modifiers.mouse !== true && client.has.touch !== true) {
return
}
function handleEvent (evt, mouseEvent) {
if (modifiers.mouse === true && mouseEvent === true) {
stopAndPrevent(evt);
}
else {
modifiers.stop === true && stop(evt);
modifiers.prevent === true && prevent(evt);
}
}
const ctx = {
uid: 'qvtp_' + (uid$2++),
handler: value,
modifiers,
direction: getModifierDirections(modifiers),
noop,
mouseStart (evt) {
if (shouldStart(evt, ctx) && leftClick(evt)) {
addEvt(ctx, 'temp', [
[ document, 'mousemove', 'move', 'notPassiveCapture' ],
[ document, 'mouseup', 'end', 'passiveCapture' ]
]);
ctx.start(evt, true);
}
},
touchStart (evt) {
if (shouldStart(evt, ctx)) {
const target = evt.target;
addEvt(ctx, 'temp', [
[ target, 'touchmove', 'move', 'notPassiveCapture' ],
[ target, 'touchcancel', 'end', 'passiveCapture' ],
[ target, 'touchend', 'end', 'passiveCapture' ]
]);
ctx.start(evt);
}
},
start (evt, mouseEvent) {
client.is.firefox === true && preventDraggable(el, true);
ctx.lastEvt = evt;
/*
* Stop propagation so possible upper v-touch-pan don't catch this as well;
* If we're not the target (based on modifiers), we'll re-emit the event later
*/
if (mouseEvent === true || modifiers.stop === true) {
/*
* are we directly switching to detected state?
* clone event only otherwise
*/
if (
ctx.direction.all !== true
// account for UMD too where modifiers will be lowercased to work
&& (mouseEvent !== true || (ctx.modifiers.mouseAllDir !== true && ctx.modifiers.mousealldir !== true))
) {
const clone = evt.type.indexOf('mouse') > -1
? new MouseEvent(evt.type, evt)
: new TouchEvent(evt.type, evt);
evt.defaultPrevented === true && prevent(clone);
evt.cancelBubble === true && stop(clone);
Object.assign(clone, {
qKeyEvent: evt.qKeyEvent,
qClickOutside: evt.qClickOutside,
qAnchorHandled: evt.qAnchorHandled,
qClonedBy: evt.qClonedBy === void 0
? [ ctx.uid ]
: evt.qClonedBy.concat(ctx.uid)
});
ctx.initialEvent = {
target: evt.target,
event: clone
};
}
stop(evt);
}
const { left, top } = position(evt);
ctx.event = {
x: left,
y: top,
time: Date.now(),
mouse: mouseEvent === true,
detected: false,
isFirst: true,
isFinal: false,
lastX: left,
lastY: top
};
},
move (evt) {
if (ctx.event === void 0) {
return
}
const
pos = position(evt),
distX = pos.left - ctx.event.x,
distY = pos.top - ctx.event.y;
// prevent buggy browser behavior (like Blink-based engine ones on Windows)
// where the mousemove event occurs even if there's no movement after mousedown
// https://bugs.chromium.org/p/chromium/issues/detail?id=161464
// https://bugs.chromium.org/p/chromium/issues/detail?id=721341
// https://github.com/quasarframework/quasar/issues/10721
if (distX === 0 && distY === 0) {
return
}
ctx.lastEvt = evt;
const isMouseEvt = ctx.event.mouse === true;
const start = () => {
handleEvent(evt, isMouseEvt);
let cursor;
if (modifiers.preserveCursor !== true && modifiers.preservecursor !== true) {
cursor = document.documentElement.style.cursor || '';
document.documentElement.style.cursor = 'grabbing';
}
isMouseEvt === true && document.body.classList.add('no-pointer-events--children');
document.body.classList.add('non-selectable');
clearSelection();
ctx.styleCleanup = withDelayedFn => {
ctx.styleCleanup = void 0;
if (cursor !== void 0) {
document.documentElement.style.cursor = cursor;
}
document.body.classList.remove('non-selectable');
if (isMouseEvt === true) {
const remove = () => {
document.body.classList.remove('no-pointer-events--children');
};
if (withDelayedFn !== void 0) {
setTimeout(() => {
remove();
withDelayedFn();
}, 50);
}
else { remove(); }
}
else if (withDelayedFn !== void 0) {
withDelayedFn();
}
};
};
if (ctx.event.detected === true) {
ctx.event.isFirst !== true && handleEvent(evt, ctx.event.mouse);
const { payload, synthetic } = getChanges(evt, ctx, false);
if (payload !== void 0) {
if (ctx.handler(payload) === false) {
ctx.end(evt);
}
else {
if (ctx.styleCleanup === void 0 && ctx.event.isFirst === true) {
start();
}
ctx.event.lastX = payload.position.left;
ctx.event.lastY = payload.position.top;
ctx.event.lastDir = synthetic === true ? void 0 : payload.direction;
ctx.event.isFirst = false;
}
}
return
}
if (
ctx.direction.all === true
// account for UMD too where modifiers will be lowercased to work
|| (isMouseEvt === true && (ctx.modifiers.mouseAllDir === true || ctx.modifiers.mousealldir === true))
) {
start();
ctx.event.detected = true;
ctx.move(evt);
return
}
const
absX = Math.abs(distX),
absY = Math.abs(distY);
if (absX !== absY) {
if (
(ctx.direction.horizontal === true && absX > absY)
|| (ctx.direction.vertical === true && absX < absY)
|| (ctx.direction.up === true && absX < absY && distY < 0)
|| (ctx.direction.down === true && absX < absY && distY > 0)
|| (ctx.direction.left === true && absX > absY && distX < 0)
|| (ctx.direction.right === true && absX > absY && distX > 0)
) {
ctx.event.detected = true;
ctx.move(evt);
}
else {
ctx.end(evt, true);
}
}
},
end (evt, abort) {
if (ctx.event === void 0) {
return
}
cleanEvt(ctx, 'temp');
client.is.firefox === true && preventDraggable(el, false);
if (abort === true) {
ctx.styleCleanup !== void 0 && ctx.styleCleanup();
if (ctx.event.detected !== true && ctx.initialEvent !== void 0) {
ctx.initialEvent.target.dispatchEvent(ctx.initialEvent.event);
}
}
else if (ctx.event.detected === true) {
ctx.event.isFirst === true && ctx.handler(getChanges(evt === void 0 ? ctx.lastEvt : evt, ctx).payload);
const { payload } = getChanges(evt === void 0 ? ctx.lastEvt : evt, ctx, true);
const fn = () => { ctx.handler(payload); };
if (ctx.styleCleanup !== void 0) {
ctx.styleCleanup(fn);
}
else {
fn();
}
}
ctx.event = void 0;
ctx.initialEvent = void 0;
ctx.lastEvt = void 0;
}
};
el.__qtouchpan = ctx;
if (modifiers.mouse === true) {
// account for UMD too where modifiers will be lowercased to work
const capture = modifiers.mouseCapture === true || modifiers.mousecapture === true
? 'Capture'
: '';
addEvt(ctx, 'main', [
[ el, 'mousedown', 'mouseStart', `passive${ capture }` ]
]);
}
client.has.touch === true && addEvt(ctx, 'main', [
[ el, 'touchstart', 'touchStart', `passive${ modifiers.capture === true ? 'Capture' : '' }` ],
[ el, 'touchmove', 'noop', 'notPassiveCapture' ] // cannot be passive (ex: iOS scroll)
]);
},
updated (el, bindings) {
const ctx = el.__qtouchpan;
if (ctx !== void 0) {
if (bindings.oldValue !== bindings.value) {
typeof value !== 'function' && ctx.end();
ctx.handler = bindings.value;
}
ctx.direction = getModifierDirections(bindings.modifiers);
}
},
beforeUnmount (el) {
const ctx = el.__qtouchpan;
if (ctx !== void 0) {
// emit the end event when the directive is destroyed while active
// this is only needed in TouchPan because the rest of the touch directives do not emit an end event
// the condition is also checked in the start of function but we avoid the call
ctx.event !== void 0 && ctx.end();
cleanEvt(ctx, 'main');
cleanEvt(ctx, 'temp');
client.is.firefox === true && preventDraggable(el, false);
ctx.styleCleanup !== void 0 && ctx.styleCleanup();
delete el.__qtouchpan;
}
}
}
);
const markerPrefixClass = 'q-slider__marker-labels';
const defaultMarkerConvertFn = v => ({ value: v });
const defaultMarkerLabelRenderFn = ({ marker }) => vue.h('div', {
key: marker.value,
style: marker.style,
class: marker.classes
}, marker.label);
// PGDOWN, LEFT, DOWN, PGUP, RIGHT, UP
const keyCodes$2 = [ 34, 37, 40, 33, 39, 38 ];
const useSliderProps = {
...useDarkProps,
...useFormProps,
min: {
type: Number,
default: 0
},
max: {
type: Number,
default: 100
},
innerMin: Number,
innerMax: Number,
step: {
type: Number,
default: 1,
validator: v => v >= 0
},
snap: Boolean,
vertical: Boolean,
reverse: Boolean,
hideSelection: Boolean,
color: String,
markerLabelsClass: String,
label: Boolean,
labelColor: String,
labelTextColor: String,
labelAlways: Boolean,
switchLabelSide: Boolean,
markers: [ Boolean, Number ],
markerLabels: [ Boolean, Array, Object, Function ],
switchMarkerLabelsSide: Boolean,
trackImg: String,
trackColor: String,
innerTrackImg: String,
innerTrackColor: String,
selectionColor: String,
selectionImg: String,
thumbSize: {
type: String,
default: '20px'
},
trackSize: {
type: String,
default: '4px'
},
disable: Boolean,
readonly: Boolean,
dense: Boolean,
tabindex: [ String, Number ],
thumbColor: String,
thumbPath: {
type: String,
default: 'M 4, 10 a 6,6 0 1,0 12,0 a 6,6 0 1,0 -12,0'
}
};
const useSliderEmits = [ 'pan', 'update:modelValue', 'change' ];
function useSlider ({ updateValue, updatePosition, getDragging, formAttrs }) {
const { props, emit, slots, proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const injectFormInput = useFormInject(formAttrs);
const active = vue.ref(false);
const preventFocus = vue.ref(false);
const focus = vue.ref(false);
const dragging = vue.ref(false);
const axis = vue.computed(() => (props.vertical === true ? '--v' : '--h'));
const labelSide = vue.computed(() => '-' + (props.switchLabelSide === true ? 'switched' : 'standard'));
const isReversed = vue.computed(() => (
props.vertical === true
? props.reverse === true
: props.reverse !== ($q.lang.rtl === true)
));
const innerMin = vue.computed(() => (
isNaN(props.innerMin) === true || props.innerMin < props.min
? props.min
: props.innerMin
));
const innerMax = vue.computed(() => (
isNaN(props.innerMax) === true || props.innerMax > props.max
? props.max
: props.innerMax
));
const editable = vue.computed(() => (
props.disable !== true && props.readonly !== true
&& innerMin.value < innerMax.value
));
const decimals = vue.computed(() => (String(props.step).trim().split('.')[ 1 ] || '').length);
const step = vue.computed(() => (props.step === 0 ? 1 : props.step));
const tabindex = vue.computed(() => (editable.value === true ? props.tabindex || 0 : -1));
const trackLen = vue.computed(() => props.max - props.min);
const innerBarLen = vue.computed(() => innerMax.value - innerMin.value);
const innerMinRatio = vue.computed(() => convertModelToRatio(innerMin.value));
const innerMaxRatio = vue.computed(() => convertModelToRatio(innerMax.value));
const positionProp = vue.computed(() => (
props.vertical === true
? (isReversed.value === true ? 'bottom' : 'top')
: (isReversed.value === true ? 'right' : 'left')
));
const sizeProp = vue.computed(() => (props.vertical === true ? 'height' : 'width'));
const thicknessProp = vue.computed(() => (props.vertical === true ? 'width' : 'height'));
const orientation = vue.computed(() => (props.vertical === true ? 'vertical' : 'horizontal'));
const attributes = vue.computed(() => {
const acc = {
role: 'slider',
'aria-valuemin': innerMin.value,
'aria-valuemax': innerMax.value,
'aria-orientation': orientation.value,
'data-step': props.step
};
if (props.disable === true) {
acc[ 'aria-disabled' ] = 'true';
}
else if (props.readonly === true) {
acc[ 'aria-readonly' ] = 'true';
}
return acc
});
const classes = vue.computed(() =>
`q-slider q-slider${ axis.value } q-slider--${ active.value === true ? '' : 'in' }active inline no-wrap `
+ (props.vertical === true ? 'row' : 'column')
+ (props.disable === true ? ' disabled' : ' q-slider--enabled' + (editable.value === true ? ' q-slider--editable' : ''))
+ (focus.value === 'both' ? ' q-slider--focus' : '')
+ (props.label || props.labelAlways === true ? ' q-slider--label' : '')
+ (props.labelAlways === true ? ' q-slider--label-always' : '')
+ (isDark.value === true ? ' q-slider--dark' : '')
+ (props.dense === true ? ' q-slider--dense q-slider--dense' + axis.value : '')
);
function getPositionClass (name) {
const cls = 'q-slider__' + name;
return `${ cls } ${ cls }${ axis.value } ${ cls }${ axis.value }${ labelSide.value }`
}
function getAxisClass (name) {
const cls = 'q-slider__' + name;
return `${ cls } ${ cls }${ axis.value }`
}
const selectionBarClass = vue.computed(() => {
const color = props.selectionColor || props.color;
return 'q-slider__selection absolute'
+ (color !== void 0 ? ` text-${ color }` : '')
});
const markerClass = vue.computed(() => getAxisClass('markers') + ' absolute overflow-hidden');
const trackContainerClass = vue.computed(() => getAxisClass('track-container'));
const pinClass = vue.computed(() => getPositionClass('pin'));
const labelClass = vue.computed(() => getPositionClass('label'));
const textContainerClass = vue.computed(() => getPositionClass('text-container'));
const markerLabelsContainerClass = vue.computed(() =>
getPositionClass('marker-labels-container')
+ (props.markerLabelsClass !== void 0 ? ` ${ props.markerLabelsClass }` : '')
);
const trackClass = vue.computed(() =>
'q-slider__track relative-position no-outline'
+ (props.trackColor !== void 0 ? ` bg-${ props.trackColor }` : '')
);
const trackStyle = vue.computed(() => {
const acc = { [ thicknessProp.value ]: props.trackSize };
if (props.trackImg !== void 0) {
acc.backgroundImage = `url(${ props.trackImg }) !important`;
}
return acc
});
const innerBarClass = vue.computed(() =>
'q-slider__inner absolute'
+ (props.innerTrackColor !== void 0 ? ` bg-${ props.innerTrackColor }` : '')
);
const innerBarStyle = vue.computed(() => {
const acc = {
[ positionProp.value ]: `${ 100 * innerMinRatio.value }%`,
[ sizeProp.value ]: `${ 100 * (innerMaxRatio.value - innerMinRatio.value) }%`
};
if (props.innerTrackImg !== void 0) {
acc.backgroundImage = `url(${ props.innerTrackImg }) !important`;
}
return acc
});
function convertRatioToModel (ratio) {
const { min, max, step } = props;
let model = min + ratio * (max - min);
if (step > 0) {
const modulo = (model - min) % step;
model += (Math.abs(modulo) >= step / 2 ? (modulo < 0 ? -1 : 1) * step : 0) - modulo;
}
if (decimals.value > 0) {
model = parseFloat(model.toFixed(decimals.value));
}
return between(model, innerMin.value, innerMax.value)
}
function convertModelToRatio (model) {
return trackLen.value === 0
? 0
: (model - props.min) / trackLen.value
}
function getDraggingRatio (evt, dragging) {
const
pos = position(evt),
val = props.vertical === true
? between((pos.top - dragging.top) / dragging.height, 0, 1)
: between((pos.left - dragging.left) / dragging.width, 0, 1);
return between(
isReversed.value === true ? 1.0 - val : val,
innerMinRatio.value,
innerMaxRatio.value
)
}
const markerStep = vue.computed(() => (
isNumber(props.markers) === true ? props.markers : step.value)
);
const markerTicks = vue.computed(() => {
const acc = [];
const step = markerStep.value;
const max = props.max;
let value = props.min;
do {
acc.push(value);
value += step;
} while (value < max)
acc.push(max);
return acc
});
const markerLabelClass = vue.computed(() => {
const prefix = ` ${ markerPrefixClass }${ axis.value }-`;
return markerPrefixClass
+ `${ prefix }${ props.switchMarkerLabelsSide === true ? 'switched' : 'standard' }`
+ `${ prefix }${ isReversed.value === true ? 'rtl' : 'ltr' }`
});
const markerLabelsList = vue.computed(() => {
if (props.markerLabels === false) { return null }
return getMarkerList(props.markerLabels).map((entry, index) => ({
index,
value: entry.value,
label: entry.label || entry.value,
classes: markerLabelClass.value
+ (entry.classes !== void 0 ? ' ' + entry.classes : ''),
style: {
...getMarkerLabelStyle(entry.value),
...(entry.style || {})
}
}))
});
const markerScope = vue.computed(() => ({
markerList: markerLabelsList.value,
markerMap: markerLabelsMap.value,
classes: markerLabelClass.value, // TODO ts definition
getStyle: getMarkerLabelStyle
}));
const markerStyle = vue.computed(() => {
if (innerBarLen.value !== 0) {
const size = 100 * markerStep.value / innerBarLen.value;
return {
...innerBarStyle.value,
backgroundSize: props.vertical === true
? `2px ${ size }%`
: `${ size }% 2px`
}
}
return null
});
function getMarkerList (def) {
if (def === false) { return null }
if (def === true) {
return markerTicks.value.map(defaultMarkerConvertFn)
}
if (typeof def === 'function') {
return markerTicks.value.map(value => {
const item = def(value);
return isObject(item) === true ? { ...item, value } : { value, label: item }
})
}
const filterFn = ({ value }) => value >= props.min && value <= props.max;
if (Array.isArray(def) === true) {
return def
.map(item => (isObject(item) === true ? item : { value: item }))
.filter(filterFn)
}
return Object.keys(def).map(key => {
const item = def[ key ];
const value = Number(key);
return isObject(item) === true ? { ...item, value } : { value, label: item }
}).filter(filterFn)
}
function getMarkerLabelStyle (val) {
return { [ positionProp.value ]: `${ 100 * (val - props.min) / trackLen.value }%` }
}
const markerLabelsMap = vue.computed(() => {
if (props.markerLabels === false) { return null }
const acc = {};
markerLabelsList.value.forEach(entry => {
acc[ entry.value ] = entry;
});
return acc
});
function getMarkerLabelsContent () {
if (slots[ 'marker-label-group' ] !== void 0) {
return slots[ 'marker-label-group' ](markerScope.value)
}
const fn = slots[ 'marker-label' ] || defaultMarkerLabelRenderFn;
return markerLabelsList.value.map(marker => fn({
marker,
...markerScope.value
}))
}
const panDirective = vue.computed(() => {
// if editable.value === true
return [ [
TouchPan,
onPan,
void 0,
{
[ orientation.value ]: true,
prevent: true,
stop: true,
mouse: true,
mouseAllDir: true
}
] ]
});
function onPan (event) {
if (event.isFinal === true) {
if (dragging.value !== void 0) {
updatePosition(event.evt);
// only if touch, because we also have mousedown/up:
event.touch === true && updateValue(true);
dragging.value = void 0;
emit('pan', 'end');
}
active.value = false;
focus.value = false;
}
else if (event.isFirst === true) {
dragging.value = getDragging(event.evt);
updatePosition(event.evt);
updateValue();
active.value = true;
emit('pan', 'start');
}
else {
updatePosition(event.evt);
updateValue();
}
}
function onBlur () {
focus.value = false;
}
function onActivate (evt) {
updatePosition(evt, getDragging(evt));
updateValue();
preventFocus.value = true;
active.value = true;
document.addEventListener('mouseup', onDeactivate, true);
}
function onDeactivate () {
preventFocus.value = false;
active.value = false;
updateValue(true);
onBlur();
document.removeEventListener('mouseup', onDeactivate, true);
}
function onMobileClick (evt) {
updatePosition(evt, getDragging(evt));
updateValue(true);
}
function onKeyup (evt) {
if (keyCodes$2.includes(evt.keyCode)) {
updateValue(true);
}
}
function getTextContainerStyle (ratio) {
if (props.vertical === true) { return null }
const p = $q.lang.rtl !== props.reverse ? 1 - ratio : ratio;
return {
transform: `translateX(calc(${ 2 * p - 1 } * ${ props.thumbSize } / 2 + ${ 50 - 100 * p }%))`
}
}
function getThumbRenderFn (thumb) {
const focusClass = vue.computed(() => (
preventFocus.value === false && (focus.value === thumb.focusValue || focus.value === 'both')
? ' q-slider--focus'
: ''
));
const classes = vue.computed(() =>
`q-slider__thumb q-slider__thumb${ axis.value } q-slider__thumb${ axis.value }-${ isReversed.value === true ? 'rtl' : 'ltr' } absolute non-selectable`
+ focusClass.value
+ (thumb.thumbColor.value !== void 0 ? ` text-${ thumb.thumbColor.value }` : '')
);
const style = vue.computed(() => ({
width: props.thumbSize,
height: props.thumbSize,
[ positionProp.value ]: `${ 100 * thumb.ratio.value }%`,
zIndex: focus.value === thumb.focusValue ? 2 : void 0
}));
const pinColor = vue.computed(() => (
thumb.labelColor.value !== void 0
? ` text-${ thumb.labelColor.value }`
: ''
));
const textContainerStyle = vue.computed(() => getTextContainerStyle(thumb.ratio.value));
const textClass = vue.computed(() => (
'q-slider__text'
+ (thumb.labelTextColor.value !== void 0 ? ` text-${ thumb.labelTextColor.value }` : '')
));
return () => {
const thumbContent = [
vue.h('svg', {
class: 'q-slider__thumb-shape absolute-full',
viewBox: '0 0 20 20',
'aria-hidden': 'true'
}, [
vue.h('path', { d: props.thumbPath })
]),
vue.h('div', { class: 'q-slider__focus-ring fit' })
];
if (props.label === true || props.labelAlways === true) {
thumbContent.push(
vue.h('div', {
class: pinClass.value + ' absolute fit no-pointer-events' + pinColor.value
}, [
vue.h('div', {
class: labelClass.value,
style: { minWidth: props.thumbSize }
}, [
vue.h('div', {
class: textContainerClass.value,
style: textContainerStyle.value
}, [
vue.h('span', { class: textClass.value }, thumb.label.value)
])
])
])
);
if (props.name !== void 0 && props.disable !== true) {
injectFormInput(thumbContent, 'push');
}
}
return vue.h('div', {
class: classes.value,
style: style.value,
...thumb.getNodeData()
}, thumbContent)
}
}
function getContent (selectionBarStyle, trackContainerTabindex, trackContainerEvents, injectThumb) {
const trackContent = [];
props.innerTrackColor !== 'transparent' && trackContent.push(
vue.h('div', {
key: 'inner',
class: innerBarClass.value,
style: innerBarStyle.value
})
);
props.selectionColor !== 'transparent' && trackContent.push(
vue.h('div', {
key: 'selection',
class: selectionBarClass.value,
style: selectionBarStyle.value
})
);
props.markers !== false && trackContent.push(
vue.h('div', {
key: 'marker',
class: markerClass.value,
style: markerStyle.value
})
);
injectThumb(trackContent);
const content = [
hDir(
'div',
{
key: 'trackC',
class: trackContainerClass.value,
tabindex: trackContainerTabindex.value,
...trackContainerEvents.value
},
[
vue.h('div', {
class: trackClass.value,
style: trackStyle.value
}, trackContent)
],
'slide',
editable.value, () => panDirective.value
)
];
if (props.markerLabels !== false) {
const action = props.switchMarkerLabelsSide === true
? 'unshift'
: 'push';
content[ action ](
vue.h('div', {
key: 'markerL',
class: markerLabelsContainerClass.value
}, getMarkerLabelsContent())
);
}
return content
}
vue.onBeforeUnmount(() => {
document.removeEventListener('mouseup', onDeactivate, true);
});
return {
state: {
active,
focus,
preventFocus,
dragging,
editable,
classes,
tabindex,
attributes,
step,
decimals,
trackLen,
innerMin,
innerMinRatio,
innerMax,
innerMaxRatio,
positionProp,
sizeProp,
isReversed
},
methods: {
onActivate,
onMobileClick,
onBlur,
onKeyup,
getContent,
getThumbRenderFn,
convertRatioToModel,
convertModelToRatio,
getDraggingRatio
}
}
}
const getNodeData = () => ({});
var QSlider = createComponent({
name: 'QSlider',
props: {
...useSliderProps,
modelValue: {
required: true,
default: null,
validator: v => typeof v === 'number' || v === null
},
labelValue: [ String, Number ]
},
emits: useSliderEmits,
setup (props, { emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const { state, methods } = useSlider({
updateValue, updatePosition, getDragging,
formAttrs: useFormAttrs(props)
});
const rootRef = vue.ref(null);
const curRatio = vue.ref(0);
const model = vue.ref(0);
function normalizeModel () {
model.value = props.modelValue === null
? state.innerMin.value
: between(props.modelValue, state.innerMin.value, state.innerMax.value);
}
vue.watch(
() => `${ props.modelValue }|${ state.innerMin.value }|${ state.innerMax.value }`,
normalizeModel
);
normalizeModel();
const modelRatio = vue.computed(() => methods.convertModelToRatio(model.value));
const ratio = vue.computed(() => (state.active.value === true ? curRatio.value : modelRatio.value));
const selectionBarStyle = vue.computed(() => {
const acc = {
[ state.positionProp.value ]: `${ 100 * state.innerMinRatio.value }%`,
[ state.sizeProp.value ]: `${ 100 * (ratio.value - state.innerMinRatio.value) }%`
};
if (props.selectionImg !== void 0) {
acc.backgroundImage = `url(${ props.selectionImg }) !important`;
}
return acc
});
const getThumb = methods.getThumbRenderFn({
focusValue: true,
getNodeData,
ratio,
label: vue.computed(() => (
props.labelValue !== void 0
? props.labelValue
: model.value
)),
thumbColor: vue.computed(() => props.thumbColor || props.color),
labelColor: vue.computed(() => props.labelColor),
labelTextColor: vue.computed(() => props.labelTextColor)
});
const trackContainerEvents = vue.computed(() => {
if (state.editable.value !== true) {
return {}
}
return $q.platform.is.mobile === true
? { onClick: methods.onMobileClick }
: {
onMousedown: methods.onActivate,
onFocus,
onBlur: methods.onBlur,
onKeydown,
onKeyup: methods.onKeyup
}
});
function updateValue (change) {
if (model.value !== props.modelValue) {
emit('update:modelValue', model.value);
}
change === true && emit('change', model.value);
}
function getDragging () {
return rootRef.value.getBoundingClientRect()
}
function updatePosition (event, dragging = state.dragging.value) {
const ratio = methods.getDraggingRatio(event, dragging);
model.value = methods.convertRatioToModel(ratio);
curRatio.value = props.snap !== true || props.step === 0
? ratio
: methods.convertModelToRatio(model.value);
}
function onFocus () {
state.focus.value = true;
}
function onKeydown (evt) {
if (!keyCodes$2.includes(evt.keyCode)) {
return
}
stopAndPrevent(evt);
const
stepVal = ([ 34, 33 ].includes(evt.keyCode) ? 10 : 1) * state.step.value,
offset = (
([ 34, 37, 40 ].includes(evt.keyCode) ? -1 : 1)
* (state.isReversed.value === true ? -1 : 1)
* (props.vertical === true ? -1 : 1) * stepVal
);
model.value = between(
parseFloat((model.value + offset).toFixed(state.decimals.value)),
state.innerMin.value,
state.innerMax.value
);
updateValue();
}
return () => {
const content = methods.getContent(
selectionBarStyle,
state.tabindex,
trackContainerEvents,
node => { node.push(getThumb()); }
);
return vue.h('div', {
ref: rootRef,
class: state.classes.value + (props.modelValue === null ? ' q-slider--no-value' : ''),
...state.attributes.value,
'aria-valuenow': props.modelValue
}, content)
}
}
});
function useCanRender () {
const canRender = vue.ref(!isRuntimeSsrPreHydration.value);
if (canRender.value === false) {
vue.onMounted(() => {
canRender.value = true;
});
}
return canRender
}
const hasObserver = typeof ResizeObserver !== 'undefined';
const resizeProps = hasObserver === true
? {}
: {
style: 'display:block;position:absolute;top:0;left:0;right:0;bottom:0;height:100%;width:100%;overflow:hidden;pointer-events:none;z-index:-1;',
url: 'about:blank'
};
var QResizeObserver = createComponent({
name: 'QResizeObserver',
props: {
debounce: {
type: [ String, Number ],
default: 100
}
},
emits: [ 'resize' ],
setup (props, { emit }) {
let timer = null, targetEl, size = { width: -1, height: -1 };
function trigger (immediately) {
if (immediately === true || props.debounce === 0 || props.debounce === '0') {
emitEvent();
}
else if (timer === null) {
timer = setTimeout(emitEvent, props.debounce);
}
}
function emitEvent () {
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
if (targetEl) {
const { offsetWidth: width, offsetHeight: height } = targetEl;
if (width !== size.width || height !== size.height) {
size = { width, height };
emit('resize', size);
}
}
}
const { proxy } = vue.getCurrentInstance();
if (hasObserver === true) {
let observer;
// initialize as soon as possible
const init = stop => {
targetEl = proxy.$el.parentNode;
if (targetEl) {
observer = new ResizeObserver(trigger);
observer.observe(targetEl);
emitEvent();
}
else if (stop !== true) {
vue.nextTick(() => { init(true); });
}
};
vue.onMounted(() => { init(); });
vue.onBeforeUnmount(() => {
timer !== null && clearTimeout(timer);
if (observer !== void 0) {
if (observer.disconnect !== void 0) {
observer.disconnect();
}
else if (targetEl) { // FF for Android
observer.unobserve(targetEl);
}
}
});
return noop
}
else { // no observer, so fallback to old iframe method
const canRender = useCanRender();
let curDocView;
function cleanup () {
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
if (curDocView !== void 0) {
// iOS is fuzzy, need to check it first
if (curDocView.removeEventListener !== void 0) {
curDocView.removeEventListener('resize', trigger, listenOpts.passive);
}
curDocView = void 0;
}
}
function onObjLoad () {
cleanup();
if (targetEl && targetEl.contentDocument) {
curDocView = targetEl.contentDocument.defaultView;
curDocView.addEventListener('resize', trigger, listenOpts.passive);
emitEvent();
}
}
vue.onMounted(() => {
vue.nextTick(() => {
targetEl = proxy.$el;
targetEl && onObjLoad();
});
});
vue.onBeforeUnmount(cleanup);
// expose public method
proxy.trigger = trigger;
return () => {
if (canRender.value === true) {
return vue.h('object', {
style: resizeProps.style,
tabindex: -1, // fix for Firefox
type: 'text/html',
data: resizeProps.url,
'aria-hidden': 'true',
onLoad: onObjLoad
})
}
}
}
}
});
let rtlHasScrollBug = false;
// mobile Chrome takes the crown for this
{
const scroller = document.createElement('div');
scroller.setAttribute('dir', 'rtl');
Object.assign(scroller.style, {
width: '1px',
height: '1px',
overflow: 'auto'
});
const spacer = document.createElement('div');
Object.assign(spacer.style, {
width: '1000px',
height: '1px'
});
document.body.appendChild(scroller);
scroller.appendChild(spacer);
scroller.scrollLeft = -1000;
rtlHasScrollBug = scroller.scrollLeft >= 0;
scroller.remove();
}
function getIndicatorClass (color, top, vertical) {
const pos = vertical === true
? [ 'left', 'right' ]
: [ 'top', 'bottom' ];
return `absolute-${ top === true ? pos[ 0 ] : pos[ 1 ] }${ color ? ` text-${ color }` : '' }`
}
const alignValues$1 = [ 'left', 'center', 'right', 'justify' ];
var QTabs = createComponent({
name: 'QTabs',
props: {
modelValue: [ Number, String ],
align: {
type: String,
default: 'center',
validator: v => alignValues$1.includes(v)
},
breakpoint: {
type: [ String, Number ],
default: 600
},
vertical: Boolean,
shrink: Boolean,
stretch: Boolean,
activeClass: String,
activeColor: String,
activeBgColor: String,
indicatorColor: String,
leftIcon: String,
rightIcon: String,
outsideArrows: Boolean,
mobileArrows: Boolean,
switchIndicator: Boolean,
narrowIndicator: Boolean,
inlineLabel: Boolean,
noCaps: Boolean,
dense: Boolean,
contentClass: String,
'onUpdate:modelValue': [ Function, Array ]
},
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const { registerTick: registerScrollTick } = useTick();
const { registerTick: registerUpdateArrowsTick } = useTick();
const { registerTick: registerAnimateTick } = useTick();
const { registerTimeout: registerFocusTimeout, removeTimeout: removeFocusTimeout } = useTimeout();
const { registerTimeout: registerScrollToTabTimeout, removeTimeout: removeScrollToTabTimeout } = useTimeout();
const rootRef = vue.ref(null);
const contentRef = vue.ref(null);
const currentModel = vue.ref(props.modelValue);
const scrollable = vue.ref(false);
const leftArrow = vue.ref(true);
const rightArrow = vue.ref(false);
const justify = vue.ref(false);
const tabDataList = [];
const tabDataListLen = vue.ref(0);
const hasFocus = vue.ref(false);
let animateTimer = null, scrollTimer = null, unwatchRoute;
const tabProps = vue.computed(() => ({
activeClass: props.activeClass,
activeColor: props.activeColor,
activeBgColor: props.activeBgColor,
indicatorClass: getIndicatorClass(
props.indicatorColor,
props.switchIndicator,
props.vertical
),
narrowIndicator: props.narrowIndicator,
inlineLabel: props.inlineLabel,
noCaps: props.noCaps
}));
const hasActiveTab = vue.computed(() => {
const len = tabDataListLen.value;
const val = currentModel.value;
for (let i = 0; i < len; i++) {
if (tabDataList[ i ].name.value === val) {
return true
}
}
return false
});
const alignClass = vue.computed(() => {
const align = scrollable.value === true
? 'left'
: (justify.value === true ? 'justify' : props.align);
return `q-tabs__content--align-${ align }`
});
const classes = vue.computed(() =>
'q-tabs row no-wrap items-center'
+ ` q-tabs--${ scrollable.value === true ? '' : 'not-' }scrollable`
+ ` q-tabs--${ props.vertical === true ? 'vertical' : 'horizontal' }`
+ ` q-tabs__arrows--${ props.outsideArrows === true ? 'outside' : 'inside' }`
+ ` q-tabs--mobile-with${ props.mobileArrows === true ? '' : 'out' }-arrows`
+ (props.dense === true ? ' q-tabs--dense' : '')
+ (props.shrink === true ? ' col-shrink' : '')
+ (props.stretch === true ? ' self-stretch' : '')
);
const innerClass = vue.computed(() =>
'q-tabs__content scroll--mobile row no-wrap items-center self-stretch hide-scrollbar relative-position '
+ alignClass.value
+ (props.contentClass !== void 0 ? ` ${ props.contentClass }` : '')
);
const domProps = vue.computed(() => (
props.vertical === true
? { container: 'height', content: 'offsetHeight', scroll: 'scrollHeight' }
: { container: 'width', content: 'offsetWidth', scroll: 'scrollWidth' }
));
const isRTL = vue.computed(() => props.vertical !== true && $q.lang.rtl === true);
const rtlPosCorrection = vue.computed(() => rtlHasScrollBug === false && isRTL.value === true);
vue.watch(isRTL, updateArrows);
vue.watch(() => props.modelValue, name => {
updateModel({ name, setCurrent: true, skipEmit: true });
});
vue.watch(() => props.outsideArrows, recalculateScroll);
function updateModel ({ name, setCurrent, skipEmit }) {
if (currentModel.value !== name) {
if (skipEmit !== true && props[ 'onUpdate:modelValue' ] !== void 0) {
emit('update:modelValue', name);
}
if (
setCurrent === true
|| props[ 'onUpdate:modelValue' ] === void 0
) {
animate(currentModel.value, name);
currentModel.value = name;
}
}
}
function recalculateScroll () {
registerScrollTick(() => {
updateContainer({
width: rootRef.value.offsetWidth,
height: rootRef.value.offsetHeight
});
});
}
function updateContainer (domSize) {
// it can be called faster than component being initialized
// so we need to protect against that case
// (one example of such case is the docs release notes page)
if (domProps.value === void 0 || contentRef.value === null) { return }
const
size = domSize[ domProps.value.container ],
scrollSize = Math.min(
contentRef.value[ domProps.value.scroll ],
Array.prototype.reduce.call(
contentRef.value.children,
(acc, el) => acc + (el[ domProps.value.content ] || 0),
0
)
),
scroll = size > 0 && scrollSize > size; // when there is no tab, in Chrome, size === 0 and scrollSize === 1
scrollable.value = scroll;
// Arrows need to be updated even if the scroll status was already true
scroll === true && registerUpdateArrowsTick(updateArrows);
justify.value = size < parseInt(props.breakpoint, 10);
}
function animate (oldName, newName) {
const
oldTab = oldName !== void 0 && oldName !== null && oldName !== ''
? tabDataList.find(tab => tab.name.value === oldName)
: null,
newTab = newName !== void 0 && newName !== null && newName !== ''
? tabDataList.find(tab => tab.name.value === newName)
: null;
if (oldTab && newTab) {
const
oldEl = oldTab.tabIndicatorRef.value,
newEl = newTab.tabIndicatorRef.value;
if (animateTimer !== null) {
clearTimeout(animateTimer);
animateTimer = null;
}
oldEl.style.transition = 'none';
oldEl.style.transform = 'none';
newEl.style.transition = 'none';
newEl.style.transform = 'none';
const
oldPos = oldEl.getBoundingClientRect(),
newPos = newEl.getBoundingClientRect();
newEl.style.transform = props.vertical === true
? `translate3d(0,${ oldPos.top - newPos.top }px,0) scale3d(1,${ newPos.height ? oldPos.height / newPos.height : 1 },1)`
: `translate3d(${ oldPos.left - newPos.left }px,0,0) scale3d(${ newPos.width ? oldPos.width / newPos.width : 1 },1,1)`;
// allow scope updates to kick in (QRouteTab needs more time)
registerAnimateTick(() => {
animateTimer = setTimeout(() => {
animateTimer = null;
newEl.style.transition = 'transform .25s cubic-bezier(.4, 0, .2, 1)';
newEl.style.transform = 'none';
}, 70);
});
}
if (newTab && scrollable.value === true) {
scrollToTabEl(newTab.rootRef.value);
}
}
function scrollToTabEl (el) {
const
{ left, width, top, height } = contentRef.value.getBoundingClientRect(),
newPos = el.getBoundingClientRect();
let offset = props.vertical === true ? newPos.top - top : newPos.left - left;
if (offset < 0) {
contentRef.value[ props.vertical === true ? 'scrollTop' : 'scrollLeft' ] += Math.floor(offset);
updateArrows();
return
}
offset += props.vertical === true ? newPos.height - height : newPos.width - width;
if (offset > 0) {
contentRef.value[ props.vertical === true ? 'scrollTop' : 'scrollLeft' ] += Math.ceil(offset);
updateArrows();
}
}
function updateArrows () {
const content = contentRef.value;
if (content === null) { return }
const
rect = content.getBoundingClientRect(),
pos = props.vertical === true ? content.scrollTop : Math.abs(content.scrollLeft);
if (isRTL.value === true) {
leftArrow.value = Math.ceil(pos + rect.width) < content.scrollWidth - 1;
rightArrow.value = pos > 0;
}
else {
leftArrow.value = pos > 0;
rightArrow.value = props.vertical === true
? Math.ceil(pos + rect.height) < content.scrollHeight
: Math.ceil(pos + rect.width) < content.scrollWidth;
}
}
function animScrollTo (value) {
scrollTimer !== null && clearInterval(scrollTimer);
scrollTimer = setInterval(() => {
if (scrollTowards(value) === true) {
stopAnimScroll();
}
}, 5);
}
function scrollToStart () {
animScrollTo(rtlPosCorrection.value === true ? Number.MAX_SAFE_INTEGER : 0);
}
function scrollToEnd () {
animScrollTo(rtlPosCorrection.value === true ? 0 : Number.MAX_SAFE_INTEGER);
}
function stopAnimScroll () {
if (scrollTimer !== null) {
clearInterval(scrollTimer);
scrollTimer = null;
}
}
function onKbdNavigate (keyCode, fromEl) {
const tabs = Array.prototype.filter.call(
contentRef.value.children,
el => el === fromEl || (el.matches && el.matches('.q-tab.q-focusable') === true)
);
const len = tabs.length;
if (len === 0) { return }
if (keyCode === 36) { // Home
scrollToTabEl(tabs[ 0 ]);
tabs[ 0 ].focus();
return true
}
if (keyCode === 35) { // End
scrollToTabEl(tabs[ len - 1 ]);
tabs[ len - 1 ].focus();
return true
}
const dirPrev = keyCode === (props.vertical === true ? 38 /* ArrowUp */ : 37 /* ArrowLeft */);
const dirNext = keyCode === (props.vertical === true ? 40 /* ArrowDown */ : 39 /* ArrowRight */);
const dir = dirPrev === true ? -1 : (dirNext === true ? 1 : void 0);
if (dir !== void 0) {
const rtlDir = isRTL.value === true ? -1 : 1;
const index = tabs.indexOf(fromEl) + dir * rtlDir;
if (index >= 0 && index < len) {
scrollToTabEl(tabs[ index ]);
tabs[ index ].focus({ preventScroll: true });
}
return true
}
}
// let's speed up execution of time-sensitive scrollTowards()
// with a computed variable by directly applying the minimal
// number of instructions on get/set functions
const posFn = vue.computed(() => (
rtlPosCorrection.value === true
? { get: content => Math.abs(content.scrollLeft), set: (content, pos) => { content.scrollLeft = -pos; } }
: (
props.vertical === true
? { get: content => content.scrollTop, set: (content, pos) => { content.scrollTop = pos; } }
: { get: content => content.scrollLeft, set: (content, pos) => { content.scrollLeft = pos; } }
)
));
function scrollTowards (value) {
const
content = contentRef.value,
{ get, set } = posFn.value;
let
done = false,
pos = get(content);
const direction = value < pos ? -1 : 1;
pos += direction * 5;
if (pos < 0) {
done = true;
pos = 0;
}
else if (
(direction === -1 && pos <= value)
|| (direction === 1 && pos >= value)
) {
done = true;
pos = value;
}
set(content, pos);
updateArrows();
return done
}
function hasQueryIncluded (targetQuery, matchingQuery) {
for (const key in targetQuery) {
if (targetQuery[ key ] !== matchingQuery[ key ]) {
return false
}
}
return true
}
// do not use directly; use verifyRouteModel() instead
function updateActiveRoute () {
let name = null, bestScore = { matchedLen: 0, queryDiff: 9999, hrefLen: 0 };
const list = tabDataList.filter(tab => tab.routeData !== void 0 && tab.routeData.hasRouterLink.value === true);
const { hash: currentHash, query: currentQuery } = proxy.$route;
const currentQueryLen = Object.keys(currentQuery).length;
// Vue Router does not keep account of hash & query when matching
// so we're doing this as well
for (const tab of list) {
const exact = tab.routeData.exact.value === true;
if (tab.routeData[ exact === true ? 'linkIsExactActive' : 'linkIsActive' ].value !== true) {
// it cannot match anything as it's not active nor exact-active
continue
}
const { hash, query, matched, href } = tab.routeData.resolvedLink.value;
const queryLen = Object.keys(query).length;
if (exact === true) {
if (hash !== currentHash) {
// it's set to exact but it doesn't matches the hash
continue
}
if (
queryLen !== currentQueryLen
|| hasQueryIncluded(currentQuery, query) === false
) {
// it's set to exact but it doesn't matches the query
continue
}
// yey, we found the perfect match (route + hash + query)
name = tab.name.value;
break
}
if (hash !== '' && hash !== currentHash) {
// it has hash and it doesn't matches
continue
}
if (
queryLen !== 0
&& hasQueryIncluded(query, currentQuery) === false
) {
// it has query and it doesn't includes the current one
continue
}
const newScore = {
matchedLen: matched.length,
queryDiff: currentQueryLen - queryLen,
hrefLen: href.length - hash.length
};
if (newScore.matchedLen > bestScore.matchedLen) {
// it matches more routes so it's more specific so we set it as current champion
name = tab.name.value;
bestScore = newScore;
continue
}
else if (newScore.matchedLen !== bestScore.matchedLen) {
// it matches less routes than the current champion so we discard it
continue
}
if (newScore.queryDiff < bestScore.queryDiff) {
// query is closer to the current one so we set it as current champion
name = tab.name.value;
bestScore = newScore;
}
else if (newScore.queryDiff !== bestScore.queryDiff) {
// it matches less routes than the current champion so we discard it
continue
}
if (newScore.hrefLen > bestScore.hrefLen) {
// href is lengthier so it's more specific so we set it as current champion
name = tab.name.value;
bestScore = newScore;
}
}
if (
name === null
&& tabDataList.some(tab => tab.routeData === void 0 && tab.name.value === currentModel.value) === true
) {
// we shouldn't interfere if non-route tab is active
return
}
updateModel({ name, setCurrent: true });
}
function onFocusin (e) {
removeFocusTimeout();
if (
hasFocus.value !== true
&& rootRef.value !== null
&& e.target
&& typeof e.target.closest === 'function'
) {
const tab = e.target.closest('.q-tab');
// if the target is contained by a QTab/QRouteTab
// (it might be other elements focused, like additional QBtn)
if (tab && rootRef.value.contains(tab) === true) {
hasFocus.value = true;
scrollable.value === true && scrollToTabEl(tab);
}
}
}
function onFocusout () {
registerFocusTimeout(() => { hasFocus.value = false; }, 30);
}
function verifyRouteModel () {
if ($tabs.avoidRouteWatcher === false) {
registerScrollToTabTimeout(updateActiveRoute);
}
else {
removeScrollToTabTimeout();
}
}
function watchRoute () {
if (unwatchRoute === void 0) {
const unwatch = vue.watch(() => proxy.$route.fullPath, verifyRouteModel);
unwatchRoute = () => {
unwatch();
unwatchRoute = void 0;
};
}
}
function registerTab (tabData) {
tabDataList.push(tabData);
tabDataListLen.value++;
recalculateScroll();
// if it's a QTab or we don't have Vue Router
if (tabData.routeData === void 0 || proxy.$route === void 0) {
// we should position to the currently active tab (if any)
registerScrollToTabTimeout(() => {
if (scrollable.value === true) {
const value = currentModel.value;
const newTab = value !== void 0 && value !== null && value !== ''
? tabDataList.find(tab => tab.name.value === value)
: null;
newTab && scrollToTabEl(newTab.rootRef.value);
}
});
}
// else if it's a QRouteTab with a valid link
else {
// start watching route
watchRoute();
if (tabData.routeData.hasRouterLink.value === true) {
verifyRouteModel();
}
}
}
function unregisterTab (tabData) {
tabDataList.splice(tabDataList.indexOf(tabData), 1);
tabDataListLen.value--;
recalculateScroll();
if (unwatchRoute !== void 0 && tabData.routeData !== void 0) {
// unwatch route if we don't have any QRouteTabs left
if (tabDataList.every(tab => tab.routeData === void 0) === true) {
unwatchRoute();
}
// then update model
verifyRouteModel();
}
}
const $tabs = {
currentModel,
tabProps,
hasFocus,
hasActiveTab,
registerTab,
unregisterTab,
verifyRouteModel,
updateModel,
onKbdNavigate,
avoidRouteWatcher: false // false | string (uid)
};
vue.provide(tabsKey, $tabs);
function cleanup () {
animateTimer !== null && clearTimeout(animateTimer);
stopAnimScroll();
unwatchRoute !== void 0 && unwatchRoute();
}
let hadRouteWatcher;
vue.onBeforeUnmount(cleanup);
vue.onDeactivated(() => {
hadRouteWatcher = unwatchRoute !== void 0;
cleanup();
});
vue.onActivated(() => {
hadRouteWatcher === true && watchRoute();
recalculateScroll();
});
return () => {
return vue.h('div', {
ref: rootRef,
class: classes.value,
role: 'tablist',
onFocusin,
onFocusout
}, [
vue.h(QResizeObserver, { onResize: updateContainer }),
vue.h('div', {
ref: contentRef,
class: innerClass.value,
onScroll: updateArrows
}, hSlot(slots.default)),
vue.h(QIcon, {
class: 'q-tabs__arrow q-tabs__arrow--left absolute q-tab__icon'
+ (leftArrow.value === true ? '' : ' q-tabs__arrow--faded'),
name: props.leftIcon || $q.iconSet.tabs[ props.vertical === true ? 'up' : 'left' ],
onMousedownPassive: scrollToStart,
onTouchstartPassive: scrollToStart,
onMouseupPassive: stopAnimScroll,
onMouseleavePassive: stopAnimScroll,
onTouchendPassive: stopAnimScroll
}),
vue.h(QIcon, {
class: 'q-tabs__arrow q-tabs__arrow--right absolute q-tab__icon'
+ (rightArrow.value === true ? '' : ' q-tabs__arrow--faded'),
name: props.rightIcon || $q.iconSet.tabs[ props.vertical === true ? 'down' : 'right' ],
onMousedownPassive: scrollToEnd,
onTouchstartPassive: scrollToEnd,
onMouseupPassive: stopAnimScroll,
onMouseleavePassive: stopAnimScroll,
onTouchendPassive: stopAnimScroll
})
])
}
}
});
let id$1 = 0;
const useTabEmits = [ 'click', 'keydown' ];
const useTabProps = {
icon: String,
label: [ Number, String ],
alert: [ Boolean, String ],
alertIcon: String,
name: {
type: [ Number, String ],
default: () => `t_${ id$1++ }`
},
noCaps: Boolean,
tabindex: [ String, Number ],
disable: Boolean,
contentClass: String,
ripple: {
type: [ Boolean, Object ],
default: true
}
};
function useTab (props, slots, emit, routeData) {
const $tabs = vue.inject(tabsKey, emptyRenderFn);
if ($tabs === emptyRenderFn) {
console.error('QTab/QRouteTab component needs to be child of QTabs');
return emptyRenderFn
}
const { proxy } = vue.getCurrentInstance();
const blurTargetRef = vue.ref(null);
const rootRef = vue.ref(null);
const tabIndicatorRef = vue.ref(null);
const ripple = vue.computed(() => (
props.disable === true || props.ripple === false
? false
: Object.assign(
{ keyCodes: [ 13, 32 ], early: true },
props.ripple === true ? {} : props.ripple
)
));
const isActive = vue.computed(() => $tabs.currentModel.value === props.name);
const classes = vue.computed(() =>
'q-tab relative-position self-stretch flex flex-center text-center'
+ (
isActive.value === true
? (
' q-tab--active'
+ ($tabs.tabProps.value.activeClass ? ' ' + $tabs.tabProps.value.activeClass : '')
+ ($tabs.tabProps.value.activeColor ? ` text-${ $tabs.tabProps.value.activeColor }` : '')
+ ($tabs.tabProps.value.activeBgColor ? ` bg-${ $tabs.tabProps.value.activeBgColor }` : '')
)
: ' q-tab--inactive'
)
+ (props.icon && props.label && $tabs.tabProps.value.inlineLabel === false ? ' q-tab--full' : '')
+ (props.noCaps === true || $tabs.tabProps.value.noCaps === true ? ' q-tab--no-caps' : '')
+ (props.disable === true ? ' disabled' : ' q-focusable q-hoverable cursor-pointer')
+ (routeData !== void 0 ? routeData.linkClass.value : '')
);
const innerClass = vue.computed(() =>
'q-tab__content self-stretch flex-center relative-position q-anchor--skip non-selectable '
+ ($tabs.tabProps.value.inlineLabel === true ? 'row no-wrap q-tab__content--inline' : 'column')
+ (props.contentClass !== void 0 ? ` ${ props.contentClass }` : '')
);
const tabIndex = vue.computed(() => (
(
props.disable === true
|| $tabs.hasFocus.value === true
|| (isActive.value === false && $tabs.hasActiveTab.value === true)
)
? -1
: props.tabindex || 0
));
function onClick (e, keyboard) {
if (keyboard !== true && blurTargetRef.value !== null) {
blurTargetRef.value.focus();
}
if (props.disable === true) {
// we should hinder native navigation though
if (routeData !== void 0 && routeData.hasRouterLink.value === true) {
stopAndPrevent(e);
}
return
}
// do we have a QTab?
if (routeData === void 0) {
$tabs.updateModel({ name: props.name });
emit('click', e);
return
}
if (routeData.hasRouterLink.value === true) {
const go = (opts = {}) => {
// if requiring to go to another route, then we
// let the QTabs route watcher do its job,
// otherwise directly select this
let hardError;
const reqId = opts.to === void 0 || isDeepEqual(opts.to, props.to) === true
? ($tabs.avoidRouteWatcher = uid$3())
: null;
return routeData.navigateToRouterLink(e, { ...opts, returnRouterError: true })
.catch(err => { hardError = err; })
.then(softError => {
if (reqId === $tabs.avoidRouteWatcher) {
$tabs.avoidRouteWatcher = false;
// if we don't have any hard errors or any soft errors, except for
// when navigating to the same route (on all other soft errors,
// like when navigation was aborted in a nav guard, we don't activate this tab)
if (
hardError === void 0 && (
softError === void 0
|| softError.message.startsWith('Avoided redundant navigation') === true
)
) {
$tabs.updateModel({ name: props.name });
}
}
if (opts.returnRouterError === true) {
return hardError !== void 0 ? Promise.reject(hardError) : softError
}
})
};
emit('click', e, go);
e.defaultPrevented !== true && go();
return
}
emit('click', e);
}
function onKeydown (e) {
if (isKeyCode(e, [ 13, 32 ])) {
onClick(e, true);
}
else if (
shouldIgnoreKey(e) !== true
&& e.keyCode >= 35
&& e.keyCode <= 40
&& e.altKey !== true
&& e.metaKey !== true
) {
$tabs.onKbdNavigate(e.keyCode, proxy.$el) === true && stopAndPrevent(e);
}
emit('keydown', e);
}
function getContent () {
const
narrow = $tabs.tabProps.value.narrowIndicator,
content = [],
indicator = vue.h('div', {
ref: tabIndicatorRef,
class: [
'q-tab__indicator',
$tabs.tabProps.value.indicatorClass
]
});
props.icon !== void 0 && content.push(
vue.h(QIcon, {
class: 'q-tab__icon',
name: props.icon
})
);
props.label !== void 0 && content.push(
vue.h('div', { class: 'q-tab__label' }, props.label)
);
props.alert !== false && content.push(
props.alertIcon !== void 0
? vue.h(QIcon, {
class: 'q-tab__alert-icon',
color: props.alert !== true
? props.alert
: void 0,
name: props.alertIcon
})
: vue.h('div', {
class: 'q-tab__alert'
+ (props.alert !== true ? ` text-${ props.alert }` : '')
})
);
narrow === true && content.push(indicator);
const node = [
vue.h('div', { class: 'q-focus-helper', tabindex: -1, ref: blurTargetRef }),
vue.h('div', { class: innerClass.value }, hMergeSlot(slots.default, content))
];
narrow === false && node.push(indicator);
return node
}
const tabData = {
name: vue.computed(() => props.name),
rootRef,
tabIndicatorRef,
routeData
};
vue.onBeforeUnmount(() => {
$tabs.unregisterTab(tabData);
});
vue.onMounted(() => {
$tabs.registerTab(tabData);
});
function renderTab (tag, customData) {
const data = {
ref: rootRef,
class: classes.value,
tabindex: tabIndex.value,
role: 'tab',
'aria-selected': isActive.value === true ? 'true' : 'false',
'aria-disabled': props.disable === true ? 'true' : void 0,
onClick,
onKeydown,
...customData
};
return vue.withDirectives(
vue.h(tag, data, getContent()),
[ [ Ripple, ripple.value ] ]
)
}
return { renderTab, $tabs }
}
var QTab = createComponent({
name: 'QTab',
props: useTabProps,
emits: useTabEmits,
setup (props, { slots, emit }) {
const { renderTab } = useTab(props, slots, emit);
return () => renderTab('div')
}
});
var QTabPanels = createComponent({
name: 'QTabPanels',
props: {
...usePanelProps,
...useDarkProps
},
emits: usePanelEmits,
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const { updatePanelsList, getPanelContent, panelDirectives } = usePanel();
const classes = vue.computed(() =>
'q-tab-panels q-panel-parent'
+ (isDark.value === true ? ' q-tab-panels--dark q-dark' : '')
);
return () => {
updatePanelsList(slots);
return hDir(
'div',
{ class: classes.value },
getPanelContent(),
'pan',
props.swipeable,
() => panelDirectives.value
)
}
}
});
var QTabPanel = createComponent({
name: 'QTabPanel',
props: usePanelChildProps,
setup (_, { slots }) {
return () => vue.h('div', { class: 'q-tab-panel', role: 'tabpanel' }, hSlot(slots.default))
}
});
// file referenced from docs
const
hex = /^#[0-9a-fA-F]{3}([0-9a-fA-F]{3})?$/,
hexa = /^#[0-9a-fA-F]{4}([0-9a-fA-F]{4})?$/,
hexOrHexa = /^#([0-9a-fA-F]{3}|[0-9a-fA-F]{4}|[0-9a-fA-F]{6}|[0-9a-fA-F]{8})$/,
rgb = /^rgb\(((0|[1-9][\d]?|1[\d]{0,2}|2[\d]?|2[0-4][\d]|25[0-5]),){2}(0|[1-9][\d]?|1[\d]{0,2}|2[\d]?|2[0-4][\d]|25[0-5])\)$/,
rgba = /^rgba\(((0|[1-9][\d]?|1[\d]{0,2}|2[\d]?|2[0-4][\d]|25[0-5]),){2}(0|[1-9][\d]?|1[\d]{0,2}|2[\d]?|2[0-4][\d]|25[0-5]),(0|0\.[0-9]+[1-9]|0\.[1-9]+|1)\)$/;
// Keep in sync with ui/types/api/validation.d.ts
const testPattern = {
date: v => /^-?[\d]+\/[0-1]\d\/[0-3]\d$/.test(v),
time: v => /^([0-1]?\d|2[0-3]):[0-5]\d$/.test(v),
fulltime: v => /^([0-1]?\d|2[0-3]):[0-5]\d:[0-5]\d$/.test(v),
timeOrFulltime: v => /^([0-1]?\d|2[0-3]):[0-5]\d(:[0-5]\d)?$/.test(v),
// -- RFC 5322 --
// -- Added in v2.6.6 --
// This is a basic helper validation.
// For something more complex (like RFC 822) you should write and use your own rule.
// We won't be accepting PRs to enhance the one below because of the reason above.
// eslint-disable-next-line
email: v => /^(([^<>()\[\]\\.,;:\s@"]+(\.[^<>()\[\]\\.,;:\s@"]+)*)|(".+"))@((\[[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}])|(([a-zA-Z\-0-9]+\.)+[a-zA-Z]{2,}))$/.test(v),
hexColor: v => hex.test(v),
hexaColor: v => hexa.test(v),
hexOrHexaColor: v => hexOrHexa.test(v),
rgbColor: v => rgb.test(v),
rgbaColor: v => rgba.test(v),
rgbOrRgbaColor: v => rgb.test(v) || rgba.test(v),
hexOrRgbColor: v => hex.test(v) || rgb.test(v),
hexaOrRgbaColor: v => hexa.test(v) || rgba.test(v),
anyColor: v => hexOrHexa.test(v) || rgb.test(v) || rgba.test(v)
};
var patterns = {
testPattern
};
const reRGBA = /^rgb(a)?\((\d{1,3}),(\d{1,3}),(\d{1,3}),?([01]?\.?\d*?)?\)$/;
function rgbToHex ({ r, g, b, a }) {
const alpha = a !== void 0;
r = Math.round(r);
g = Math.round(g);
b = Math.round(b);
if (
r > 255
|| g > 255
|| b > 255
|| (alpha && a > 100)
) {
throw new TypeError('Expected 3 numbers below 256 (and optionally one below 100)')
}
a = alpha
? (Math.round(255 * a / 100) | 1 << 8).toString(16).slice(1)
: '';
return '#' + ((b | g << 8 | r << 16) | 1 << 24).toString(16).slice(1) + a
}
function rgbToString ({ r, g, b, a }) {
return `rgb${ a !== void 0 ? 'a' : '' }(${ r },${ g },${ b }${ a !== void 0 ? ',' + (a / 100) : '' })`
}
function hexToRgb (hex) {
if (typeof hex !== 'string') {
throw new TypeError('Expected a string')
}
hex = hex.replace(/^#/, '');
if (hex.length === 3) {
hex = hex[ 0 ] + hex[ 0 ] + hex[ 1 ] + hex[ 1 ] + hex[ 2 ] + hex[ 2 ];
}
else if (hex.length === 4) {
hex = hex[ 0 ] + hex[ 0 ] + hex[ 1 ] + hex[ 1 ] + hex[ 2 ] + hex[ 2 ] + hex[ 3 ] + hex[ 3 ];
}
const num = parseInt(hex, 16);
return hex.length > 6
? { r: num >> 24 & 255, g: num >> 16 & 255, b: num >> 8 & 255, a: Math.round((num & 255) / 2.55) }
: { r: num >> 16, g: num >> 8 & 255, b: num & 255 }
}
function hsvToRgb ({ h, s, v, a }) {
let r, g, b;
s = s / 100;
v = v / 100;
h = h / 360;
const
i = Math.floor(h * 6),
f = h * 6 - i,
p = v * (1 - s),
q = v * (1 - f * s),
t = v * (1 - (1 - f) * s);
switch (i % 6) {
case 0:
r = v;
g = t;
b = p;
break
case 1:
r = q;
g = v;
b = p;
break
case 2:
r = p;
g = v;
b = t;
break
case 3:
r = p;
g = q;
b = v;
break
case 4:
r = t;
g = p;
b = v;
break
case 5:
r = v;
g = p;
b = q;
break
}
return {
r: Math.round(r * 255),
g: Math.round(g * 255),
b: Math.round(b * 255),
a
}
}
function rgbToHsv ({ r, g, b, a }) {
const
max = Math.max(r, g, b),
min = Math.min(r, g, b),
d = max - min,
s = (max === 0 ? 0 : d / max),
v = max / 255;
let h;
switch (max) {
case min:
h = 0;
break
case r:
h = (g - b) + d * (g < b ? 6 : 0);
h /= 6 * d;
break
case g:
h = (b - r) + d * 2;
h /= 6 * d;
break
case b:
h = (r - g) + d * 4;
h /= 6 * d;
break
}
return {
h: Math.round(h * 360),
s: Math.round(s * 100),
v: Math.round(v * 100),
a
}
}
function textToRgb (str) {
if (typeof str !== 'string') {
throw new TypeError('Expected a string')
}
const color = str.replace(/ /g, '');
const m = reRGBA.exec(color);
if (m === null) {
return hexToRgb(color)
}
const rgb = {
r: Math.min(255, parseInt(m[ 2 ], 10)),
g: Math.min(255, parseInt(m[ 3 ], 10)),
b: Math.min(255, parseInt(m[ 4 ], 10))
};
if (m[ 1 ]) {
const alpha = parseFloat(m[ 5 ]);
rgb.a = Math.min(1, isNaN(alpha) === true ? 1 : alpha) * 100;
}
return rgb
}
/* works as darken if percent < 0 */
function lighten (color, percent) {
if (typeof color !== 'string') {
throw new TypeError('Expected a string as color')
}
if (typeof percent !== 'number') {
throw new TypeError('Expected a numeric percent')
}
const rgb = textToRgb(color),
t = percent < 0 ? 0 : 255,
p = Math.abs(percent) / 100,
R = rgb.r,
G = rgb.g,
B = rgb.b;
return '#' + (
0x1000000 + (Math.round((t - R) * p) + R) * 0x10000
+ (Math.round((t - G) * p) + G) * 0x100
+ (Math.round((t - B) * p) + B)
).toString(16).slice(1)
}
function luminosity (color) {
if (typeof color !== 'string' && (!color || color.r === void 0)) {
throw new TypeError('Expected a string or a {r, g, b} object as color')
}
const
rgb = typeof color === 'string' ? textToRgb(color) : color,
r = rgb.r / 255,
g = rgb.g / 255,
b = rgb.b / 255,
R = r <= 0.03928 ? r / 12.92 : Math.pow((r + 0.055) / 1.055, 2.4),
G = g <= 0.03928 ? g / 12.92 : Math.pow((g + 0.055) / 1.055, 2.4),
B = b <= 0.03928 ? b / 12.92 : Math.pow((b + 0.055) / 1.055, 2.4);
return 0.2126 * R + 0.7152 * G + 0.0722 * B
}
function brightness (color) {
if (typeof color !== 'string' && (!color || color.r === void 0)) {
throw new TypeError('Expected a string or a {r, g, b} object as color')
}
const rgb = typeof color === 'string'
? textToRgb(color)
: color;
return (rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000
}
function blend (fgColor, bgColor) {
if (typeof fgColor !== 'string' && (!fgColor || fgColor.r === void 0)) {
throw new TypeError('Expected a string or a {r, g, b[, a]} object as fgColor')
}
if (typeof bgColor !== 'string' && (!bgColor || bgColor.r === void 0)) {
throw new TypeError('Expected a string or a {r, g, b[, a]} object as bgColor')
}
const
rgb1 = typeof fgColor === 'string' ? textToRgb(fgColor) : fgColor,
r1 = rgb1.r / 255,
g1 = rgb1.g / 255,
b1 = rgb1.b / 255,
a1 = rgb1.a !== void 0 ? rgb1.a / 100 : 1,
rgb2 = typeof bgColor === 'string' ? textToRgb(bgColor) : bgColor,
r2 = rgb2.r / 255,
g2 = rgb2.g / 255,
b2 = rgb2.b / 255,
a2 = rgb2.a !== void 0 ? rgb2.a / 100 : 1,
a = a1 + a2 * (1 - a1),
r = Math.round(((r1 * a1 + r2 * a2 * (1 - a1)) / a) * 255),
g = Math.round(((g1 * a1 + g2 * a2 * (1 - a1)) / a) * 255),
b = Math.round(((b1 * a1 + b2 * a2 * (1 - a1)) / a) * 255);
const ret = { r, g, b, a: Math.round(a * 100) };
return typeof fgColor === 'string'
? rgbToHex(ret)
: ret
}
function changeAlpha (color, offset) {
if (typeof color !== 'string') {
throw new TypeError('Expected a string as color')
}
if (offset === void 0 || offset < -1 || offset > 1) {
throw new TypeError('Expected offset to be between -1 and 1')
}
const { r, g, b, a } = textToRgb(color);
const alpha = a !== void 0 ? a / 100 : 0;
return rgbToHex({
r, g, b, a: Math.round(Math.min(1, Math.max(0, alpha + offset)) * 100)
})
}
function getPaletteColor (colorName) {
if (typeof colorName !== 'string') {
throw new TypeError('Expected a string as color')
}
const el = document.createElement('div');
el.className = `text-${ colorName } invisible fixed no-pointer-events`;
document.body.appendChild(el);
const result = getComputedStyle(el).getPropertyValue('color');
el.remove();
return rgbToHex(textToRgb(result))
}
var colors = {
rgbToHex,
hexToRgb,
hsvToRgb,
rgbToHsv,
textToRgb,
lighten,
luminosity,
brightness,
blend,
changeAlpha,
getPaletteColor
};
const palette = [
'rgb(255,204,204)', 'rgb(255,230,204)', 'rgb(255,255,204)', 'rgb(204,255,204)', 'rgb(204,255,230)', 'rgb(204,255,255)', 'rgb(204,230,255)', 'rgb(204,204,255)', 'rgb(230,204,255)', 'rgb(255,204,255)',
'rgb(255,153,153)', 'rgb(255,204,153)', 'rgb(255,255,153)', 'rgb(153,255,153)', 'rgb(153,255,204)', 'rgb(153,255,255)', 'rgb(153,204,255)', 'rgb(153,153,255)', 'rgb(204,153,255)', 'rgb(255,153,255)',
'rgb(255,102,102)', 'rgb(255,179,102)', 'rgb(255,255,102)', 'rgb(102,255,102)', 'rgb(102,255,179)', 'rgb(102,255,255)', 'rgb(102,179,255)', 'rgb(102,102,255)', 'rgb(179,102,255)', 'rgb(255,102,255)',
'rgb(255,51,51)', 'rgb(255,153,51)', 'rgb(255,255,51)', 'rgb(51,255,51)', 'rgb(51,255,153)', 'rgb(51,255,255)', 'rgb(51,153,255)', 'rgb(51,51,255)', 'rgb(153,51,255)', 'rgb(255,51,255)',
'rgb(255,0,0)', 'rgb(255,128,0)', 'rgb(255,255,0)', 'rgb(0,255,0)', 'rgb(0,255,128)', 'rgb(0,255,255)', 'rgb(0,128,255)', 'rgb(0,0,255)', 'rgb(128,0,255)', 'rgb(255,0,255)',
'rgb(245,0,0)', 'rgb(245,123,0)', 'rgb(245,245,0)', 'rgb(0,245,0)', 'rgb(0,245,123)', 'rgb(0,245,245)', 'rgb(0,123,245)', 'rgb(0,0,245)', 'rgb(123,0,245)', 'rgb(245,0,245)',
'rgb(214,0,0)', 'rgb(214,108,0)', 'rgb(214,214,0)', 'rgb(0,214,0)', 'rgb(0,214,108)', 'rgb(0,214,214)', 'rgb(0,108,214)', 'rgb(0,0,214)', 'rgb(108,0,214)', 'rgb(214,0,214)',
'rgb(163,0,0)', 'rgb(163,82,0)', 'rgb(163,163,0)', 'rgb(0,163,0)', 'rgb(0,163,82)', 'rgb(0,163,163)', 'rgb(0,82,163)', 'rgb(0,0,163)', 'rgb(82,0,163)', 'rgb(163,0,163)',
'rgb(92,0,0)', 'rgb(92,46,0)', 'rgb(92,92,0)', 'rgb(0,92,0)', 'rgb(0,92,46)', 'rgb(0,92,92)', 'rgb(0,46,92)', 'rgb(0,0,92)', 'rgb(46,0,92)', 'rgb(92,0,92)',
'rgb(255,255,255)', 'rgb(205,205,205)', 'rgb(178,178,178)', 'rgb(153,153,153)', 'rgb(127,127,127)', 'rgb(102,102,102)', 'rgb(76,76,76)', 'rgb(51,51,51)', 'rgb(25,25,25)', 'rgb(0,0,0)'
];
const thumbPath = 'M5 5 h10 v10 h-10 v-10 z';
const alphaTrackImg = '';
var QColor = createComponent({
name: 'QColor',
props: {
...useDarkProps,
...useFormProps,
modelValue: String,
defaultValue: String,
defaultView: {
type: String,
default: 'spectrum',
validator: v => [ 'spectrum', 'tune', 'palette' ].includes(v)
},
formatModel: {
type: String,
default: 'auto',
validator: v => [ 'auto', 'hex', 'rgb', 'hexa', 'rgba' ].includes(v)
},
palette: Array,
noHeader: Boolean,
noHeaderTabs: Boolean,
noFooter: Boolean,
square: Boolean,
flat: Boolean,
bordered: Boolean,
disable: Boolean,
readonly: Boolean
},
emits: [ 'update:modelValue', 'change' ],
setup (props, { emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const { getCache } = useCache();
const spectrumRef = vue.ref(null);
const errorIconRef = vue.ref(null);
const forceHex = vue.computed(() => (
props.formatModel === 'auto'
? null
: props.formatModel.indexOf('hex') > -1
));
const forceAlpha = vue.computed(() => (
props.formatModel === 'auto'
? null
: props.formatModel.indexOf('a') > -1
));
const topView = vue.ref(
props.formatModel === 'auto'
? (
(props.modelValue === void 0 || props.modelValue === null || props.modelValue === '' || props.modelValue.startsWith('#'))
? 'hex'
: 'rgb'
)
: (props.formatModel.startsWith('hex') ? 'hex' : 'rgb')
);
const view = vue.ref(props.defaultView);
const model = vue.ref(parseModel(props.modelValue || props.defaultValue));
const editable = vue.computed(() => props.disable !== true && props.readonly !== true);
const isHex = vue.computed(() =>
props.modelValue === void 0
|| props.modelValue === null
|| props.modelValue === ''
|| props.modelValue.startsWith('#')
);
const isOutputHex = vue.computed(() => (
forceHex.value !== null
? forceHex.value
: isHex.value
));
const formAttrs = vue.computed(() => ({
type: 'hidden',
name: props.name,
value: model.value[ isOutputHex.value === true ? 'hex' : 'rgb' ]
}));
const injectFormInput = useFormInject(formAttrs);
const hasAlpha = vue.computed(() => (
forceAlpha.value !== null
? forceAlpha.value
: model.value.a !== void 0
));
const currentBgColor = vue.computed(() => ({
backgroundColor: model.value.rgb || '#000'
}));
const headerClass = vue.computed(() => {
const light = model.value.a !== void 0 && model.value.a < 65
? true
: luminosity(model.value) > 0.4;
return 'q-color-picker__header-content'
+ ` q-color-picker__header-content--${ light ? 'light' : 'dark' }`
});
const spectrumStyle = vue.computed(() => ({
background: `hsl(${ model.value.h },100%,50%)`
}));
const spectrumPointerStyle = vue.computed(() => ({
top: `${ 100 - model.value.v }%`,
[ $q.lang.rtl === true ? 'right' : 'left' ]: `${ model.value.s }%`
}));
const computedPalette = vue.computed(() => (
props.palette !== void 0 && props.palette.length !== 0
? props.palette
: palette
));
const classes = vue.computed(() =>
'q-color-picker'
+ (props.bordered === true ? ' q-color-picker--bordered' : '')
+ (props.square === true ? ' q-color-picker--square no-border-radius' : '')
+ (props.flat === true ? ' q-color-picker--flat no-shadow' : '')
+ (props.disable === true ? ' disabled' : '')
+ (isDark.value === true ? ' q-color-picker--dark q-dark' : '')
);
const attributes = vue.computed(() => {
if (props.disable === true) {
return { 'aria-disabled': 'true' }
}
if (props.readonly === true) {
return { 'aria-readonly': 'true' }
}
return {}
});
const spectrumDirective = vue.computed(() => {
// if editable.value === true
return [ [
TouchPan,
onSpectrumPan,
void 0,
{ prevent: true, stop: true, mouse: true }
] ]
});
vue.watch(() => props.modelValue, v => {
const localModel = parseModel(v || props.defaultValue);
if (localModel.hex !== model.value.hex) {
model.value = localModel;
}
});
vue.watch(() => props.defaultValue, v => {
if (!props.modelValue && v) {
const localModel = parseModel(v);
if (localModel.hex !== model.value.hex) {
model.value = localModel;
}
}
});
function updateModel (rgb, change) {
// update internally
model.value.hex = rgbToHex(rgb);
model.value.rgb = rgbToString(rgb);
model.value.r = rgb.r;
model.value.g = rgb.g;
model.value.b = rgb.b;
model.value.a = rgb.a;
const value = model.value[ isOutputHex.value === true ? 'hex' : 'rgb' ];
// emit new value
emit('update:modelValue', value);
change === true && emit('change', value);
}
function parseModel (v) {
const alpha = forceAlpha.value !== void 0
? forceAlpha.value
: (
props.formatModel === 'auto'
? null
: props.formatModel.indexOf('a') > -1
);
if (typeof v !== 'string' || v.length === 0 || testPattern.anyColor(v.replace(/ /g, '')) !== true) {
return {
h: 0,
s: 0,
v: 0,
r: 0,
g: 0,
b: 0,
a: alpha === true ? 100 : void 0,
hex: void 0,
rgb: void 0
}
}
const model = textToRgb(v);
if (alpha === true && model.a === void 0) {
model.a = 100;
}
model.hex = rgbToHex(model);
model.rgb = rgbToString(model);
return Object.assign(model, rgbToHsv(model))
}
function changeSpectrum (left, top, change) {
const panel = spectrumRef.value;
if (panel === null) { return }
const
width = panel.clientWidth,
height = panel.clientHeight,
rect = panel.getBoundingClientRect();
let x = Math.min(width, Math.max(0, left - rect.left));
if ($q.lang.rtl === true) {
x = width - x;
}
const
y = Math.min(height, Math.max(0, top - rect.top)),
s = Math.round(100 * x / width),
v = Math.round(100 * Math.max(0, Math.min(1, -(y / height) + 1))),
rgb = hsvToRgb({
h: model.value.h,
s,
v,
a: hasAlpha.value === true ? model.value.a : void 0
});
model.value.s = s;
model.value.v = v;
updateModel(rgb, change);
}
function onHueChange (val, change) {
const h = Math.round(val);
const rgb = hsvToRgb({
h,
s: model.value.s,
v: model.value.v,
a: hasAlpha.value === true ? model.value.a : void 0
});
model.value.h = h;
updateModel(rgb, change);
}
function onNumericChange (value, formatModel, max, evt, change) {
evt !== void 0 && stop(evt);
if (!/^[0-9]+$/.test(value)) {
change === true && proxy.$forceUpdate();
return
}
const val = Math.floor(Number(value));
if (val < 0 || val > max) {
change === true && proxy.$forceUpdate();
return
}
const rgb = {
r: formatModel === 'r' ? val : model.value.r,
g: formatModel === 'g' ? val : model.value.g,
b: formatModel === 'b' ? val : model.value.b,
a: hasAlpha.value === true
? (formatModel === 'a' ? val : model.value.a)
: void 0
};
if (formatModel !== 'a') {
const hsv = rgbToHsv(rgb);
model.value.h = hsv.h;
model.value.s = hsv.s;
model.value.v = hsv.v;
}
updateModel(rgb, change);
if (evt !== void 0 && change !== true && evt.target.selectionEnd !== void 0) {
const index = evt.target.selectionEnd;
vue.nextTick(() => {
evt.target.setSelectionRange(index, index);
});
}
}
function onEditorChange (evt, change) {
let rgb;
const inp = evt.target.value;
stop(evt);
if (topView.value === 'hex') {
if (
inp.length !== (hasAlpha.value === true ? 9 : 7)
|| !/^#[0-9A-Fa-f]+$/.test(inp)
) {
return true
}
rgb = hexToRgb(inp);
}
else {
let model;
if (!inp.endsWith(')')) {
return true
}
else if (hasAlpha.value !== true && inp.startsWith('rgb(')) {
model = inp.substring(4, inp.length - 1).split(',').map(n => parseInt(n, 10));
if (
model.length !== 3
|| !/^rgb\([0-9]{1,3},[0-9]{1,3},[0-9]{1,3}\)$/.test(inp)
) {
return true
}
}
else if (hasAlpha.value === true && inp.startsWith('rgba(')) {
model = inp.substring(5, inp.length - 1).split(',');
if (
model.length !== 4
|| !/^rgba\([0-9]{1,3},[0-9]{1,3},[0-9]{1,3},(0|0\.[0-9]+[1-9]|0\.[1-9]+|1)\)$/.test(inp)
) {
return true
}
for (let i = 0; i < 3; i++) {
const v = parseInt(model[ i ], 10);
if (v < 0 || v > 255) {
return true
}
model[ i ] = v;
}
const v = parseFloat(model[ 3 ]);
if (v < 0 || v > 1) {
return true
}
model[ 3 ] = v;
}
else {
return true
}
if (
model[ 0 ] < 0 || model[ 0 ] > 255
|| model[ 1 ] < 0 || model[ 1 ] > 255
|| model[ 2 ] < 0 || model[ 2 ] > 255
|| (hasAlpha.value === true && (model[ 3 ] < 0 || model[ 3 ] > 1))
) {
return true
}
rgb = {
r: model[ 0 ],
g: model[ 1 ],
b: model[ 2 ],
a: hasAlpha.value === true
? model[ 3 ] * 100
: void 0
};
}
const hsv = rgbToHsv(rgb);
model.value.h = hsv.h;
model.value.s = hsv.s;
model.value.v = hsv.v;
updateModel(rgb, change);
if (change !== true) {
const index = evt.target.selectionEnd;
vue.nextTick(() => {
evt.target.setSelectionRange(index, index);
});
}
}
function onPalettePick (color) {
const def = parseModel(color);
const rgb = { r: def.r, g: def.g, b: def.b, a: def.a };
if (rgb.a === void 0) {
rgb.a = model.value.a;
}
model.value.h = def.h;
model.value.s = def.s;
model.value.v = def.v;
updateModel(rgb, true);
}
function onSpectrumPan (evt) {
if (evt.isFinal) {
changeSpectrum(
evt.position.left,
evt.position.top,
true
);
}
else {
onSpectrumChange(evt);
}
}
const onSpectrumChange = throttle(
evt => { changeSpectrum(evt.position.left, evt.position.top); },
20
);
function onSpectrumClick (evt) {
changeSpectrum(
evt.pageX - window.pageXOffset,
evt.pageY - window.pageYOffset,
true
);
}
function onActivate (evt) {
changeSpectrum(
evt.pageX - window.pageXOffset,
evt.pageY - window.pageYOffset
);
}
function updateErrorIcon (val) {
// we MUST avoid vue triggering a render,
// so manually changing this
if (errorIconRef.value !== null) {
errorIconRef.value.$el.style.opacity = val ? 1 : 0;
}
}
function getHeader () {
const child = [];
props.noHeaderTabs !== true && child.push(
vue.h(QTabs, {
class: 'q-color-picker__header-tabs',
modelValue: topView.value,
dense: true,
align: 'justify',
...getCache('topVTab', {
'onUpdate:modelValue': val => { topView.value = val; }
})
}, () => [
vue.h(QTab, {
label: 'HEX' + (hasAlpha.value === true ? 'A' : ''),
name: 'hex',
ripple: false
}),
vue.h(QTab, {
label: 'RGB' + (hasAlpha.value === true ? 'A' : ''),
name: 'rgb',
ripple: false
})
])
);
child.push(
vue.h('div', {
class: 'q-color-picker__header-banner row flex-center no-wrap'
}, [
vue.h('input', {
class: 'fit',
value: model.value[ topView.value ],
...(editable.value !== true
? { readonly: true }
: {}
),
...getCache('topIn', {
onInput: evt => {
updateErrorIcon(onEditorChange(evt) === true);
},
onChange: stop,
onBlur: evt => {
onEditorChange(evt, true) === true && proxy.$forceUpdate();
updateErrorIcon(false);
}
})
}),
vue.h(QIcon, {
ref: errorIconRef,
class: 'q-color-picker__error-icon absolute no-pointer-events',
name: $q.iconSet.type.negative
})
])
);
return vue.h('div', {
class: 'q-color-picker__header relative-position overflow-hidden'
}, [
vue.h('div', { class: 'q-color-picker__header-bg absolute-full' }),
vue.h('div', {
class: headerClass.value,
style: currentBgColor.value
}, child)
])
}
function getContent () {
return vue.h(QTabPanels, {
modelValue: view.value,
animated: true
}, () => [
vue.h(QTabPanel, {
class: 'q-color-picker__spectrum-tab overflow-hidden',
name: 'spectrum'
}, getSpectrumTab),
vue.h(QTabPanel, {
class: 'q-pa-md q-color-picker__tune-tab',
name: 'tune'
}, getTuneTab),
vue.h(QTabPanel, {
class: 'q-color-picker__palette-tab',
name: 'palette'
}, getPaletteTab)
])
}
function getFooter () {
return vue.h('div', {
class: 'q-color-picker__footer relative-position overflow-hidden'
}, [
vue.h(QTabs, {
class: 'absolute-full',
modelValue: view.value,
dense: true,
align: 'justify',
...getCache('ftIn', {
'onUpdate:modelValue': val => { view.value = val; }
})
}, () => [
vue.h(QTab, {
icon: $q.iconSet.colorPicker.spectrum,
name: 'spectrum',
ripple: false
}),
vue.h(QTab, {
icon: $q.iconSet.colorPicker.tune,
name: 'tune',
ripple: false
}),
vue.h(QTab, {
icon: $q.iconSet.colorPicker.palette,
name: 'palette',
ripple: false
})
])
])
}
function getSpectrumTab () {
const data = {
ref: spectrumRef,
class: 'q-color-picker__spectrum non-selectable relative-position cursor-pointer'
+ (editable.value !== true ? ' readonly' : ''),
style: spectrumStyle.value,
...(editable.value === true
? {
onClick: onSpectrumClick,
onMousedown: onActivate
}
: {}
)
};
const child = [
vue.h('div', { style: { paddingBottom: '100%' } }),
vue.h('div', { class: 'q-color-picker__spectrum-white absolute-full' }),
vue.h('div', { class: 'q-color-picker__spectrum-black absolute-full' }),
vue.h('div', {
class: 'absolute',
style: spectrumPointerStyle.value
}, [
model.value.hex !== void 0
? vue.h('div', { class: 'q-color-picker__spectrum-circle' })
: null
])
];
const sliders = [
vue.h(QSlider, {
class: 'q-color-picker__hue non-selectable',
modelValue: model.value.h,
min: 0,
max: 360,
trackSize: '8px',
innerTrackColor: 'transparent',
selectionColor: 'transparent',
readonly: editable.value !== true,
thumbPath,
'onUpdate:modelValue': onHueChange,
...getCache('lazyhue', {
onChange: val => onHueChange(val, true)
})
})
];
hasAlpha.value === true && sliders.push(
vue.h(QSlider, {
class: 'q-color-picker__alpha non-selectable',
modelValue: model.value.a,
min: 0,
max: 100,
trackSize: '8px',
trackColor: 'white',
innerTrackColor: 'transparent',
selectionColor: 'transparent',
trackImg: alphaTrackImg,
readonly: editable.value !== true,
hideSelection: true,
thumbPath,
...getCache('alphaSlide', {
'onUpdate:modelValue': value => onNumericChange(value, 'a', 100),
onChange: value => onNumericChange(value, 'a', 100, void 0, true)
})
})
);
return [
hDir('div', data, child, 'spec', editable.value, () => spectrumDirective.value),
vue.h('div', { class: 'q-color-picker__sliders' }, sliders)
]
}
function getTuneTab () {
return [
vue.h('div', { class: 'row items-center no-wrap' }, [
vue.h('div', 'R'),
vue.h(QSlider, {
modelValue: model.value.r,
min: 0,
max: 255,
color: 'red',
dark: isDark.value,
readonly: editable.value !== true,
...getCache('rSlide', {
'onUpdate:modelValue': value => onNumericChange(value, 'r', 255),
onChange: value => onNumericChange(value, 'r', 255, void 0, true)
})
}),
vue.h('input', {
value: model.value.r,
maxlength: 3,
readonly: editable.value !== true,
onChange: stop,
...getCache('rIn', {
onInput: evt => onNumericChange(evt.target.value, 'r', 255, evt),
onBlur: evt => onNumericChange(evt.target.value, 'r', 255, evt, true)
})
})
]),
vue.h('div', { class: 'row items-center no-wrap' }, [
vue.h('div', 'G'),
vue.h(QSlider, {
modelValue: model.value.g,
min: 0,
max: 255,
color: 'green',
dark: isDark.value,
readonly: editable.value !== true,
...getCache('gSlide', {
'onUpdate:modelValue': value => onNumericChange(value, 'g', 255),
onChange: value => onNumericChange(value, 'g', 255, void 0, true)
})
}),
vue.h('input', {
value: model.value.g,
maxlength: 3,
readonly: editable.value !== true,
onChange: stop,
...getCache('gIn', {
onInput: evt => onNumericChange(evt.target.value, 'g', 255, evt),
onBlur: evt => onNumericChange(evt.target.value, 'g', 255, evt, true)
})
})
]),
vue.h('div', { class: 'row items-center no-wrap' }, [
vue.h('div', 'B'),
vue.h(QSlider, {
modelValue: model.value.b,
min: 0,
max: 255,
color: 'blue',
readonly: editable.value !== true,
dark: isDark.value,
...getCache('bSlide', {
'onUpdate:modelValue': value => onNumericChange(value, 'b', 255),
onChange: value => onNumericChange(value, 'b', 255, void 0, true)
})
}),
vue.h('input', {
value: model.value.b,
maxlength: 3,
readonly: editable.value !== true,
onChange: stop,
...getCache('bIn', {
onInput: evt => onNumericChange(evt.target.value, 'b', 255, evt),
onBlur: evt => onNumericChange(evt.target.value, 'b', 255, evt, true)
})
})
]),
hasAlpha.value === true ? vue.h('div', { class: 'row items-center no-wrap' }, [
vue.h('div', 'A'),
vue.h(QSlider, {
modelValue: model.value.a,
color: 'grey',
readonly: editable.value !== true,
dark: isDark.value,
...getCache('aSlide', {
'onUpdate:modelValue': value => onNumericChange(value, 'a', 100),
onChange: value => onNumericChange(value, 'a', 100, void 0, true)
})
}),
vue.h('input', {
value: model.value.a,
maxlength: 3,
readonly: editable.value !== true,
onChange: stop,
...getCache('aIn', {
onInput: evt => onNumericChange(evt.target.value, 'a', 100, evt),
onBlur: evt => onNumericChange(evt.target.value, 'a', 100, evt, true)
})
})
]) : null
]
}
function getPaletteTab () {
const fn = color => vue.h('div', {
class: 'q-color-picker__cube col-auto',
style: { backgroundColor: color },
...(
editable.value === true
? getCache('palette#' + color, {
onClick: () => { onPalettePick(color); }
})
: {}
)
});
return [
vue.h('div', {
class: 'row items-center q-color-picker__palette-rows'
+ (editable.value === true ? ' q-color-picker__palette-rows--editable' : '')
}, computedPalette.value.map(fn))
]
}
return () => {
const child = [ getContent() ];
if (props.name !== void 0 && props.disable !== true) {
injectFormInput(child, 'push');
}
props.noHeader !== true && child.unshift(
getHeader()
);
props.noFooter !== true && child.push(
getFooter()
);
return vue.h('div', {
class: classes.value,
...attributes.value
}, child)
}
}
});
// taken from https://github.com/jalaali/jalaali-js
/*
Jalaali years starting the 33-year rule.
*/
const breaks = [
-61, 9, 38, 199, 426, 686, 756, 818, 1111, 1181, 1210,
1635, 2060, 2097, 2192, 2262, 2324, 2394, 2456, 3178
];
/*
Converts a Gregorian date to Jalaali.
*/
function toJalaali (gy, gm, gd) {
if (Object.prototype.toString.call(gy) === '[object Date]') {
gd = gy.getDate();
gm = gy.getMonth() + 1;
gy = gy.getFullYear();
}
return d2j(g2d(gy, gm, gd))
}
/*
Converts a Jalaali date to Gregorian.
*/
function toGregorian (jy, jm, jd) {
return d2g(j2d(jy, jm, jd))
}
/*
Is this a leap year or not?
*/
function isLeapJalaaliYear (jy) {
return jalCalLeap(jy) === 0
}
/*
Number of days in a given month in a Jalaali year.
*/
function jalaaliMonthLength (jy, jm) {
if (jm <= 6) return 31
if (jm <= 11) return 30
if (isLeapJalaaliYear(jy)) return 30
return 29
}
/*
This function determines if the Jalaali (Persian) year is
leap (366-day long) or is the common year (365 days)
@param jy Jalaali calendar year (-61 to 3177)
@returns number of years since the last leap year (0 to 4)
*/
function jalCalLeap (jy) {
const bl = breaks.length;
let
jp = breaks[ 0 ],
jm,
jump,
leap,
n,
i;
if (jy < jp || jy >= breaks[ bl - 1 ]) { throw new Error('Invalid Jalaali year ' + jy) }
for (i = 1; i < bl; i += 1) {
jm = breaks[ i ];
jump = jm - jp;
if (jy < jm) { break }
jp = jm;
}
n = jy - jp;
if (jump - n < 6) { n = n - jump + div(jump + 4, 33) * 33; }
leap = mod(mod(n + 1, 33) - 1, 4);
if (leap === -1) {
leap = 4;
}
return leap
}
/*
This function determines if the Jalaali (Persian) year is
leap (366-day long) or is the common year (365 days), and
finds the day in March (Gregorian calendar) of the first
day of the Jalaali year (jy).
@param jy Jalaali calendar year (-61 to 3177)
@param withoutLeap when don't need leap (true or false) default is false
@return
leap: number of years since the last leap year (0 to 4)
gy: Gregorian year of the beginning of Jalaali year
march: the March day of Farvardin the 1st (1st day of jy)
@see: http://www.astro.uni.torun.pl/~kb/Papers/EMP/PersianC-EMP.htm
@see: http://www.fourmilab.ch/documents/calendar/
*/
function jalCal (jy, withoutLeap) {
const
bl = breaks.length,
gy = jy + 621;
let
leapJ = -14,
jp = breaks[ 0 ],
jm,
jump,
leap,
n,
i;
if (jy < jp || jy >= breaks[ bl - 1 ]) { throw new Error('Invalid Jalaali year ' + jy) }
// Find the limiting years for the Jalaali year jy.
for (i = 1; i < bl; i += 1) {
jm = breaks[ i ];
jump = jm - jp;
if (jy < jm) { break }
leapJ = leapJ + div(jump, 33) * 8 + div(mod(jump, 33), 4);
jp = jm;
}
n = jy - jp;
// Find the number of leap years from AD 621 to the beginning
// of the current Jalaali year in the Persian calendar.
leapJ = leapJ + div(n, 33) * 8 + div(mod(n, 33) + 3, 4);
if (mod(jump, 33) === 4 && jump - n === 4) { leapJ += 1; }
// And the same in the Gregorian calendar (until the year gy).
const leapG = div(gy, 4) - div((div(gy, 100) + 1) * 3, 4) - 150;
// Determine the Gregorian date of Farvardin the 1st.
const march = 20 + leapJ - leapG;
// Find how many years have passed since the last leap year.
if (!withoutLeap) {
if (jump - n < 6) { n = n - jump + div(jump + 4, 33) * 33; }
leap = mod(mod(n + 1, 33) - 1, 4);
if (leap === -1) {
leap = 4;
}
}
return {
leap,
gy,
march
}
}
/*
Converts a date of the Jalaali calendar to the Julian Day number.
@param jy Jalaali year (1 to 3100)
@param jm Jalaali month (1 to 12)
@param jd Jalaali day (1 to 29/31)
@return Julian Day number
*/
function j2d (jy, jm, jd) {
const r = jalCal(jy, true);
return g2d(r.gy, 3, r.march) + (jm - 1) * 31 - div(jm, 7) * (jm - 7) + jd - 1
}
/*
Converts the Julian Day number to a date in the Jalaali calendar.
@param jdn Julian Day number
@return
jy: Jalaali year (1 to 3100)
jm: Jalaali month (1 to 12)
jd: Jalaali day (1 to 29/31)
*/
function d2j (jdn) {
const gy = d2g(jdn).gy; // Calculate Gregorian year (gy).
let
jy = gy - 621,
jd,
jm,
k;
const
r = jalCal(jy, false),
jdn1f = g2d(gy, 3, r.march);
// Find number of days that passed since 1 Farvardin.
k = jdn - jdn1f;
if (k >= 0) {
if (k <= 185) {
// The first 6 months.
jm = 1 + div(k, 31);
jd = mod(k, 31) + 1;
return {
jy,
jm,
jd
}
}
else {
// The remaining months.
k -= 186;
}
}
else {
// Previous Jalaali year.
jy -= 1;
k += 179;
if (r.leap === 1) { k += 1; }
}
jm = 7 + div(k, 30);
jd = mod(k, 30) + 1;
return {
jy,
jm,
jd
}
}
/*
Calculates the Julian Day number from Gregorian or Julian
calendar dates. This integer number corresponds to the noon of
the date (i.e. 12 hours of Universal Time).
The procedure was tested to be good since 1 March, -100100 (of both
calendars) up to a few million years into the future.
@param gy Calendar year (years BC numbered 0, -1, -2, ...)
@param gm Calendar month (1 to 12)
@param gd Calendar day of the month (1 to 28/29/30/31)
@return Julian Day number
*/
function g2d (gy, gm, gd) {
let d = div((gy + div(gm - 8, 6) + 100100) * 1461, 4)
+ div(153 * mod(gm + 9, 12) + 2, 5)
+ gd - 34840408;
d = d - div(div(gy + 100100 + div(gm - 8, 6), 100) * 3, 4) + 752;
return d
}
/*
Calculates Gregorian and Julian calendar dates from the Julian Day number
(jdn) for the period since jdn=-34839655 (i.e. the year -100100 of both
calendars) to some millions years ahead of the present.
@param jdn Julian Day number
@return
gy: Calendar year (years BC numbered 0, -1, -2, ...)
gm: Calendar month (1 to 12)
gd: Calendar day of the month M (1 to 28/29/30/31)
*/
function d2g (jdn) {
let j = 4 * jdn + 139361631;
j = j + div(div(4 * jdn + 183187720, 146097) * 3, 4) * 4 - 3908;
const
i = div(mod(j, 1461), 4) * 5 + 308,
gd = div(mod(i, 153), 5) + 1,
gm = mod(div(i, 153), 12) + 1,
gy = div(j, 1461) - 100100 + div(8 - gm, 6);
return {
gy,
gm,
gd
}
}
/*
Utility helper functions.
*/
function div (a, b) {
return ~~(a / b)
}
function mod (a, b) {
return a - ~~(a / b) * b
}
const calendars = [ 'gregorian', 'persian' ];
const useDatetimeProps = {
modelValue: {
required: true
},
mask: {
type: String
},
locale: Object,
calendar: {
type: String,
validator: v => calendars.includes(v),
default: 'gregorian'
},
landscape: Boolean,
color: String,
textColor: String,
square: Boolean,
flat: Boolean,
bordered: Boolean,
readonly: Boolean,
disable: Boolean
};
const useDatetimeEmits = [ 'update:modelValue' ];
function getDayHash (date) {
return date.year + '/' + pad(date.month) + '/' + pad(date.day)
}
function useDatetime (props, $q) {
const editable = vue.computed(() => {
return props.disable !== true && props.readonly !== true
});
const tabindex = vue.computed(() => {
return editable.value === true ? 0 : -1
});
const headerClass = vue.computed(() => {
const cls = [];
props.color !== void 0 && cls.push(`bg-${ props.color }`);
props.textColor !== void 0 && cls.push(`text-${ props.textColor }`);
return cls.join(' ')
});
function getLocale () {
return props.locale !== void 0
? { ...$q.lang.date, ...props.locale }
: $q.lang.date
}
function getCurrentDate (dateOnly) {
const d = new Date();
const timeFill = dateOnly === true ? null : 0;
if (props.calendar === 'persian') {
const jDate = toJalaali(d);
return {
year: jDate.jy,
month: jDate.jm,
day: jDate.jd
}
}
return {
year: d.getFullYear(),
month: d.getMonth() + 1,
day: d.getDate(),
hour: timeFill,
minute: timeFill,
second: timeFill,
millisecond: timeFill
}
}
return {
editable,
tabindex,
headerClass,
getLocale,
getCurrentDate
}
}
/* eslint no-fallthrough: 0 */
const
MILLISECONDS_IN_DAY = 86400000,
MILLISECONDS_IN_HOUR = 3600000,
MILLISECONDS_IN_MINUTE = 60000,
defaultMask = 'YYYY-MM-DDTHH:mm:ss.SSSZ',
token = /\[((?:[^\]\\]|\\]|\\)*)\]|d{1,4}|M{1,4}|m{1,2}|w{1,2}|Qo|Do|D{1,4}|YY(?:YY)?|H{1,2}|h{1,2}|s{1,2}|S{1,3}|Z{1,2}|a{1,2}|[AQExX]/g,
reverseToken = /(\[[^\]]*\])|d{1,4}|M{1,4}|m{1,2}|w{1,2}|Qo|Do|D{1,4}|YY(?:YY)?|H{1,2}|h{1,2}|s{1,2}|S{1,3}|Z{1,2}|a{1,2}|[AQExX]|([.*+:?^,\s${}()|\\]+)/g,
regexStore = {};
function getRegexData (mask, dateLocale) {
const
days = '(' + dateLocale.days.join('|') + ')',
key = mask + days;
if (regexStore[ key ] !== void 0) {
return regexStore[ key ]
}
const
daysShort = '(' + dateLocale.daysShort.join('|') + ')',
months = '(' + dateLocale.months.join('|') + ')',
monthsShort = '(' + dateLocale.monthsShort.join('|') + ')';
const map = {};
let index = 0;
const regexText = mask.replace(reverseToken, match => {
index++;
switch (match) {
case 'YY':
map.YY = index;
return '(-?\\d{1,2})'
case 'YYYY':
map.YYYY = index;
return '(-?\\d{1,4})'
case 'M':
map.M = index;
return '(\\d{1,2})'
case 'MM':
map.M = index; // bumping to M
return '(\\d{2})'
case 'MMM':
map.MMM = index;
return monthsShort
case 'MMMM':
map.MMMM = index;
return months
case 'D':
map.D = index;
return '(\\d{1,2})'
case 'Do':
map.D = index++; // bumping to D
return '(\\d{1,2}(st|nd|rd|th))'
case 'DD':
map.D = index; // bumping to D
return '(\\d{2})'
case 'H':
map.H = index;
return '(\\d{1,2})'
case 'HH':
map.H = index; // bumping to H
return '(\\d{2})'
case 'h':
map.h = index;
return '(\\d{1,2})'
case 'hh':
map.h = index; // bumping to h
return '(\\d{2})'
case 'm':
map.m = index;
return '(\\d{1,2})'
case 'mm':
map.m = index; // bumping to m
return '(\\d{2})'
case 's':
map.s = index;
return '(\\d{1,2})'
case 'ss':
map.s = index; // bumping to s
return '(\\d{2})'
case 'S':
map.S = index;
return '(\\d{1})'
case 'SS':
map.S = index; // bump to S
return '(\\d{2})'
case 'SSS':
map.S = index; // bump to S
return '(\\d{3})'
case 'A':
map.A = index;
return '(AM|PM)'
case 'a':
map.a = index;
return '(am|pm)'
case 'aa':
map.aa = index;
return '(a\\.m\\.|p\\.m\\.)'
case 'ddd':
return daysShort
case 'dddd':
return days
case 'Q':
case 'd':
case 'E':
return '(\\d{1})'
case 'Qo':
return '(1st|2nd|3rd|4th)'
case 'DDD':
case 'DDDD':
return '(\\d{1,3})'
case 'w':
return '(\\d{1,2})'
case 'ww':
return '(\\d{2})'
case 'Z': // to split: (?:(Z)()()|([+-])?(\\d{2}):?(\\d{2}))
map.Z = index;
return '(Z|[+-]\\d{2}:\\d{2})'
case 'ZZ':
map.ZZ = index;
return '(Z|[+-]\\d{2}\\d{2})'
case 'X':
map.X = index;
return '(-?\\d+)'
case 'x':
map.x = index;
return '(-?\\d{4,})'
default:
index--;
if (match[ 0 ] === '[') {
match = match.substring(1, match.length - 1);
}
return match.replace(/[.*+?^${}()|[\]\\]/g, '\\$&')
}
});
const res = { map, regex: new RegExp('^' + regexText) };
regexStore[ key ] = res;
return res
}
function getDateLocale (paramDateLocale, langProps) {
return paramDateLocale !== void 0
? paramDateLocale
: (
langProps !== void 0
? langProps.date
: defaultLang.date
)
}
function formatTimezone (offset, delimeter = '') {
const
sign = offset > 0 ? '-' : '+',
absOffset = Math.abs(offset),
hours = Math.floor(absOffset / 60),
minutes = absOffset % 60;
return sign + pad(hours) + delimeter + pad(minutes)
}
function applyYearMonthDayChange (date, mod, sign) {
let
year = date.getFullYear(),
month = date.getMonth();
const day = date.getDate();
if (mod.year !== void 0) {
year += sign * mod.year;
delete mod.year;
}
if (mod.month !== void 0) {
month += sign * mod.month;
delete mod.month;
}
date.setDate(1);
date.setMonth(2);
date.setFullYear(year);
date.setMonth(month);
date.setDate(Math.min(day, daysInMonth(date)));
if (mod.date !== void 0) {
date.setDate(date.getDate() + sign * mod.date);
delete mod.date;
}
return date
}
function applyYearMonthDay (date, mod, middle) {
const
year = mod.year !== void 0 ? mod.year : date[ `get${ middle }FullYear` ](),
month = mod.month !== void 0 ? mod.month - 1 : date[ `get${ middle }Month` ](),
maxDay = (new Date(year, month + 1, 0)).getDate(),
day = Math.min(maxDay, mod.date !== void 0 ? mod.date : date[ `get${ middle }Date` ]());
date[ `set${ middle }Date` ](1);
date[ `set${ middle }Month` ](2);
date[ `set${ middle }FullYear` ](year);
date[ `set${ middle }Month` ](month);
date[ `set${ middle }Date` ](day);
delete mod.year;
delete mod.month;
delete mod.date;
return date
}
function getChange (date, rawMod, sign) {
const
mod = normalizeMod(rawMod),
d = new Date(date),
t = mod.year !== void 0 || mod.month !== void 0 || mod.date !== void 0
? applyYearMonthDayChange(d, mod, sign) // removes year/month/day
: d;
for (const key in mod) {
const op = capitalize(key);
t[ `set${ op }` ](t[ `get${ op }` ]() + sign * mod[ key ]);
}
return t
}
function normalizeMod (mod) {
const acc = { ...mod };
if (mod.years !== void 0) {
acc.year = mod.years;
delete acc.years;
}
if (mod.months !== void 0) {
acc.month = mod.months;
delete acc.months;
}
if (mod.days !== void 0) {
acc.date = mod.days;
delete acc.days;
}
if (mod.day !== void 0) {
acc.date = mod.day;
delete acc.day;
}
if (mod.hour !== void 0) {
acc.hours = mod.hour;
delete acc.hour;
}
if (mod.minute !== void 0) {
acc.minutes = mod.minute;
delete acc.minute;
}
if (mod.second !== void 0) {
acc.seconds = mod.second;
delete acc.second;
}
if (mod.millisecond !== void 0) {
acc.milliseconds = mod.millisecond;
delete acc.millisecond;
}
return acc
}
function adjustDate (date, rawMod, utc) {
const
mod = normalizeMod(rawMod),
middle = utc === true ? 'UTC' : '',
d = new Date(date),
t = mod.year !== void 0 || mod.month !== void 0 || mod.date !== void 0
? applyYearMonthDay(d, mod, middle) // removes year/month/day
: d;
for (const key in mod) {
const op = key.charAt(0).toUpperCase() + key.slice(1);
t[ `set${ middle }${ op }` ](mod[ key ]);
}
return t
}
function extractDate (str, mask, dateLocale) {
const d = __splitDate(str, mask, dateLocale);
const date = new Date(
d.year,
d.month === null ? null : d.month - 1,
d.day === null ? 1 : d.day,
d.hour,
d.minute,
d.second,
d.millisecond
);
const tzOffset = date.getTimezoneOffset();
return d.timezoneOffset === null || d.timezoneOffset === tzOffset
? date
: getChange(date, { minutes: d.timezoneOffset - tzOffset }, 1)
}
function __splitDate (str, mask, dateLocale, calendar, defaultModel) {
const date = {
year: null,
month: null,
day: null,
hour: null,
minute: null,
second: null,
millisecond: null,
timezoneOffset: null,
dateHash: null,
timeHash: null
};
defaultModel !== void 0 && Object.assign(date, defaultModel);
if (
str === void 0
|| str === null
|| str === ''
|| typeof str !== 'string'
) {
return date
}
if (mask === void 0) {
mask = defaultMask;
}
const
langOpts = getDateLocale(dateLocale, Plugin$8.props),
months = langOpts.months,
monthsShort = langOpts.monthsShort;
const { regex, map } = getRegexData(mask, langOpts);
const match = str.match(regex);
if (match === null) {
return date
}
let tzString = '';
if (map.X !== void 0 || map.x !== void 0) {
const stamp = parseInt(match[ map.X !== void 0 ? map.X : map.x ], 10);
if (isNaN(stamp) === true || stamp < 0) {
return date
}
const d = new Date(stamp * (map.X !== void 0 ? 1000 : 1));
date.year = d.getFullYear();
date.month = d.getMonth() + 1;
date.day = d.getDate();
date.hour = d.getHours();
date.minute = d.getMinutes();
date.second = d.getSeconds();
date.millisecond = d.getMilliseconds();
}
else {
if (map.YYYY !== void 0) {
date.year = parseInt(match[ map.YYYY ], 10);
}
else if (map.YY !== void 0) {
const y = parseInt(match[ map.YY ], 10);
date.year = y < 0 ? y : 2000 + y;
}
if (map.M !== void 0) {
date.month = parseInt(match[ map.M ], 10);
if (date.month < 1 || date.month > 12) {
return date
}
}
else if (map.MMM !== void 0) {
date.month = monthsShort.indexOf(match[ map.MMM ]) + 1;
}
else if (map.MMMM !== void 0) {
date.month = months.indexOf(match[ map.MMMM ]) + 1;
}
if (map.D !== void 0) {
date.day = parseInt(match[ map.D ], 10);
if (date.year === null || date.month === null || date.day < 1) {
return date
}
const maxDay = calendar !== 'persian'
? (new Date(date.year, date.month, 0)).getDate()
: jalaaliMonthLength(date.year, date.month);
if (date.day > maxDay) {
return date
}
}
if (map.H !== void 0) {
date.hour = parseInt(match[ map.H ], 10) % 24;
}
else if (map.h !== void 0) {
date.hour = parseInt(match[ map.h ], 10) % 12;
if (
(map.A && match[ map.A ] === 'PM')
|| (map.a && match[ map.a ] === 'pm')
|| (map.aa && match[ map.aa ] === 'p.m.')
) {
date.hour += 12;
}
date.hour = date.hour % 24;
}
if (map.m !== void 0) {
date.minute = parseInt(match[ map.m ], 10) % 60;
}
if (map.s !== void 0) {
date.second = parseInt(match[ map.s ], 10) % 60;
}
if (map.S !== void 0) {
date.millisecond = parseInt(match[ map.S ], 10) * 10 ** (3 - match[ map.S ].length);
}
if (map.Z !== void 0 || map.ZZ !== void 0) {
tzString = (map.Z !== void 0 ? match[ map.Z ].replace(':', '') : match[ map.ZZ ]);
date.timezoneOffset = (tzString[ 0 ] === '+' ? -1 : 1) * (60 * tzString.slice(1, 3) + 1 * tzString.slice(3, 5));
}
}
date.dateHash = pad(date.year, 6) + '/' + pad(date.month) + '/' + pad(date.day);
date.timeHash = pad(date.hour) + ':' + pad(date.minute) + ':' + pad(date.second) + tzString;
return date
}
function isValid (date) {
return typeof date === 'number'
? true
: isNaN(Date.parse(date)) === false
}
function buildDate (mod, utc) {
return adjustDate(new Date(), mod, utc)
}
function getDayOfWeek (date) {
const dow = new Date(date).getDay();
return dow === 0 ? 7 : dow
}
function getWeekOfYear (date) {
// Remove time components of date
const thursday = new Date(date.getFullYear(), date.getMonth(), date.getDate());
// Change date to Thursday same week
thursday.setDate(thursday.getDate() - ((thursday.getDay() + 6) % 7) + 3);
// Take January 4th as it is always in week 1 (see ISO 8601)
const firstThursday = new Date(thursday.getFullYear(), 0, 4);
// Change date to Thursday same week
firstThursday.setDate(firstThursday.getDate() - ((firstThursday.getDay() + 6) % 7) + 3);
// Check if daylight-saving-time-switch occurred and correct for it
const ds = thursday.getTimezoneOffset() - firstThursday.getTimezoneOffset();
thursday.setHours(thursday.getHours() - ds);
// Number of weeks between target Thursday and first Thursday
const weekDiff = (thursday - firstThursday) / (MILLISECONDS_IN_DAY * 7);
return 1 + Math.floor(weekDiff)
}
function getDayIdentifier (date) {
return date.getFullYear() * 10000 + date.getMonth() * 100 + date.getDate()
}
function getDateIdentifier (date, onlyDate /* = false */) {
const d = new Date(date);
return onlyDate === true ? getDayIdentifier(d) : d.getTime()
}
function isBetweenDates (date, from, to, opts = {}) {
const
d1 = getDateIdentifier(from, opts.onlyDate),
d2 = getDateIdentifier(to, opts.onlyDate),
cur = getDateIdentifier(date, opts.onlyDate);
return (cur > d1 || (opts.inclusiveFrom === true && cur === d1))
&& (cur < d2 || (opts.inclusiveTo === true && cur === d2))
}
function addToDate (date, mod) {
return getChange(date, mod, 1)
}
function subtractFromDate (date, mod) {
return getChange(date, mod, -1)
}
function startOfDate (date, unit, utc) {
const
t = new Date(date),
prefix = `set${ utc === true ? 'UTC' : '' }`;
switch (unit) {
case 'year':
case 'years':
t[ `${ prefix }Month` ](0);
case 'month':
case 'months':
t[ `${ prefix }Date` ](1);
case 'day':
case 'days':
case 'date':
t[ `${ prefix }Hours` ](0);
case 'hour':
case 'hours':
t[ `${ prefix }Minutes` ](0);
case 'minute':
case 'minutes':
t[ `${ prefix }Seconds` ](0);
case 'second':
case 'seconds':
t[ `${ prefix }Milliseconds` ](0);
}
return t
}
function endOfDate (date, unit, utc) {
const
t = new Date(date),
prefix = `set${ utc === true ? 'UTC' : '' }`;
switch (unit) {
case 'year':
case 'years':
t[ `${ prefix }Month` ](11);
case 'month':
case 'months':
t[ `${ prefix }Date` ](daysInMonth(t));
case 'day':
case 'days':
case 'date':
t[ `${ prefix }Hours` ](23);
case 'hour':
case 'hours':
t[ `${ prefix }Minutes` ](59);
case 'minute':
case 'minutes':
t[ `${ prefix }Seconds` ](59);
case 'second':
case 'seconds':
t[ `${ prefix }Milliseconds` ](999);
}
return t
}
function getMaxDate (date /* , ...args */) {
let t = new Date(date);
Array.prototype.slice.call(arguments, 1).forEach(d => {
t = Math.max(t, new Date(d));
});
return t
}
function getMinDate (date /*, ...args */) {
let t = new Date(date);
Array.prototype.slice.call(arguments, 1).forEach(d => {
t = Math.min(t, new Date(d));
});
return t
}
function getDiff (t, sub, interval) {
return (
(t.getTime() - t.getTimezoneOffset() * MILLISECONDS_IN_MINUTE)
- (sub.getTime() - sub.getTimezoneOffset() * MILLISECONDS_IN_MINUTE)
) / interval
}
function getDateDiff (date, subtract, unit = 'days') {
const
t = new Date(date),
sub = new Date(subtract);
switch (unit) {
case 'years':
case 'year':
return (t.getFullYear() - sub.getFullYear())
case 'months':
case 'month':
return (t.getFullYear() - sub.getFullYear()) * 12 + t.getMonth() - sub.getMonth()
case 'days':
case 'day':
case 'date':
return getDiff(startOfDate(t, 'day'), startOfDate(sub, 'day'), MILLISECONDS_IN_DAY)
case 'hours':
case 'hour':
return getDiff(startOfDate(t, 'hour'), startOfDate(sub, 'hour'), MILLISECONDS_IN_HOUR)
case 'minutes':
case 'minute':
return getDiff(startOfDate(t, 'minute'), startOfDate(sub, 'minute'), MILLISECONDS_IN_MINUTE)
case 'seconds':
case 'second':
return getDiff(startOfDate(t, 'second'), startOfDate(sub, 'second'), 1000)
}
}
function getDayOfYear (date) {
return getDateDiff(date, startOfDate(date, 'year'), 'days') + 1
}
function inferDateFormat (date) {
return isDate(date) === true
? 'date'
: (typeof date === 'number' ? 'number' : 'string')
}
function getDateBetween (date, min, max) {
const t = new Date(date);
if (min) {
const low = new Date(min);
if (t < low) {
return low
}
}
if (max) {
const high = new Date(max);
if (t > high) {
return high
}
}
return t
}
function isSameDate (date, date2, unit) {
const
t = new Date(date),
d = new Date(date2);
if (unit === void 0) {
return t.getTime() === d.getTime()
}
switch (unit) {
case 'second':
case 'seconds':
if (t.getSeconds() !== d.getSeconds()) {
return false
}
case 'minute': // intentional fall-through
case 'minutes':
if (t.getMinutes() !== d.getMinutes()) {
return false
}
case 'hour': // intentional fall-through
case 'hours':
if (t.getHours() !== d.getHours()) {
return false
}
case 'day': // intentional fall-through
case 'days':
case 'date':
if (t.getDate() !== d.getDate()) {
return false
}
case 'month': // intentional fall-through
case 'months':
if (t.getMonth() !== d.getMonth()) {
return false
}
case 'year': // intentional fall-through
case 'years':
if (t.getFullYear() !== d.getFullYear()) {
return false
}
break
default:
throw new Error(`date isSameDate unknown unit ${ unit }`)
}
return true
}
function daysInMonth (date) {
return (new Date(date.getFullYear(), date.getMonth() + 1, 0)).getDate()
}
function getOrdinal (n) {
if (n >= 11 && n <= 13) {
return `${ n }th`
}
switch (n % 10) {
case 1: return `${ n }st`
case 2: return `${ n }nd`
case 3: return `${ n }rd`
}
return `${ n }th`
}
const formatter = {
// Year: 00, 01, ..., 99
YY (date, dateLocale, forcedYear) {
// workaround for < 1900 with new Date()
const y = this.YYYY(date, dateLocale, forcedYear) % 100;
return y >= 0
? pad(y)
: '-' + pad(Math.abs(y))
},
// Year: 1900, 1901, ..., 2099
YYYY (date, _dateLocale, forcedYear) {
// workaround for < 1900 with new Date()
return forcedYear !== void 0 && forcedYear !== null
? forcedYear
: date.getFullYear()
},
// Month: 1, 2, ..., 12
M (date) {
return date.getMonth() + 1
},
// Month: 01, 02, ..., 12
MM (date) {
return pad(date.getMonth() + 1)
},
// Month Short Name: Jan, Feb, ...
MMM (date, dateLocale) {
return dateLocale.monthsShort[ date.getMonth() ]
},
// Month Name: January, February, ...
MMMM (date, dateLocale) {
return dateLocale.months[ date.getMonth() ]
},
// Quarter: 1, 2, 3, 4
Q (date) {
return Math.ceil((date.getMonth() + 1) / 3)
},
// Quarter: 1st, 2nd, 3rd, 4th
Qo (date) {
return getOrdinal(this.Q(date))
},
// Day of month: 1, 2, ..., 31
D (date) {
return date.getDate()
},
// Day of month: 1st, 2nd, ..., 31st
Do (date) {
return getOrdinal(date.getDate())
},
// Day of month: 01, 02, ..., 31
DD (date) {
return pad(date.getDate())
},
// Day of year: 1, 2, ..., 366
DDD (date) {
return getDayOfYear(date)
},
// Day of year: 001, 002, ..., 366
DDDD (date) {
return pad(getDayOfYear(date), 3)
},
// Day of week: 0, 1, ..., 6
d (date) {
return date.getDay()
},
// Day of week: Su, Mo, ...
dd (date, dateLocale) {
return this.dddd(date, dateLocale).slice(0, 2)
},
// Day of week: Sun, Mon, ...
ddd (date, dateLocale) {
return dateLocale.daysShort[ date.getDay() ]
},
// Day of week: Sunday, Monday, ...
dddd (date, dateLocale) {
return dateLocale.days[ date.getDay() ]
},
// Day of ISO week: 1, 2, ..., 7
E (date) {
return date.getDay() || 7
},
// Week of Year: 1 2 ... 52 53
w (date) {
return getWeekOfYear(date)
},
// Week of Year: 01 02 ... 52 53
ww (date) {
return pad(getWeekOfYear(date))
},
// Hour: 0, 1, ... 23
H (date) {
return date.getHours()
},
// Hour: 00, 01, ..., 23
HH (date) {
return pad(date.getHours())
},
// Hour: 1, 2, ..., 12
h (date) {
const hours = date.getHours();
return hours === 0
? 12
: (hours > 12 ? hours % 12 : hours)
},
// Hour: 01, 02, ..., 12
hh (date) {
return pad(this.h(date))
},
// Minute: 0, 1, ..., 59
m (date) {
return date.getMinutes()
},
// Minute: 00, 01, ..., 59
mm (date) {
return pad(date.getMinutes())
},
// Second: 0, 1, ..., 59
s (date) {
return date.getSeconds()
},
// Second: 00, 01, ..., 59
ss (date) {
return pad(date.getSeconds())
},
// 1/10 of second: 0, 1, ..., 9
S (date) {
return Math.floor(date.getMilliseconds() / 100)
},
// 1/100 of second: 00, 01, ..., 99
SS (date) {
return pad(Math.floor(date.getMilliseconds() / 10))
},
// Millisecond: 000, 001, ..., 999
SSS (date) {
return pad(date.getMilliseconds(), 3)
},
// Meridiem: AM, PM
A (date) {
return this.H(date) < 12 ? 'AM' : 'PM'
},
// Meridiem: am, pm
a (date) {
return this.H(date) < 12 ? 'am' : 'pm'
},
// Meridiem: a.m., p.m.
aa (date) {
return this.H(date) < 12 ? 'a.m.' : 'p.m.'
},
// Timezone: -01:00, +00:00, ... +12:00
Z (date, _dateLocale, _forcedYear, forcedTimezoneOffset) {
const tzOffset = forcedTimezoneOffset === void 0 || forcedTimezoneOffset === null
? date.getTimezoneOffset()
: forcedTimezoneOffset;
return formatTimezone(tzOffset, ':')
},
// Timezone: -0100, +0000, ... +1200
ZZ (date, _dateLocale, _forcedYear, forcedTimezoneOffset) {
const tzOffset = forcedTimezoneOffset === void 0 || forcedTimezoneOffset === null
? date.getTimezoneOffset()
: forcedTimezoneOffset;
return formatTimezone(tzOffset)
},
// Seconds timestamp: 512969520
X (date) {
return Math.floor(date.getTime() / 1000)
},
// Milliseconds timestamp: 512969520900
x (date) {
return date.getTime()
}
};
function formatDate (val, mask, dateLocale, __forcedYear, __forcedTimezoneOffset) {
if (
(val !== 0 && !val)
|| val === Infinity
|| val === -Infinity
) {
return
}
const date = new Date(val);
if (isNaN(date)) {
return
}
if (mask === void 0) {
mask = defaultMask;
}
const locale = getDateLocale(dateLocale, Plugin$8.props);
return mask.replace(
token,
(match, text) => (
match in formatter
? formatter[ match ](date, locale, __forcedYear, __forcedTimezoneOffset)
: (text === void 0 ? match : text.split('\\]').join(']'))
)
)
}
function clone (date) {
return isDate(date) === true
? new Date(date.getTime())
: date
}
var date = {
isValid,
extractDate,
buildDate,
getDayOfWeek,
getWeekOfYear,
isBetweenDates,
addToDate,
subtractFromDate,
adjustDate,
startOfDate,
endOfDate,
getMaxDate,
getMinDate,
getDateDiff,
getDayOfYear,
inferDateFormat,
getDateBetween,
isSameDate,
daysInMonth,
formatDate,
clone
};
const yearsInterval = 20;
const views = [ 'Calendar', 'Years', 'Months' ];
const viewIsValid = v => views.includes(v);
const yearMonthValidator = v => /^-?[\d]+\/[0-1]\d$/.test(v);
const lineStr = ' \u2014 ';
function getMonthHash (date) {
return date.year + '/' + pad(date.month)
}
var QDate = createComponent({
name: 'QDate',
props: {
...useDatetimeProps,
...useFormProps,
...useDarkProps,
multiple: Boolean,
range: Boolean,
title: String,
subtitle: String,
mask: {
// this mask is forced
// when using persian calendar
default: 'YYYY/MM/DD'
},
defaultYearMonth: {
type: String,
validator: yearMonthValidator
},
yearsInMonthView: Boolean,
events: [ Array, Function ],
eventColor: [ String, Function ],
emitImmediately: Boolean,
options: [ Array, Function ],
navigationMinYearMonth: {
type: String,
validator: yearMonthValidator
},
navigationMaxYearMonth: {
type: String,
validator: yearMonthValidator
},
noUnset: Boolean,
firstDayOfWeek: [ String, Number ],
todayBtn: Boolean,
minimal: Boolean,
defaultView: {
type: String,
default: 'Calendar',
validator: viewIsValid
}
},
emits: [
...useDatetimeEmits,
'rangeStart', 'rangeEnd', 'navigation'
],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const { getCache } = useCache();
const { tabindex, headerClass, getLocale, getCurrentDate } = useDatetime(props, $q);
let lastEmitValue;
const formAttrs = useFormAttrs(props);
const injectFormInput = useFormInject(formAttrs);
const blurTargetRef = vue.ref(null);
const innerMask = vue.ref(getMask());
const innerLocale = vue.ref(getLocale());
const mask = vue.computed(() => getMask());
const locale = vue.computed(() => getLocale());
const today = vue.computed(() => getCurrentDate());
// model of current calendar view:
const viewModel = vue.ref(getViewModel(innerMask.value, innerLocale.value));
const view = vue.ref(props.defaultView);
const direction = $q.lang.rtl === true ? 'right' : 'left';
const monthDirection = vue.ref(direction.value);
const yearDirection = vue.ref(direction.value);
const year = viewModel.value.year;
const startYear = vue.ref(year - (year % yearsInterval) - (year < 0 ? yearsInterval : 0));
const editRange = vue.ref(null);
const classes = vue.computed(() => {
const type = props.landscape === true ? 'landscape' : 'portrait';
return `q-date q-date--${ type } q-date--${ type }-${ props.minimal === true ? 'minimal' : 'standard' }`
+ (isDark.value === true ? ' q-date--dark q-dark' : '')
+ (props.bordered === true ? ' q-date--bordered' : '')
+ (props.square === true ? ' q-date--square no-border-radius' : '')
+ (props.flat === true ? ' q-date--flat no-shadow' : '')
+ (props.disable === true ? ' disabled' : (props.readonly === true ? ' q-date--readonly' : ''))
});
const computedColor = vue.computed(() => {
return props.color || 'primary'
});
const computedTextColor = vue.computed(() => {
return props.textColor || 'white'
});
const isImmediate = vue.computed(() =>
props.emitImmediately === true
&& props.multiple !== true
&& props.range !== true
);
const normalizedModel = vue.computed(() => (
Array.isArray(props.modelValue) === true
? props.modelValue
: (props.modelValue !== null && props.modelValue !== void 0 ? [ props.modelValue ] : [])
));
const daysModel = vue.computed(() =>
normalizedModel.value
.filter(date => typeof date === 'string')
.map(date => decodeString(date, innerMask.value, innerLocale.value))
.filter(date =>
date.dateHash !== null
&& date.day !== null
&& date.month !== null
&& date.year !== null
)
);
const rangeModel = vue.computed(() => {
const fn = date => decodeString(date, innerMask.value, innerLocale.value);
return normalizedModel.value
.filter(date => isObject(date) === true && date.from !== void 0 && date.to !== void 0)
.map(range => ({ from: fn(range.from), to: fn(range.to) }))
.filter(range => range.from.dateHash !== null && range.to.dateHash !== null && range.from.dateHash < range.to.dateHash)
});
const getNativeDateFn = vue.computed(() => (
props.calendar !== 'persian'
? model => new Date(model.year, model.month - 1, model.day)
: model => {
const gDate = toGregorian(model.year, model.month, model.day);
return new Date(gDate.gy, gDate.gm - 1, gDate.gd)
}
));
const encodeObjectFn = vue.computed(() => (
props.calendar === 'persian'
? getDayHash
: (date, mask, locale) => formatDate(
new Date(
date.year,
date.month - 1,
date.day,
date.hour,
date.minute,
date.second,
date.millisecond
),
mask === void 0 ? innerMask.value : mask,
locale === void 0 ? innerLocale.value : locale,
date.year,
date.timezoneOffset
)
));
const daysInModel = vue.computed(() =>
daysModel.value.length + rangeModel.value.reduce(
(acc, range) => acc + 1 + getDateDiff(
getNativeDateFn.value(range.to),
getNativeDateFn.value(range.from)
),
0
)
);
const headerTitle = vue.computed(() => {
if (props.title !== void 0 && props.title !== null && props.title.length !== 0) {
return props.title
}
if (editRange.value !== null) {
const model = editRange.value.init;
const date = getNativeDateFn.value(model);
return innerLocale.value.daysShort[ date.getDay() ] + ', '
+ innerLocale.value.monthsShort[ model.month - 1 ] + ' '
+ model.day + lineStr + '?'
}
if (daysInModel.value === 0) {
return lineStr
}
if (daysInModel.value > 1) {
return `${ daysInModel.value } ${ innerLocale.value.pluralDay }`
}
const model = daysModel.value[ 0 ];
const date = getNativeDateFn.value(model);
if (isNaN(date.valueOf()) === true) {
return lineStr
}
if (innerLocale.value.headerTitle !== void 0) {
return innerLocale.value.headerTitle(date, model)
}
return innerLocale.value.daysShort[ date.getDay() ] + ', '
+ innerLocale.value.monthsShort[ model.month - 1 ] + ' '
+ model.day
});
const minSelectedModel = vue.computed(() => {
const model = daysModel.value.concat(rangeModel.value.map(range => range.from))
.sort((a, b) => a.year - b.year || a.month - b.month);
return model[ 0 ]
});
const maxSelectedModel = vue.computed(() => {
const model = daysModel.value.concat(rangeModel.value.map(range => range.to))
.sort((a, b) => b.year - a.year || b.month - a.month);
return model[ 0 ]
});
const headerSubtitle = vue.computed(() => {
if (props.subtitle !== void 0 && props.subtitle !== null && props.subtitle.length !== 0) {
return props.subtitle
}
if (daysInModel.value === 0) {
return lineStr
}
if (daysInModel.value > 1) {
const from = minSelectedModel.value;
const to = maxSelectedModel.value;
const month = innerLocale.value.monthsShort;
return month[ from.month - 1 ] + (
from.year !== to.year
? ' ' + from.year + lineStr + month[ to.month - 1 ] + ' '
: (
from.month !== to.month
? lineStr + month[ to.month - 1 ]
: ''
)
) + ' ' + to.year
}
return daysModel.value[ 0 ].year
});
const dateArrow = vue.computed(() => {
const val = [ $q.iconSet.datetime.arrowLeft, $q.iconSet.datetime.arrowRight ];
return $q.lang.rtl === true ? val.reverse() : val
});
const computedFirstDayOfWeek = vue.computed(() => (
props.firstDayOfWeek !== void 0
? Number(props.firstDayOfWeek)
: innerLocale.value.firstDayOfWeek
));
const daysOfWeek = vue.computed(() => {
const
days = innerLocale.value.daysShort,
first = computedFirstDayOfWeek.value;
return first > 0
? days.slice(first, 7).concat(days.slice(0, first))
: days
});
const daysInMonth = vue.computed(() => {
const date = viewModel.value;
return props.calendar !== 'persian'
? (new Date(date.year, date.month, 0)).getDate()
: jalaaliMonthLength(date.year, date.month)
});
const evtColor = vue.computed(() => (
typeof props.eventColor === 'function'
? props.eventColor
: () => props.eventColor
));
const minNav = vue.computed(() => {
if (props.navigationMinYearMonth === void 0) {
return null
}
const data = props.navigationMinYearMonth.split('/');
return { year: parseInt(data[ 0 ], 10), month: parseInt(data[ 1 ], 10) }
});
const maxNav = vue.computed(() => {
if (props.navigationMaxYearMonth === void 0) {
return null
}
const data = props.navigationMaxYearMonth.split('/');
return { year: parseInt(data[ 0 ], 10), month: parseInt(data[ 1 ], 10) }
});
const navBoundaries = vue.computed(() => {
const data = {
month: { prev: true, next: true },
year: { prev: true, next: true }
};
if (minNav.value !== null && minNav.value.year >= viewModel.value.year) {
data.year.prev = false;
if (minNav.value.year === viewModel.value.year && minNav.value.month >= viewModel.value.month) {
data.month.prev = false;
}
}
if (maxNav.value !== null && maxNav.value.year <= viewModel.value.year) {
data.year.next = false;
if (maxNav.value.year === viewModel.value.year && maxNav.value.month <= viewModel.value.month) {
data.month.next = false;
}
}
return data
});
const daysMap = vue.computed(() => {
const map = {};
daysModel.value.forEach(entry => {
const hash = getMonthHash(entry);
if (map[ hash ] === void 0) {
map[ hash ] = [];
}
map[ hash ].push(entry.day);
});
return map
});
const rangeMap = vue.computed(() => {
const map = {};
rangeModel.value.forEach(entry => {
const hashFrom = getMonthHash(entry.from);
const hashTo = getMonthHash(entry.to);
if (map[ hashFrom ] === void 0) {
map[ hashFrom ] = [];
}
map[ hashFrom ].push({
from: entry.from.day,
to: hashFrom === hashTo ? entry.to.day : void 0,
range: entry
});
if (hashFrom < hashTo) {
let hash;
const { year, month } = entry.from;
const cur = month < 12
? { year, month: month + 1 }
: { year: year + 1, month: 1 };
while ((hash = getMonthHash(cur)) <= hashTo) {
if (map[ hash ] === void 0) {
map[ hash ] = [];
}
map[ hash ].push({
from: void 0,
to: hash === hashTo ? entry.to.day : void 0,
range: entry
});
cur.month++;
if (cur.month > 12) {
cur.year++;
cur.month = 1;
}
}
}
});
return map
});
const rangeView = vue.computed(() => {
if (editRange.value === null) {
return
}
const { init, initHash, final, finalHash } = editRange.value;
const [ from, to ] = initHash <= finalHash
? [ init, final ]
: [ final, init ];
const fromHash = getMonthHash(from);
const toHash = getMonthHash(to);
if (fromHash !== viewMonthHash.value && toHash !== viewMonthHash.value) {
return
}
const view = {};
if (fromHash === viewMonthHash.value) {
view.from = from.day;
view.includeFrom = true;
}
else {
view.from = 1;
}
if (toHash === viewMonthHash.value) {
view.to = to.day;
view.includeTo = true;
}
else {
view.to = daysInMonth.value;
}
return view
});
const viewMonthHash = vue.computed(() => getMonthHash(viewModel.value));
const selectionDaysMap = vue.computed(() => {
const map = {};
if (props.options === void 0) {
for (let i = 1; i <= daysInMonth.value; i++) {
map[ i ] = true;
}
return map
}
const fn = typeof props.options === 'function'
? props.options
: date => props.options.includes(date);
for (let i = 1; i <= daysInMonth.value; i++) {
const dayHash = viewMonthHash.value + '/' + pad(i);
map[ i ] = fn(dayHash);
}
return map
});
const eventDaysMap = vue.computed(() => {
const map = {};
if (props.events === void 0) {
for (let i = 1; i <= daysInMonth.value; i++) {
map[ i ] = false;
}
}
else {
const fn = typeof props.events === 'function'
? props.events
: date => props.events.includes(date);
for (let i = 1; i <= daysInMonth.value; i++) {
const dayHash = viewMonthHash.value + '/' + pad(i);
map[ i ] = fn(dayHash) === true && evtColor.value(dayHash);
}
}
return map
});
const viewDays = vue.computed(() => {
let date, endDay;
const { year, month } = viewModel.value;
if (props.calendar !== 'persian') {
date = new Date(year, month - 1, 1);
endDay = (new Date(year, month - 1, 0)).getDate();
}
else {
const gDate = toGregorian(year, month, 1);
date = new Date(gDate.gy, gDate.gm - 1, gDate.gd);
let prevJM = month - 1;
let prevJY = year;
if (prevJM === 0) {
prevJM = 12;
prevJY--;
}
endDay = jalaaliMonthLength(prevJY, prevJM);
}
return {
days: date.getDay() - computedFirstDayOfWeek.value - 1,
endDay
}
});
const days = vue.computed(() => {
const res = [];
const { days, endDay } = viewDays.value;
const len = days < 0 ? days + 7 : days;
if (len < 6) {
for (let i = endDay - len; i <= endDay; i++) {
res.push({ i, fill: true });
}
}
const index = res.length;
for (let i = 1; i <= daysInMonth.value; i++) {
const day = { i, event: eventDaysMap.value[ i ], classes: [] };
if (selectionDaysMap.value[ i ] === true) {
day.in = true;
day.flat = true;
}
res.push(day);
}
// if current view has days in model
if (daysMap.value[ viewMonthHash.value ] !== void 0) {
daysMap.value[ viewMonthHash.value ].forEach(day => {
const i = index + day - 1;
Object.assign(res[ i ], {
selected: true,
unelevated: true,
flat: false,
color: computedColor.value,
textColor: computedTextColor.value
});
});
}
// if current view has ranges in model
if (rangeMap.value[ viewMonthHash.value ] !== void 0) {
rangeMap.value[ viewMonthHash.value ].forEach(entry => {
if (entry.from !== void 0) {
const from = index + entry.from - 1;
const to = index + (entry.to || daysInMonth.value) - 1;
for (let day = from; day <= to; day++) {
Object.assign(res[ day ], {
range: entry.range,
unelevated: true,
color: computedColor.value,
textColor: computedTextColor.value
});
}
Object.assign(res[ from ], {
rangeFrom: true,
flat: false
});
entry.to !== void 0 && Object.assign(res[ to ], {
rangeTo: true,
flat: false
});
}
else if (entry.to !== void 0) {
const to = index + entry.to - 1;
for (let day = index; day <= to; day++) {
Object.assign(res[ day ], {
range: entry.range,
unelevated: true,
color: computedColor.value,
textColor: computedTextColor.value
});
}
Object.assign(res[ to ], {
flat: false,
rangeTo: true
});
}
else {
const to = index + daysInMonth.value - 1;
for (let day = index; day <= to; day++) {
Object.assign(res[ day ], {
range: entry.range,
unelevated: true,
color: computedColor.value,
textColor: computedTextColor.value
});
}
}
});
}
if (rangeView.value !== void 0) {
const from = index + rangeView.value.from - 1;
const to = index + rangeView.value.to - 1;
for (let day = from; day <= to; day++) {
res[ day ].color = computedColor.value;
res[ day ].editRange = true;
}
if (rangeView.value.includeFrom === true) {
res[ from ].editRangeFrom = true;
}
if (rangeView.value.includeTo === true) {
res[ to ].editRangeTo = true;
}
}
if (viewModel.value.year === today.value.year && viewModel.value.month === today.value.month) {
res[ index + today.value.day - 1 ].today = true;
}
const left = res.length % 7;
if (left > 0) {
const afterDays = 7 - left;
for (let i = 1; i <= afterDays; i++) {
res.push({ i, fill: true });
}
}
res.forEach(day => {
let cls = 'q-date__calendar-item ';
if (day.fill === true) {
cls += 'q-date__calendar-item--fill';
}
else {
cls += `q-date__calendar-item--${ day.in === true ? 'in' : 'out' }`;
if (day.range !== void 0) {
cls += ` q-date__range${ day.rangeTo === true ? '-to' : (day.rangeFrom === true ? '-from' : '') }`;
}
if (day.editRange === true) {
cls += ` q-date__edit-range${ day.editRangeFrom === true ? '-from' : '' }${ day.editRangeTo === true ? '-to' : '' }`;
}
if (day.range !== void 0 || day.editRange === true) {
cls += ` text-${ day.color }`;
}
}
day.classes = cls;
});
return res
});
const attributes = vue.computed(() => (
props.disable === true
? { 'aria-disabled': 'true' }
: (props.readonly === true ? { 'aria-readonly': 'true' } : {})
));
vue.watch(() => props.modelValue, v => {
if (lastEmitValue === v) {
lastEmitValue = 0;
}
else {
const model = getViewModel(innerMask.value, innerLocale.value);
updateViewModel(model.year, model.month, model);
}
});
vue.watch(view, () => {
if (blurTargetRef.value !== null && proxy.$el.contains(document.activeElement) === true) {
blurTargetRef.value.focus();
}
});
vue.watch(() => viewModel.value.year + '|' + viewModel.value.month, () => {
emit('navigation', { year: viewModel.value.year, month: viewModel.value.month });
});
vue.watch(mask, val => {
updateValue(val, innerLocale.value, 'mask');
innerMask.value = val;
});
vue.watch(locale, val => {
updateValue(innerMask.value, val, 'locale');
innerLocale.value = val;
});
function setToday () {
const date = today.value;
const month = daysMap.value[ getMonthHash(date) ];
if (month === void 0 || month.includes(date.day) === false) {
addToModel(date);
}
setCalendarTo(date.year, date.month);
}
function setView (viewMode) {
if (viewIsValid(viewMode) === true) {
view.value = viewMode;
}
}
function offsetCalendar (type, descending) {
if ([ 'month', 'year' ].includes(type)) {
const fn = type === 'month' ? goToMonth : goToYear;
fn(descending === true ? -1 : 1);
}
}
function setCalendarTo (year, month) {
view.value = 'Calendar';
updateViewModel(year, month);
}
function setEditingRange (from, to) {
if (props.range === false || !from) {
editRange.value = null;
return
}
const init = Object.assign({ ...viewModel.value }, from);
const final = to !== void 0
? Object.assign({ ...viewModel.value }, to)
: init;
editRange.value = {
init,
initHash: getDayHash(init),
final,
finalHash: getDayHash(final)
};
setCalendarTo(init.year, init.month);
}
function getMask () {
return props.calendar === 'persian' ? 'YYYY/MM/DD' : props.mask
}
function decodeString (date, mask, locale) {
return __splitDate(
date,
mask,
locale,
props.calendar,
{
hour: 0,
minute: 0,
second: 0,
millisecond: 0
}
)
}
function getViewModel (mask, locale) {
const model = Array.isArray(props.modelValue) === true
? props.modelValue
: (props.modelValue ? [ props.modelValue ] : []);
if (model.length === 0) {
return getDefaultViewModel()
}
const target = model[ model.length - 1 ];
const decoded = decodeString(
target.from !== void 0 ? target.from : target,
mask,
locale
);
return decoded.dateHash === null
? getDefaultViewModel()
: decoded
}
function getDefaultViewModel () {
let year, month;
if (props.defaultYearMonth !== void 0) {
const d = props.defaultYearMonth.split('/');
year = parseInt(d[ 0 ], 10);
month = parseInt(d[ 1 ], 10);
}
else {
// may come from data() where computed
// props are not yet available
const d = today.value !== void 0
? today.value
: getCurrentDate();
year = d.year;
month = d.month;
}
return {
year,
month,
day: 1,
hour: 0,
minute: 0,
second: 0,
millisecond: 0,
dateHash: year + '/' + pad(month) + '/01'
}
}
function goToMonth (offset) {
let year = viewModel.value.year;
let month = Number(viewModel.value.month) + offset;
if (month === 13) {
month = 1;
year++;
}
else if (month === 0) {
month = 12;
year--;
}
updateViewModel(year, month);
isImmediate.value === true && emitImmediately('month');
}
function goToYear (offset) {
const year = Number(viewModel.value.year) + offset;
updateViewModel(year, viewModel.value.month);
isImmediate.value === true && emitImmediately('year');
}
function setYear (year) {
updateViewModel(year, viewModel.value.month);
view.value = props.defaultView === 'Years' ? 'Months' : 'Calendar';
isImmediate.value === true && emitImmediately('year');
}
function setMonth (month) {
updateViewModel(viewModel.value.year, month);
view.value = 'Calendar';
isImmediate.value === true && emitImmediately('month');
}
function toggleDate (date, monthHash) {
const month = daysMap.value[ monthHash ];
const fn = month !== void 0 && month.includes(date.day) === true
? removeFromModel
: addToModel;
fn(date);
}
function getShortDate (date) {
return { year: date.year, month: date.month, day: date.day }
}
function updateViewModel (year, month, time) {
if (minNav.value !== null && year <= minNav.value.year) {
year = minNav.value.year;
if (month < minNav.value.month) {
month = minNav.value.month;
}
}
if (maxNav.value !== null && year >= maxNav.value.year) {
year = maxNav.value.year;
if (month > maxNav.value.month) {
month = maxNav.value.month;
}
}
if (time !== void 0) {
const { hour, minute, second, millisecond, timezoneOffset, timeHash } = time;
Object.assign(viewModel.value, { hour, minute, second, millisecond, timezoneOffset, timeHash });
}
const newHash = year + '/' + pad(month) + '/01';
if (newHash !== viewModel.value.dateHash) {
monthDirection.value = (viewModel.value.dateHash < newHash) === ($q.lang.rtl !== true) ? 'left' : 'right';
if (year !== viewModel.value.year) {
yearDirection.value = monthDirection.value;
}
vue.nextTick(() => {
startYear.value = year - year % yearsInterval - (year < 0 ? yearsInterval : 0);
Object.assign(viewModel.value, {
year,
month,
day: 1,
dateHash: newHash
});
});
}
}
function emitValue (val, action, date) {
const value = val !== null && val.length === 1 && props.multiple === false
? val[ 0 ]
: val;
lastEmitValue = value;
const { reason, details } = getEmitParams(action, date);
emit('update:modelValue', value, reason, details);
}
function emitImmediately (reason) {
const date = daysModel.value[ 0 ] !== void 0 && daysModel.value[ 0 ].dateHash !== null
? { ...daysModel.value[ 0 ] }
: { ...viewModel.value }; // inherit day, hours, minutes, milliseconds...
// nextTick required because of animation delay in viewModel
vue.nextTick(() => {
date.year = viewModel.value.year;
date.month = viewModel.value.month;
const maxDay = props.calendar !== 'persian'
? (new Date(date.year, date.month, 0)).getDate()
: jalaaliMonthLength(date.year, date.month);
date.day = Math.min(Math.max(1, date.day), maxDay);
const value = encodeEntry(date);
lastEmitValue = value;
const { details } = getEmitParams('', date);
emit('update:modelValue', value, reason, details);
});
}
function getEmitParams (action, date) {
return date.from !== void 0
? {
reason: `${ action }-range`,
details: {
...getShortDate(date.target),
from: getShortDate(date.from),
to: getShortDate(date.to)
}
}
: {
reason: `${ action }-day`,
details: getShortDate(date)
}
}
function encodeEntry (date, mask, locale) {
return date.from !== void 0
? { from: encodeObjectFn.value(date.from, mask, locale), to: encodeObjectFn.value(date.to, mask, locale) }
: encodeObjectFn.value(date, mask, locale)
}
function addToModel (date) {
let value;
if (props.multiple === true) {
if (date.from !== void 0) {
// we also need to filter out intersections
const fromHash = getDayHash(date.from);
const toHash = getDayHash(date.to);
const days = daysModel.value
.filter(day => day.dateHash < fromHash || day.dateHash > toHash);
const ranges = rangeModel.value
.filter(({ from, to }) => to.dateHash < fromHash || from.dateHash > toHash);
value = days.concat(ranges).concat(date).map(entry => encodeEntry(entry));
}
else {
const model = normalizedModel.value.slice();
model.push(encodeEntry(date));
value = model;
}
}
else {
value = encodeEntry(date);
}
emitValue(value, 'add', date);
}
function removeFromModel (date) {
if (props.noUnset === true) {
return
}
let model = null;
if (props.multiple === true && Array.isArray(props.modelValue) === true) {
const val = encodeEntry(date);
if (date.from !== void 0) {
model = props.modelValue.filter(
date => (
date.from !== void 0
? (date.from !== val.from && date.to !== val.to)
: true
)
);
}
else {
model = props.modelValue.filter(date => date !== val);
}
if (model.length === 0) {
model = null;
}
}
emitValue(model, 'remove', date);
}
function updateValue (mask, locale, reason) {
const model = daysModel.value
.concat(rangeModel.value)
.map(entry => encodeEntry(entry, mask, locale))
.filter(entry => {
return entry.from !== void 0
? entry.from.dateHash !== null && entry.to.dateHash !== null
: entry.dateHash !== null
});
emit('update:modelValue', (props.multiple === true ? model : model[ 0 ]) || null, reason);
}
function getHeader () {
if (props.minimal === true) { return }
return vue.h('div', {
class: 'q-date__header ' + headerClass.value
}, [
vue.h('div', {
class: 'relative-position'
}, [
vue.h(vue.Transition, {
name: 'q-transition--fade'
}, () => vue.h('div', {
key: 'h-yr-' + headerSubtitle.value,
class: 'q-date__header-subtitle q-date__header-link '
+ (view.value === 'Years' ? 'q-date__header-link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
...getCache('vY', {
onClick () { view.value = 'Years'; },
onKeyup (e) { e.keyCode === 13 && (view.value = 'Years'); }
})
}, [ headerSubtitle.value ]))
]),
vue.h('div', {
class: 'q-date__header-title relative-position flex no-wrap'
}, [
vue.h('div', {
class: 'relative-position col'
}, [
vue.h(vue.Transition, {
name: 'q-transition--fade'
}, () => vue.h('div', {
key: 'h-sub' + headerTitle.value,
class: 'q-date__header-title-label q-date__header-link '
+ (view.value === 'Calendar' ? 'q-date__header-link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
...getCache('vC', {
onClick () { view.value = 'Calendar'; },
onKeyup (e) { e.keyCode === 13 && (view.value = 'Calendar'); }
})
}, [ headerTitle.value ]))
]),
props.todayBtn === true ? vue.h(QBtn, {
class: 'q-date__header-today self-start',
icon: $q.iconSet.datetime.today,
flat: true,
size: 'sm',
round: true,
tabindex: tabindex.value,
onClick: setToday
}) : null
])
])
}
function getNavigation ({ label, type, key, dir, goTo, boundaries, cls }) {
return [
vue.h('div', {
class: 'row items-center q-date__arrow'
}, [
vue.h(QBtn, {
round: true,
dense: true,
size: 'sm',
flat: true,
icon: dateArrow.value[ 0 ],
tabindex: tabindex.value,
disable: boundaries.prev === false,
...getCache('go-#' + type, { onClick () { goTo(-1); } })
})
]),
vue.h('div', {
class: 'relative-position overflow-hidden flex flex-center' + cls
}, [
vue.h(vue.Transition, {
name: 'q-transition--jump-' + dir
}, () => vue.h('div', { key }, [
vue.h(QBtn, {
flat: true,
dense: true,
noCaps: true,
label,
tabindex: tabindex.value,
...getCache('view#' + type, { onClick: () => { view.value = type; } })
})
]))
]),
vue.h('div', {
class: 'row items-center q-date__arrow'
}, [
vue.h(QBtn, {
round: true,
dense: true,
size: 'sm',
flat: true,
icon: dateArrow.value[ 1 ],
tabindex: tabindex.value,
disable: boundaries.next === false,
...getCache('go+#' + type, { onClick () { goTo(1); } })
})
])
]
}
const renderViews = {
Calendar: () => ([
vue.h('div', {
key: 'calendar-view',
class: 'q-date__view q-date__calendar'
}, [
vue.h('div', {
class: 'q-date__navigation row items-center no-wrap'
}, getNavigation({
label: innerLocale.value.months[ viewModel.value.month - 1 ],
type: 'Months',
key: viewModel.value.month,
dir: monthDirection.value,
goTo: goToMonth,
boundaries: navBoundaries.value.month,
cls: ' col'
}).concat(getNavigation({
label: viewModel.value.year,
type: 'Years',
key: viewModel.value.year,
dir: yearDirection.value,
goTo: goToYear,
boundaries: navBoundaries.value.year,
cls: ''
}))),
vue.h('div', {
class: 'q-date__calendar-weekdays row items-center no-wrap'
}, daysOfWeek.value.map(day => vue.h('div', { class: 'q-date__calendar-item' }, [ vue.h('div', day) ]))),
vue.h('div', {
class: 'q-date__calendar-days-container relative-position overflow-hidden'
}, [
vue.h(vue.Transition, {
name: 'q-transition--slide-' + monthDirection.value
}, () => vue.h('div', {
key: viewMonthHash.value,
class: 'q-date__calendar-days fit'
}, days.value.map(day => vue.h('div', { class: day.classes }, [
day.in === true
? vue.h(
QBtn, {
class: day.today === true ? 'q-date__today' : '',
dense: true,
flat: day.flat,
unelevated: day.unelevated,
color: day.color,
textColor: day.textColor,
label: day.i,
tabindex: tabindex.value,
...getCache('day#' + day.i, {
onClick: () => { onDayClick(day.i); },
onMouseover: () => { onDayMouseover(day.i); }
})
},
day.event !== false
? () => vue.h('div', { class: 'q-date__event bg-' + day.event })
: null
)
: vue.h('div', '' + day.i)
]))))
])
])
]),
Months () {
const currentYear = viewModel.value.year === today.value.year;
const isDisabled = month => {
return (
(minNav.value !== null && viewModel.value.year === minNav.value.year && minNav.value.month > month)
|| (maxNav.value !== null && viewModel.value.year === maxNav.value.year && maxNav.value.month < month)
)
};
const content = innerLocale.value.monthsShort.map((month, i) => {
const active = viewModel.value.month === i + 1;
return vue.h('div', {
class: 'q-date__months-item flex flex-center'
}, [
vue.h(QBtn, {
class: currentYear === true && today.value.month === i + 1 ? 'q-date__today' : null,
flat: active !== true,
label: month,
unelevated: active,
color: active === true ? computedColor.value : null,
textColor: active === true ? computedTextColor.value : null,
tabindex: tabindex.value,
disable: isDisabled(i + 1),
...getCache('month#' + i, { onClick: () => { setMonth(i + 1); } })
})
])
});
props.yearsInMonthView === true && content.unshift(
vue.h('div', { class: 'row no-wrap full-width' }, [
getNavigation({
label: viewModel.value.year,
type: 'Years',
key: viewModel.value.year,
dir: yearDirection.value,
goTo: goToYear,
boundaries: navBoundaries.value.year,
cls: ' col'
})
])
);
return vue.h('div', {
key: 'months-view',
class: 'q-date__view q-date__months flex flex-center'
}, content)
},
Years () {
const
start = startYear.value,
stop = start + yearsInterval,
years = [];
const isDisabled = year => {
return (
(minNav.value !== null && minNav.value.year > year)
|| (maxNav.value !== null && maxNav.value.year < year)
)
};
for (let i = start; i <= stop; i++) {
const active = viewModel.value.year === i;
years.push(
vue.h('div', {
class: 'q-date__years-item flex flex-center'
}, [
vue.h(QBtn, {
key: 'yr' + i,
class: today.value.year === i ? 'q-date__today' : null,
flat: !active,
label: i,
dense: true,
unelevated: active,
color: active === true ? computedColor.value : null,
textColor: active === true ? computedTextColor.value : null,
tabindex: tabindex.value,
disable: isDisabled(i),
...getCache('yr#' + i, { onClick: () => { setYear(i); } })
})
])
);
}
return vue.h('div', {
class: 'q-date__view q-date__years flex flex-center'
}, [
vue.h('div', {
class: 'col-auto'
}, [
vue.h(QBtn, {
round: true,
dense: true,
flat: true,
icon: dateArrow.value[ 0 ],
tabindex: tabindex.value,
disable: isDisabled(start),
...getCache('y-', { onClick: () => { startYear.value -= yearsInterval; } })
})
]),
vue.h('div', {
class: 'q-date__years-content col self-stretch row items-center'
}, years),
vue.h('div', {
class: 'col-auto'
}, [
vue.h(QBtn, {
round: true,
dense: true,
flat: true,
icon: dateArrow.value[ 1 ],
tabindex: tabindex.value,
disable: isDisabled(stop),
...getCache('y+', { onClick: () => { startYear.value += yearsInterval; } })
})
])
])
}
};
function onDayClick (dayIndex) {
const day = { ...viewModel.value, day: dayIndex };
if (props.range === false) {
toggleDate(day, viewMonthHash.value);
return
}
if (editRange.value === null) {
const dayProps = days.value.find(day => day.fill !== true && day.i === dayIndex);
if (props.noUnset !== true && dayProps.range !== void 0) {
removeFromModel({ target: day, from: dayProps.range.from, to: dayProps.range.to });
return
}
if (dayProps.selected === true) {
removeFromModel(day);
return
}
const initHash = getDayHash(day);
editRange.value = {
init: day,
initHash,
final: day,
finalHash: initHash
};
emit('rangeStart', getShortDate(day));
}
else {
const
initHash = editRange.value.initHash,
finalHash = getDayHash(day),
payload = initHash <= finalHash
? { from: editRange.value.init, to: day }
: { from: day, to: editRange.value.init };
editRange.value = null;
addToModel(initHash === finalHash ? day : { target: day, ...payload });
emit('rangeEnd', {
from: getShortDate(payload.from),
to: getShortDate(payload.to)
});
}
}
function onDayMouseover (dayIndex) {
if (editRange.value !== null) {
const final = { ...viewModel.value, day: dayIndex };
Object.assign(editRange.value, {
final,
finalHash: getDayHash(final)
});
}
}
// expose public methods
Object.assign(proxy, {
setToday, setView, offsetCalendar, setCalendarTo, setEditingRange
});
return () => {
const content = [
vue.h('div', {
class: 'q-date__content col relative-position'
}, [
vue.h(vue.Transition, {
name: 'q-transition--fade'
}, renderViews[ view.value ])
])
];
const def = hSlot(slots.default);
def !== void 0 && content.push(
vue.h('div', { class: 'q-date__actions' }, def)
);
if (props.name !== void 0 && props.disable !== true) {
injectFormInput(content, 'push');
}
return vue.h('div', {
class: classes.value,
...attributes.value
}, [
getHeader(),
vue.h('div', {
ref: blurTargetRef,
class: 'q-date__main col column',
tabindex: -1
}, content)
])
}
}
});
function useHistory (showing, hide, hideOnRouteChange) {
let historyEntry;
function removeFromHistory () {
if (historyEntry !== void 0) {
History.remove(historyEntry);
historyEntry = void 0;
}
}
vue.onBeforeUnmount(() => {
showing.value === true && removeFromHistory();
});
return {
removeFromHistory,
addToHistory () {
historyEntry = {
condition: () => hideOnRouteChange.value === true,
handler: hide
};
History.add(historyEntry);
}
}
}
let
registered = 0,
scrollPositionX,
scrollPositionY,
maxScrollTop,
vpPendingUpdate = false,
bodyLeft,
bodyTop,
href,
closeTimer = null;
function onWheel (e) {
if (shouldPreventScroll(e)) {
stopAndPrevent(e);
}
}
function shouldPreventScroll (e) {
if (e.target === document.body || e.target.classList.contains('q-layout__backdrop')) {
return true
}
const
path = getEventPath(e),
shift = e.shiftKey && !e.deltaX,
scrollY = !shift && Math.abs(e.deltaX) <= Math.abs(e.deltaY),
delta = shift || scrollY ? e.deltaY : e.deltaX;
for (let index = 0; index < path.length; index++) {
const el = path[ index ];
if (hasScrollbar(el, scrollY)) {
return scrollY
? (
delta < 0 && el.scrollTop === 0
? true
: delta > 0 && el.scrollTop + el.clientHeight === el.scrollHeight
)
: (
delta < 0 && el.scrollLeft === 0
? true
: delta > 0 && el.scrollLeft + el.clientWidth === el.scrollWidth
)
}
}
return true
}
function onAppleScroll (e) {
if (e.target === document) {
// required, otherwise iOS blocks further scrolling
// until the mobile scrollbar dissappears
document.scrollingElement.scrollTop = document.scrollingElement.scrollTop; // eslint-disable-line
}
}
function onAppleResize (evt) {
if (vpPendingUpdate === true) {
return
}
vpPendingUpdate = true;
requestAnimationFrame(() => {
vpPendingUpdate = false;
const
{ height } = evt.target,
{ clientHeight, scrollTop } = document.scrollingElement;
if (maxScrollTop === void 0 || height !== window.innerHeight) {
maxScrollTop = clientHeight - height;
document.scrollingElement.scrollTop = scrollTop;
}
if (scrollTop > maxScrollTop) {
document.scrollingElement.scrollTop -= Math.ceil((scrollTop - maxScrollTop) / 8);
}
});
}
function apply$1 (action) {
const
body = document.body,
hasViewport = window.visualViewport !== void 0;
if (action === 'add') {
const { overflowY, overflowX } = window.getComputedStyle(body);
scrollPositionX = getHorizontalScrollPosition(window);
scrollPositionY = getVerticalScrollPosition(window);
bodyLeft = body.style.left;
bodyTop = body.style.top;
href = window.location.href;
body.style.left = `-${ scrollPositionX }px`;
body.style.top = `-${ scrollPositionY }px`;
if (overflowX !== 'hidden' && (overflowX === 'scroll' || body.scrollWidth > window.innerWidth)) {
body.classList.add('q-body--force-scrollbar-x');
}
if (overflowY !== 'hidden' && (overflowY === 'scroll' || body.scrollHeight > window.innerHeight)) {
body.classList.add('q-body--force-scrollbar-y');
}
body.classList.add('q-body--prevent-scroll');
document.qScrollPrevented = true;
if (client.is.ios === true) {
if (hasViewport === true) {
window.scrollTo(0, 0);
window.visualViewport.addEventListener('resize', onAppleResize, listenOpts.passiveCapture);
window.visualViewport.addEventListener('scroll', onAppleResize, listenOpts.passiveCapture);
window.scrollTo(0, 0);
}
else {
window.addEventListener('scroll', onAppleScroll, listenOpts.passiveCapture);
}
}
}
if (client.is.desktop === true && client.is.mac === true) {
// ref. https://developers.google.com/web/updates/2017/01/scrolling-intervention
window[ `${ action }EventListener` ]('wheel', onWheel, listenOpts.notPassive);
}
if (action === 'remove') {
if (client.is.ios === true) {
if (hasViewport === true) {
window.visualViewport.removeEventListener('resize', onAppleResize, listenOpts.passiveCapture);
window.visualViewport.removeEventListener('scroll', onAppleResize, listenOpts.passiveCapture);
}
else {
window.removeEventListener('scroll', onAppleScroll, listenOpts.passiveCapture);
}
}
body.classList.remove('q-body--prevent-scroll');
body.classList.remove('q-body--force-scrollbar-x');
body.classList.remove('q-body--force-scrollbar-y');
document.qScrollPrevented = false;
body.style.left = bodyLeft;
body.style.top = bodyTop;
// scroll back only if route has not changed
if (window.location.href === href) {
window.scrollTo(scrollPositionX, scrollPositionY);
}
maxScrollTop = void 0;
}
}
function preventScroll (state) {
let action = 'add';
if (state === true) {
registered++;
if (closeTimer !== null) {
clearTimeout(closeTimer);
closeTimer = null;
return
}
if (registered > 1) {
return
}
}
else {
if (registered === 0) {
return
}
registered--;
if (registered > 0) {
return
}
action = 'remove';
if (client.is.ios === true && client.is.nativeMobile === true) {
closeTimer !== null && clearTimeout(closeTimer);
closeTimer = setTimeout(() => {
apply$1(action);
closeTimer = null;
}, 100);
return
}
}
apply$1(action);
}
function usePreventScroll () {
let currentState;
return {
preventBodyScroll (state) {
if (
state !== currentState
&& (currentState !== void 0 || state === true)
) {
currentState = state;
preventScroll(state);
}
}
}
}
let maximizedModals = 0;
const positionClass$1 = {
standard: 'fixed-full flex-center',
top: 'fixed-top justify-center',
bottom: 'fixed-bottom justify-center',
right: 'fixed-right items-center',
left: 'fixed-left items-center'
};
const defaultTransitions = {
standard: [ 'scale', 'scale' ],
top: [ 'slide-down', 'slide-up' ],
bottom: [ 'slide-up', 'slide-down' ],
right: [ 'slide-left', 'slide-right' ],
left: [ 'slide-right', 'slide-left' ]
};
var QDialog = createComponent({
name: 'QDialog',
inheritAttrs: false,
props: {
...useModelToggleProps,
...useTransitionProps,
transitionShow: String, // override useTransitionProps
transitionHide: String, // override useTransitionProps
persistent: Boolean,
autoClose: Boolean,
allowFocusOutside: Boolean,
noEscDismiss: Boolean,
noBackdropDismiss: Boolean,
noRouteDismiss: Boolean,
noRefocus: Boolean,
noFocus: Boolean,
noShake: Boolean,
seamless: Boolean,
maximized: Boolean,
fullWidth: Boolean,
fullHeight: Boolean,
square: Boolean,
position: {
type: String,
default: 'standard',
validator: val => val === 'standard'
|| [ 'top', 'bottom', 'left', 'right' ].includes(val)
}
},
emits: [
...useModelToggleEmits,
'shake', 'click', 'escapeKey'
],
setup (props, { slots, emit, attrs }) {
const vm = vue.getCurrentInstance();
const innerRef = vue.ref(null);
const showing = vue.ref(false);
const animating = vue.ref(false);
let shakeTimeout = null, refocusTarget = null, isMaximized, avoidAutoClose;
const hideOnRouteChange = vue.computed(() =>
props.persistent !== true
&& props.noRouteDismiss !== true
&& props.seamless !== true
);
const { preventBodyScroll } = usePreventScroll();
const { registerTimeout } = useTimeout();
const { registerTick, removeTick } = useTick();
const { transitionProps, transitionStyle } = useTransition(
props,
() => defaultTransitions[ props.position ][ 0 ],
() => defaultTransitions[ props.position ][ 1 ]
);
const { showPortal, hidePortal, portalIsAccessible, renderPortal } = usePortal(
vm, innerRef, renderPortalContent, 'dialog'
);
const { hide } = useModelToggle({
showing,
hideOnRouteChange,
handleShow,
handleHide,
processOnMount: true
});
const { addToHistory, removeFromHistory } = useHistory(showing, hide, hideOnRouteChange);
const classes = vue.computed(() =>
'q-dialog__inner flex no-pointer-events'
+ ` q-dialog__inner--${ props.maximized === true ? 'maximized' : 'minimized' }`
+ ` q-dialog__inner--${ props.position } ${ positionClass$1[ props.position ] }`
+ (animating.value === true ? ' q-dialog__inner--animating' : '')
+ (props.fullWidth === true ? ' q-dialog__inner--fullwidth' : '')
+ (props.fullHeight === true ? ' q-dialog__inner--fullheight' : '')
+ (props.square === true ? ' q-dialog__inner--square' : '')
);
const useBackdrop = vue.computed(() => showing.value === true && props.seamless !== true);
const onEvents = vue.computed(() => (
props.autoClose === true
? { onClick: onAutoClose }
: {}
));
const rootClasses = vue.computed(() => [
'q-dialog fullscreen no-pointer-events '
+ `q-dialog--${ useBackdrop.value === true ? 'modal' : 'seamless' }`,
attrs.class
]);
vue.watch(() => props.maximized, state => {
showing.value === true && updateMaximized(state);
});
vue.watch(useBackdrop, val => {
preventBodyScroll(val);
if (val === true) {
addFocusout(onFocusChange);
addEscapeKey(onEscapeKey);
}
else {
removeFocusout(onFocusChange);
removeEscapeKey(onEscapeKey);
}
});
function handleShow (evt) {
addToHistory();
refocusTarget = props.noRefocus === false && document.activeElement !== null
? document.activeElement
: null;
updateMaximized(props.maximized);
showPortal();
animating.value = true;
if (props.noFocus !== true) {
document.activeElement !== null && document.activeElement.blur();
registerTick(focus);
}
else {
removeTick();
}
// should removeTimeout() if this gets removed
registerTimeout(() => {
if (vm.proxy.$q.platform.is.ios === true) {
if (props.seamless !== true && document.activeElement) {
const
{ top, bottom } = document.activeElement.getBoundingClientRect(),
{ innerHeight } = window,
height = window.visualViewport !== void 0
? window.visualViewport.height
: innerHeight;
if (top > 0 && bottom > height / 2) {
document.scrollingElement.scrollTop = Math.min(
document.scrollingElement.scrollHeight - height,
bottom >= innerHeight
? Infinity
: Math.ceil(document.scrollingElement.scrollTop + bottom - height / 2)
);
}
document.activeElement.scrollIntoView();
}
// required in order to avoid the "double-tap needed" issue
avoidAutoClose = true;
innerRef.value.click();
avoidAutoClose = false;
}
showPortal(true); // done showing portal
animating.value = false;
emit('show', evt);
}, props.transitionDuration);
}
function handleHide (evt) {
removeTick();
removeFromHistory();
cleanup(true);
animating.value = true;
hidePortal();
if (refocusTarget !== null) {
((evt && evt.type.indexOf('key') === 0
? refocusTarget.closest('[tabindex]:not([tabindex^="-"])')
: void 0
) || refocusTarget).focus();
refocusTarget = null;
}
// should removeTimeout() if this gets removed
registerTimeout(() => {
hidePortal(true); // done hiding, now destroy
animating.value = false;
emit('hide', evt);
}, props.transitionDuration);
}
function focus (selector) {
addFocusFn(() => {
let node = innerRef.value;
if (node === null || node.contains(document.activeElement) === true) {
return
}
node = (selector !== '' ? node.querySelector(selector) : null)
|| node.querySelector('[autofocus][tabindex], [data-autofocus][tabindex]')
|| node.querySelector('[autofocus] [tabindex], [data-autofocus] [tabindex]')
|| node.querySelector('[autofocus], [data-autofocus]')
|| node;
node.focus({ preventScroll: true });
});
}
function shake (focusTarget) {
if (focusTarget && typeof focusTarget.focus === 'function') {
focusTarget.focus({ preventScroll: true });
}
else {
focus();
}
emit('shake');
const node = innerRef.value;
if (node !== null) {
node.classList.remove('q-animate--scale');
node.classList.add('q-animate--scale');
shakeTimeout !== null && clearTimeout(shakeTimeout);
shakeTimeout = setTimeout(() => {
shakeTimeout = null;
if (innerRef.value !== null) {
node.classList.remove('q-animate--scale');
// some platforms (like desktop Chrome)
// require calling focus() again
focus();
}
}, 170);
}
}
function onEscapeKey () {
if (props.seamless !== true) {
if (props.persistent === true || props.noEscDismiss === true) {
props.maximized !== true && props.noShake !== true && shake();
}
else {
emit('escapeKey');
hide();
}
}
}
function cleanup (hiding) {
if (shakeTimeout !== null) {
clearTimeout(shakeTimeout);
shakeTimeout = null;
}
if (hiding === true || showing.value === true) {
updateMaximized(false);
if (props.seamless !== true) {
preventBodyScroll(false);
removeFocusout(onFocusChange);
removeEscapeKey(onEscapeKey);
}
}
if (hiding !== true) {
refocusTarget = null;
}
}
function updateMaximized (active) {
if (active === true) {
if (isMaximized !== true) {
maximizedModals < 1 && document.body.classList.add('q-body--dialog');
maximizedModals++;
isMaximized = true;
}
}
else if (isMaximized === true) {
if (maximizedModals < 2) {
document.body.classList.remove('q-body--dialog');
}
maximizedModals--;
isMaximized = false;
}
}
function onAutoClose (e) {
if (avoidAutoClose !== true) {
hide(e);
emit('click', e);
}
}
function onBackdropClick (e) {
if (props.persistent !== true && props.noBackdropDismiss !== true) {
hide(e);
}
else if (props.noShake !== true) {
shake();
}
}
function onFocusChange (evt) {
// the focus is not in a vue child component
if (
props.allowFocusOutside !== true
&& portalIsAccessible.value === true
&& childHasFocus(innerRef.value, evt.target) !== true
) {
focus('[tabindex]:not([tabindex="-1"])');
}
}
Object.assign(vm.proxy, {
// expose public methods
focus, shake,
// private but needed by QSelect
__updateRefocusTarget (target) {
refocusTarget = target || null;
}
});
vue.onBeforeUnmount(cleanup);
function renderPortalContent () {
return vue.h('div', {
role: 'dialog',
'aria-modal': useBackdrop.value === true ? 'true' : 'false',
...attrs,
class: rootClasses.value
}, [
vue.h(vue.Transition, {
name: 'q-transition--fade',
appear: true
}, () => (
useBackdrop.value === true
? vue.h('div', {
class: 'q-dialog__backdrop fixed-full',
style: transitionStyle.value,
'aria-hidden': 'true',
tabindex: -1,
onClick: onBackdropClick
})
: null
)),
vue.h(
vue.Transition,
transitionProps.value,
() => (
showing.value === true
? vue.h('div', {
ref: innerRef,
class: classes.value,
style: transitionStyle.value,
tabindex: -1,
...onEvents.value
}, hSlot(slots.default))
: null
)
)
])
}
return renderPortal
}
});
const duration = 150;
var QDrawer = createComponent({
name: 'QDrawer',
inheritAttrs: false,
props: {
...useModelToggleProps,
...useDarkProps,
side: {
type: String,
default: 'left',
validator: v => [ 'left', 'right' ].includes(v)
},
width: {
type: Number,
default: 300
},
mini: Boolean,
miniToOverlay: Boolean,
miniWidth: {
type: Number,
default: 57
},
noMiniAnimation: Boolean,
breakpoint: {
type: Number,
default: 1023
},
showIfAbove: Boolean,
behavior: {
type: String,
validator: v => [ 'default', 'desktop', 'mobile' ].includes(v),
default: 'default'
},
bordered: Boolean,
elevated: Boolean,
overlay: Boolean,
persistent: Boolean,
noSwipeOpen: Boolean,
noSwipeClose: Boolean,
noSwipeBackdrop: Boolean
},
emits: [
...useModelToggleEmits,
'onLayout', 'miniState'
],
setup (props, { slots, emit, attrs }) {
const vm = vue.getCurrentInstance();
const { proxy: { $q } } = vm;
const isDark = useDark(props, $q);
const { preventBodyScroll } = usePreventScroll();
const { registerTimeout, removeTimeout } = useTimeout();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QDrawer needs to be child of QLayout');
return emptyRenderFn
}
let lastDesktopState, timerMini = null, layoutTotalWidthWatcher;
const belowBreakpoint = vue.ref(
props.behavior === 'mobile'
|| (props.behavior !== 'desktop' && $layout.totalWidth.value <= props.breakpoint)
);
const isMini = vue.computed(() =>
props.mini === true && belowBreakpoint.value !== true
);
const size = vue.computed(() => (
isMini.value === true
? props.miniWidth
: props.width
));
const showing = vue.ref(
props.showIfAbove === true && belowBreakpoint.value === false
? true
: props.modelValue === true
);
const hideOnRouteChange = vue.computed(() =>
props.persistent !== true
&& (belowBreakpoint.value === true || onScreenOverlay.value === true)
);
function handleShow (evt, noEvent) {
addToHistory();
evt !== false && $layout.animate();
applyPosition(0);
if (belowBreakpoint.value === true) {
const otherInstance = $layout.instances[ otherSide.value ];
if (otherInstance !== void 0 && otherInstance.belowBreakpoint === true) {
otherInstance.hide(false);
}
applyBackdrop(1);
$layout.isContainer.value !== true && preventBodyScroll(true);
}
else {
applyBackdrop(0);
evt !== false && setScrollable(false);
}
registerTimeout(() => {
evt !== false && setScrollable(true);
noEvent !== true && emit('show', evt);
}, duration);
}
function handleHide (evt, noEvent) {
removeFromHistory();
evt !== false && $layout.animate();
applyBackdrop(0);
applyPosition(stateDirection.value * size.value);
cleanup();
if (noEvent !== true) {
registerTimeout(() => { emit('hide', evt); }, duration);
}
else {
removeTimeout();
}
}
const { show, hide } = useModelToggle({
showing,
hideOnRouteChange,
handleShow,
handleHide
});
const { addToHistory, removeFromHistory } = useHistory(showing, hide, hideOnRouteChange);
const instance = {
belowBreakpoint,
hide
};
const rightSide = vue.computed(() => props.side === 'right');
const stateDirection = vue.computed(() =>
($q.lang.rtl === true ? -1 : 1) * (rightSide.value === true ? 1 : -1)
);
const flagBackdropBg = vue.ref(0);
const flagPanning = vue.ref(false);
const flagMiniAnimate = vue.ref(false);
const flagContentPosition = vue.ref( // starting with "hidden" for SSR
size.value * stateDirection.value
);
const otherSide = vue.computed(() => (rightSide.value === true ? 'left' : 'right'));
const offset = vue.computed(() => (
showing.value === true && belowBreakpoint.value === false && props.overlay === false
? (props.miniToOverlay === true ? props.miniWidth : size.value)
: 0
));
const fixed = vue.computed(() =>
props.overlay === true
|| props.miniToOverlay === true
|| $layout.view.value.indexOf(rightSide.value ? 'R' : 'L') > -1
|| ($q.platform.is.ios === true && $layout.isContainer.value === true)
);
const onLayout = vue.computed(() =>
props.overlay === false
&& showing.value === true
&& belowBreakpoint.value === false
);
const onScreenOverlay = vue.computed(() =>
props.overlay === true
&& showing.value === true
&& belowBreakpoint.value === false
);
const backdropClass = vue.computed(() =>
'fullscreen q-drawer__backdrop'
+ (showing.value === false && flagPanning.value === false ? ' hidden' : '')
);
const backdropStyle = vue.computed(() => ({
backgroundColor: `rgba(0,0,0,${ flagBackdropBg.value * 0.4 })`
}));
const headerSlot = vue.computed(() => (
rightSide.value === true
? $layout.rows.value.top[ 2 ] === 'r'
: $layout.rows.value.top[ 0 ] === 'l'
));
const footerSlot = vue.computed(() => (
rightSide.value === true
? $layout.rows.value.bottom[ 2 ] === 'r'
: $layout.rows.value.bottom[ 0 ] === 'l'
));
const aboveStyle = vue.computed(() => {
const css = {};
if ($layout.header.space === true && headerSlot.value === false) {
if (fixed.value === true) {
css.top = `${ $layout.header.offset }px`;
}
else if ($layout.header.space === true) {
css.top = `${ $layout.header.size }px`;
}
}
if ($layout.footer.space === true && footerSlot.value === false) {
if (fixed.value === true) {
css.bottom = `${ $layout.footer.offset }px`;
}
else if ($layout.footer.space === true) {
css.bottom = `${ $layout.footer.size }px`;
}
}
return css
});
const style = vue.computed(() => {
const style = {
width: `${ size.value }px`,
transform: `translateX(${ flagContentPosition.value }px)`
};
return belowBreakpoint.value === true
? style
: Object.assign(style, aboveStyle.value)
});
const contentClass = vue.computed(() =>
'q-drawer__content fit '
+ ($layout.isContainer.value !== true ? 'scroll' : 'overflow-auto')
);
const classes = vue.computed(() =>
`q-drawer q-drawer--${ props.side }`
+ (flagMiniAnimate.value === true ? ' q-drawer--mini-animate' : '')
+ (props.bordered === true ? ' q-drawer--bordered' : '')
+ (isDark.value === true ? ' q-drawer--dark q-dark' : '')
+ (
flagPanning.value === true
? ' no-transition'
: (showing.value === true ? '' : ' q-layout--prevent-focus')
)
+ (
belowBreakpoint.value === true
? ' fixed q-drawer--on-top q-drawer--mobile q-drawer--top-padding'
: ` q-drawer--${ isMini.value === true ? 'mini' : 'standard' }`
+ (fixed.value === true || onLayout.value !== true ? ' fixed' : '')
+ (props.overlay === true || props.miniToOverlay === true ? ' q-drawer--on-top' : '')
+ (headerSlot.value === true ? ' q-drawer--top-padding' : '')
)
);
const openDirective = vue.computed(() => {
// if props.noSwipeOpen !== true
const dir = $q.lang.rtl === true ? props.side : otherSide.value;
return [ [
TouchPan,
onOpenPan,
void 0,
{
[ dir ]: true,
mouse: true
}
] ]
});
const contentCloseDirective = vue.computed(() => {
// if belowBreakpoint.value === true && props.noSwipeClose !== true
const dir = $q.lang.rtl === true ? otherSide.value : props.side;
return [ [
TouchPan,
onClosePan,
void 0,
{
[ dir ]: true,
mouse: true
}
] ]
});
const backdropCloseDirective = vue.computed(() => {
// if showing.value === true && props.noSwipeBackdrop !== true
const dir = $q.lang.rtl === true ? otherSide.value : props.side;
return [ [
TouchPan,
onClosePan,
void 0,
{
[ dir ]: true,
mouse: true,
mouseAllDir: true
}
] ]
});
function updateBelowBreakpoint () {
updateLocal(belowBreakpoint, (
props.behavior === 'mobile'
|| (props.behavior !== 'desktop' && $layout.totalWidth.value <= props.breakpoint)
));
}
vue.watch(belowBreakpoint, val => {
if (val === true) { // from lg to xs
lastDesktopState = showing.value;
showing.value === true && hide(false);
}
else if (
props.overlay === false
&& props.behavior !== 'mobile'
&& lastDesktopState !== false
) { // from xs to lg
if (showing.value === true) {
applyPosition(0);
applyBackdrop(0);
cleanup();
}
else {
show(false);
}
}
});
vue.watch(() => props.side, (newSide, oldSide) => {
if ($layout.instances[ oldSide ] === instance) {
$layout.instances[ oldSide ] = void 0;
$layout[ oldSide ].space = false;
$layout[ oldSide ].offset = 0;
}
$layout.instances[ newSide ] = instance;
$layout[ newSide ].size = size.value;
$layout[ newSide ].space = onLayout.value;
$layout[ newSide ].offset = offset.value;
});
vue.watch($layout.totalWidth, () => {
if ($layout.isContainer.value === true || document.qScrollPrevented !== true) {
updateBelowBreakpoint();
}
});
vue.watch(
() => props.behavior + props.breakpoint,
updateBelowBreakpoint
);
vue.watch($layout.isContainer, val => {
showing.value === true && preventBodyScroll(val !== true);
val === true && updateBelowBreakpoint();
});
vue.watch($layout.scrollbarWidth, () => {
applyPosition(showing.value === true ? 0 : void 0);
});
vue.watch(offset, val => { updateLayout('offset', val); });
vue.watch(onLayout, val => {
emit('onLayout', val);
updateLayout('space', val);
});
vue.watch(rightSide, () => { applyPosition(); });
vue.watch(size, val => {
applyPosition();
updateSizeOnLayout(props.miniToOverlay, val);
});
vue.watch(() => props.miniToOverlay, val => {
updateSizeOnLayout(val, size.value);
});
vue.watch(() => $q.lang.rtl, () => { applyPosition(); });
vue.watch(() => props.mini, () => {
if (props.noMiniAnimation) return
if (props.modelValue === true) {
animateMini();
$layout.animate();
}
});
vue.watch(isMini, val => { emit('miniState', val); });
function applyPosition (position) {
if (position === void 0) {
vue.nextTick(() => {
position = showing.value === true ? 0 : size.value;
applyPosition(stateDirection.value * position);
});
}
else {
if (
$layout.isContainer.value === true
&& rightSide.value === true
&& (belowBreakpoint.value === true || Math.abs(position) === size.value)
) {
position += stateDirection.value * $layout.scrollbarWidth.value;
}
flagContentPosition.value = position;
}
}
function applyBackdrop (x) {
flagBackdropBg.value = x;
}
function setScrollable (v) {
const action = v === true
? 'remove'
: ($layout.isContainer.value !== true ? 'add' : '');
action !== '' && document.body.classList[ action ]('q-body--drawer-toggle');
}
function animateMini () {
timerMini !== null && clearTimeout(timerMini);
if (vm.proxy && vm.proxy.$el) {
// need to speed it up and apply it immediately,
// even faster than Vue's nextTick!
vm.proxy.$el.classList.add('q-drawer--mini-animate');
}
flagMiniAnimate.value = true;
timerMini = setTimeout(() => {
timerMini = null;
flagMiniAnimate.value = false;
if (vm && vm.proxy && vm.proxy.$el) {
vm.proxy.$el.classList.remove('q-drawer--mini-animate');
}
}, 150);
}
function onOpenPan (evt) {
if (showing.value !== false) {
// some browsers might capture and trigger this
// even if Drawer has just been opened (but animation is still pending)
return
}
const
width = size.value,
position = between(evt.distance.x, 0, width);
if (evt.isFinal === true) {
const opened = position >= Math.min(75, width);
if (opened === true) {
show();
}
else {
$layout.animate();
applyBackdrop(0);
applyPosition(stateDirection.value * width);
}
flagPanning.value = false;
return
}
applyPosition(
($q.lang.rtl === true ? rightSide.value !== true : rightSide.value)
? Math.max(width - position, 0)
: Math.min(0, position - width)
);
applyBackdrop(
between(position / width, 0, 1)
);
if (evt.isFirst === true) {
flagPanning.value = true;
}
}
function onClosePan (evt) {
if (showing.value !== true) {
// some browsers might capture and trigger this
// even if Drawer has just been closed (but animation is still pending)
return
}
const
width = size.value,
dir = evt.direction === props.side,
position = ($q.lang.rtl === true ? dir !== true : dir)
? between(evt.distance.x, 0, width)
: 0;
if (evt.isFinal === true) {
const opened = Math.abs(position) < Math.min(75, width);
if (opened === true) {
$layout.animate();
applyBackdrop(1);
applyPosition(0);
}
else {
hide();
}
flagPanning.value = false;
return
}
applyPosition(stateDirection.value * position);
applyBackdrop(between(1 - position / width, 0, 1));
if (evt.isFirst === true) {
flagPanning.value = true;
}
}
function cleanup () {
preventBodyScroll(false);
setScrollable(true);
}
function updateLayout (prop, val) {
$layout.update(props.side, prop, val);
}
function updateLocal (prop, val) {
if (prop.value !== val) {
prop.value = val;
}
}
function updateSizeOnLayout (miniToOverlay, size) {
updateLayout('size', miniToOverlay === true ? props.miniWidth : size);
}
$layout.instances[ props.side ] = instance;
updateSizeOnLayout(props.miniToOverlay, size.value);
updateLayout('space', onLayout.value);
updateLayout('offset', offset.value);
if (
props.showIfAbove === true
&& props.modelValue !== true
&& showing.value === true
&& props[ 'onUpdate:modelValue' ] !== void 0
) {
emit('update:modelValue', true);
}
vue.onMounted(() => {
emit('onLayout', onLayout.value);
emit('miniState', isMini.value);
lastDesktopState = props.showIfAbove === true;
const fn = () => {
const action = showing.value === true ? handleShow : handleHide;
action(false, true);
};
if ($layout.totalWidth.value !== 0) {
// make sure that all computed properties
// have been updated before calling handleShow/handleHide()
vue.nextTick(fn);
return
}
layoutTotalWidthWatcher = vue.watch($layout.totalWidth, () => {
layoutTotalWidthWatcher();
layoutTotalWidthWatcher = void 0;
if (showing.value === false && props.showIfAbove === true && belowBreakpoint.value === false) {
show(false);
}
else {
fn();
}
});
});
vue.onBeforeUnmount(() => {
layoutTotalWidthWatcher !== void 0 && layoutTotalWidthWatcher();
if (timerMini !== null) {
clearTimeout(timerMini);
timerMini = null;
}
showing.value === true && cleanup();
if ($layout.instances[ props.side ] === instance) {
$layout.instances[ props.side ] = void 0;
updateLayout('size', 0);
updateLayout('offset', 0);
updateLayout('space', false);
}
});
return () => {
const child = [];
if (belowBreakpoint.value === true) {
props.noSwipeOpen === false && child.push(
vue.withDirectives(
vue.h('div', {
key: 'open',
class: `q-drawer__opener fixed-${ props.side }`,
'aria-hidden': 'true'
}),
openDirective.value
)
);
child.push(
hDir(
'div',
{
ref: 'backdrop',
class: backdropClass.value,
style: backdropStyle.value,
'aria-hidden': 'true',
onClick: hide
},
void 0,
'backdrop',
props.noSwipeBackdrop !== true && showing.value === true,
() => backdropCloseDirective.value
)
);
}
const mini = isMini.value === true && slots.mini !== void 0;
const content = [
vue.h('div', {
...attrs,
key: '' + mini, // required otherwise Vue will not diff correctly
class: [
contentClass.value,
attrs.class
]
}, mini === true
? slots.mini()
: hSlot(slots.default)
)
];
if (props.elevated === true && showing.value === true) {
content.push(
vue.h('div', {
class: 'q-layout__shadow absolute-full overflow-hidden no-pointer-events'
})
);
}
child.push(
hDir(
'aside',
{ ref: 'content', class: classes.value, style: style.value },
content,
'contentclose',
props.noSwipeClose !== true && belowBreakpoint.value === true,
() => contentCloseDirective.value
)
);
return vue.h('div', { class: 'q-drawer-container' }, child)
}
}
});
function getBlockElement (el, parent) {
if (parent && el === parent) {
return null
}
const nodeName = el.nodeName.toLowerCase();
if ([ 'div', 'li', 'ul', 'ol', 'blockquote' ].includes(nodeName) === true) {
return el
}
const
style = window.getComputedStyle
? window.getComputedStyle(el)
: el.currentStyle,
display = style.display;
if (display === 'block' || display === 'table') {
return el
}
return getBlockElement(el.parentNode)
}
function isChildOf (el, parent, orSame) {
return !el || el === document.body
? false
: (orSame === true && el === parent) || (parent === document ? document.body : parent).contains(el.parentNode)
}
function createRange (node, chars, range) {
if (!range) {
range = document.createRange();
range.selectNode(node);
range.setStart(node, 0);
}
if (chars.count === 0) {
range.setEnd(node, chars.count);
}
else if (chars.count > 0) {
if (node.nodeType === Node.TEXT_NODE) {
if (node.textContent.length < chars.count) {
chars.count -= node.textContent.length;
}
else {
range.setEnd(node, chars.count);
chars.count = 0;
}
}
else {
for (let lp = 0; chars.count !== 0 && lp < node.childNodes.length; lp++) {
range = createRange(node.childNodes[ lp ], chars, range);
}
}
}
return range
}
const urlRegex = /^https?:\/\//;
class Caret {
constructor (el, eVm) {
this.el = el;
this.eVm = eVm;
this._range = null;
}
get selection () {
if (this.el) {
const sel = document.getSelection();
// only when the selection in element
if (isChildOf(sel.anchorNode, this.el, true) && isChildOf(sel.focusNode, this.el, true)) {
return sel
}
}
return null
}
get hasSelection () {
return this.selection !== null
? this.selection.toString().length !== 0
: false
}
get range () {
const sel = this.selection;
if (sel !== null && sel.rangeCount) {
return sel.getRangeAt(0)
}
return this._range
}
get parent () {
const range = this.range;
if (range !== null) {
const node = range.startContainer;
return node.nodeType === document.ELEMENT_NODE
? node
: node.parentNode
}
return null
}
get blockParent () {
const parent = this.parent;
if (parent !== null) {
return getBlockElement(parent, this.el)
}
return null
}
save (range = this.range) {
if (range !== null) {
this._range = range;
}
}
restore (range = this._range) {
const
r = document.createRange(),
sel = document.getSelection();
if (range !== null) {
r.setStart(range.startContainer, range.startOffset);
r.setEnd(range.endContainer, range.endOffset);
sel.removeAllRanges();
sel.addRange(r);
}
else {
sel.selectAllChildren(this.el);
sel.collapseToEnd();
}
}
savePosition () {
let charCount = -1, node;
const
selection = document.getSelection(),
parentEl = this.el.parentNode;
if (selection.focusNode && isChildOf(selection.focusNode, parentEl)) {
node = selection.focusNode;
charCount = selection.focusOffset;
while (node && node !== parentEl) {
if (node !== this.el && node.previousSibling) {
node = node.previousSibling;
charCount += node.textContent.length;
}
else {
node = node.parentNode;
}
}
}
this.savedPos = charCount;
}
restorePosition (length = 0) {
if (this.savedPos > 0 && this.savedPos < length) {
const
selection = window.getSelection(),
range = createRange(this.el, { count: this.savedPos });
if (range) {
range.collapse(false);
selection.removeAllRanges();
selection.addRange(range);
}
}
}
hasParent (name, spanLevel) {
const el = spanLevel
? this.parent
: this.blockParent;
return el !== null
? el.nodeName.toLowerCase() === name.toLowerCase()
: false
}
hasParents (list, recursive, el = this.parent) {
if (el === null) {
return false
}
if (list.includes(el.nodeName.toLowerCase()) === true) {
return true
}
return recursive === true
? this.hasParents(list, recursive, el.parentNode)
: false
}
is (cmd, param) {
if (this.selection === null) {
return false
}
switch (cmd) {
case 'formatBlock':
return (param === 'DIV' && this.parent === this.el)
|| this.hasParent(param, param === 'PRE')
case 'link':
return this.hasParent('A', true)
case 'fontSize':
return document.queryCommandValue(cmd) === param
case 'fontName':
const res = document.queryCommandValue(cmd);
return res === `"${ param }"` || res === param
case 'fullscreen':
return this.eVm.inFullscreen.value
case 'viewsource':
return this.eVm.isViewingSource.value
case void 0:
return false
default:
const state = document.queryCommandState(cmd);
return param !== void 0 ? state === param : state
}
}
getParentAttribute (attrib) {
if (this.parent !== null) {
return this.parent.getAttribute(attrib)
}
return null
}
can (name) {
if (name === 'outdent') {
return this.hasParents([ 'blockquote', 'li' ], true)
}
if (name === 'indent') {
return this.hasParents([ 'li' ], true)
}
if (name === 'link') {
return this.selection !== null || this.is('link')
}
}
apply (cmd, param, done = noop) {
if (cmd === 'formatBlock') {
if ([ 'BLOCKQUOTE', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6' ].includes(param) && this.is(cmd, param)) {
cmd = 'outdent';
param = null;
}
if (param === 'PRE' && this.is(cmd, 'PRE')) {
param = 'P';
}
}
else if (cmd === 'print') {
done();
const win = window.open();
win.document.write(`
<!doctype html>
<html>
<head>
<title>Print - ${ document.title }</title>
</head>
<body>
<div>${ this.el.innerHTML }</div>
</body>
</html>
`);
win.print();
win.close();
return
}
else if (cmd === 'link') {
const link = this.getParentAttribute('href');
if (link === null) {
const selection = this.selectWord(this.selection);
const url = selection ? selection.toString() : '';
if (!url.length) {
if (!this.range || !this.range.cloneContents().querySelector('img')) {
return
}
}
this.eVm.editLinkUrl.value = urlRegex.test(url) ? url : 'https://';
document.execCommand('createLink', false, this.eVm.editLinkUrl.value);
this.save(selection.getRangeAt(0));
}
else {
this.eVm.editLinkUrl.value = link;
this.range.selectNodeContents(this.parent);
this.save();
}
return
}
else if (cmd === 'fullscreen') {
this.eVm.toggleFullscreen();
done();
return
}
else if (cmd === 'viewsource') {
this.eVm.isViewingSource.value = this.eVm.isViewingSource.value === false;
this.eVm.setContent(this.eVm.props.modelValue);
done();
return
}
document.execCommand(cmd, false, param);
done();
}
selectWord (sel) {
if (sel === null || sel.isCollapsed !== true || /* IE 11 */ sel.modify === void 0) {
return sel
}
// Detect if selection is backwards
const range = document.createRange();
range.setStart(sel.anchorNode, sel.anchorOffset);
range.setEnd(sel.focusNode, sel.focusOffset);
const direction = range.collapsed ? [ 'backward', 'forward' ] : [ 'forward', 'backward' ];
range.detach();
// modify() works on the focus of the selection
const
endNode = sel.focusNode,
endOffset = sel.focusOffset;
sel.collapse(sel.anchorNode, sel.anchorOffset);
sel.modify('move', direction[ 0 ], 'character');
sel.modify('move', direction[ 1 ], 'word');
sel.extend(endNode, endOffset);
sel.modify('extend', direction[ 1 ], 'character');
sel.modify('extend', direction[ 0 ], 'word');
return sel
}
}
var QTooltip = createComponent({
name: 'QTooltip',
inheritAttrs: false,
props: {
...useAnchorProps,
...useModelToggleProps,
...useTransitionProps,
maxHeight: {
type: String,
default: null
},
maxWidth: {
type: String,
default: null
},
transitionShow: {
default: 'jump-down'
},
transitionHide: {
default: 'jump-up'
},
anchor: {
type: String,
default: 'bottom middle',
validator: validatePosition
},
self: {
type: String,
default: 'top middle',
validator: validatePosition
},
offset: {
type: Array,
default: () => [ 14, 14 ],
validator: validateOffset
},
scrollTarget: {
default: void 0
},
delay: {
type: Number,
default: 0
},
hideDelay: {
type: Number,
default: 0
}
},
emits: [
...useModelToggleEmits
],
setup (props, { slots, emit, attrs }) {
let unwatchPosition, observer;
const vm = vue.getCurrentInstance();
const { proxy: { $q } } = vm;
const innerRef = vue.ref(null);
const showing = vue.ref(false);
const anchorOrigin = vue.computed(() => parsePosition(props.anchor, $q.lang.rtl));
const selfOrigin = vue.computed(() => parsePosition(props.self, $q.lang.rtl));
const hideOnRouteChange = vue.computed(() => props.persistent !== true);
const { registerTick, removeTick } = useTick();
const { registerTimeout } = useTimeout();
const { transitionProps, transitionStyle } = useTransition(props);
const { localScrollTarget, changeScrollEvent, unconfigureScrollTarget } = useScrollTarget(props, configureScrollTarget);
const { anchorEl, canShow, anchorEvents } = useAnchor({ showing, configureAnchorEl });
const { show, hide } = useModelToggle({
showing, canShow, handleShow, handleHide,
hideOnRouteChange,
processOnMount: true
});
Object.assign(anchorEvents, { delayShow, delayHide });
const { showPortal, hidePortal, renderPortal } = usePortal(vm, innerRef, renderPortalContent, 'tooltip');
// if we're on mobile, let's improve the experience
// by closing it when user taps outside of it
if ($q.platform.is.mobile === true) {
const clickOutsideProps = {
anchorEl,
innerRef,
onClickOutside (e) {
hide(e);
// prevent click if it's on a dialog backdrop
if (e.target.classList.contains('q-dialog__backdrop')) {
stopAndPrevent(e);
}
return true
}
};
const hasClickOutside = vue.computed(() =>
// it doesn't has external model
// (null is the default value)
props.modelValue === null
// and it's not persistent
&& props.persistent !== true
&& showing.value === true
);
vue.watch(hasClickOutside, val => {
const fn = val === true ? addClickOutside : removeClickOutside;
fn(clickOutsideProps);
});
vue.onBeforeUnmount(() => {
removeClickOutside(clickOutsideProps);
});
}
function handleShow (evt) {
showPortal();
// should removeTick() if this gets removed
registerTick(() => {
observer = new MutationObserver(() => updatePosition());
observer.observe(innerRef.value, { attributes: false, childList: true, characterData: true, subtree: true });
updatePosition();
configureScrollTarget();
});
if (unwatchPosition === void 0) {
unwatchPosition = vue.watch(
() => $q.screen.width + '|' + $q.screen.height + '|' + props.self + '|' + props.anchor + '|' + $q.lang.rtl,
updatePosition
);
}
// should removeTimeout() if this gets removed
registerTimeout(() => {
showPortal(true); // done showing portal
emit('show', evt);
}, props.transitionDuration);
}
function handleHide (evt) {
removeTick();
hidePortal();
anchorCleanup();
// should removeTimeout() if this gets removed
registerTimeout(() => {
hidePortal(true); // done hiding, now destroy
emit('hide', evt);
}, props.transitionDuration);
}
function anchorCleanup () {
if (observer !== void 0) {
observer.disconnect();
observer = void 0;
}
if (unwatchPosition !== void 0) {
unwatchPosition();
unwatchPosition = void 0;
}
unconfigureScrollTarget();
cleanEvt(anchorEvents, 'tooltipTemp');
}
function updatePosition () {
setPosition({
targetEl: innerRef.value,
offset: props.offset,
anchorEl: anchorEl.value,
anchorOrigin: anchorOrigin.value,
selfOrigin: selfOrigin.value,
maxHeight: props.maxHeight,
maxWidth: props.maxWidth
});
}
function delayShow (evt) {
if ($q.platform.is.mobile === true) {
clearSelection();
document.body.classList.add('non-selectable');
const target = anchorEl.value;
const evts = [ 'touchmove', 'touchcancel', 'touchend', 'click' ]
.map(e => ([ target, e, 'delayHide', 'passiveCapture' ]));
addEvt(anchorEvents, 'tooltipTemp', evts);
}
registerTimeout(() => { show(evt); }, props.delay);
}
function delayHide (evt) {
if ($q.platform.is.mobile === true) {
cleanEvt(anchorEvents, 'tooltipTemp');
clearSelection();
// delay needed otherwise selection still occurs
setTimeout(() => {
document.body.classList.remove('non-selectable');
}, 10);
}
// should removeTimeout() if this gets removed
registerTimeout(() => { hide(evt); }, props.hideDelay);
}
function configureAnchorEl () {
if (props.noParentEvent === true || anchorEl.value === null) { return }
const evts = $q.platform.is.mobile === true
? [
[ anchorEl.value, 'touchstart', 'delayShow', 'passive' ]
]
: [
[ anchorEl.value, 'mouseenter', 'delayShow', 'passive' ],
[ anchorEl.value, 'mouseleave', 'delayHide', 'passive' ]
];
addEvt(anchorEvents, 'anchor', evts);
}
function configureScrollTarget () {
if (anchorEl.value !== null || props.scrollTarget !== void 0) {
localScrollTarget.value = getScrollTarget(anchorEl.value, props.scrollTarget);
const fn = props.noParentEvent === true
? updatePosition
: hide;
changeScrollEvent(localScrollTarget.value, fn);
}
}
function getTooltipContent () {
return showing.value === true
? vue.h('div', {
...attrs,
ref: innerRef,
class: [
'q-tooltip q-tooltip--style q-position-engine no-pointer-events',
attrs.class
],
style: [
attrs.style,
transitionStyle.value
],
role: 'tooltip'
}, hSlot(slots.default))
: null
}
function renderPortalContent () {
return vue.h(vue.Transition, transitionProps.value, getTooltipContent)
}
vue.onBeforeUnmount(anchorCleanup);
// expose public methods
Object.assign(vm.proxy, { updatePosition });
return renderPortal
}
});
var QItem = createComponent({
name: 'QItem',
props: {
...useDarkProps,
...useRouterLinkProps,
tag: {
type: String,
default: 'div'
},
active: {
type: Boolean,
default: null
},
clickable: Boolean,
dense: Boolean,
insetLevel: Number,
tabindex: [ String, Number ],
focused: Boolean,
manualFocus: Boolean
},
emits: [ 'click', 'keyup' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const { hasLink, linkAttrs, linkClass, linkTag, navigateOnClick } = useRouterLink();
const rootRef = vue.ref(null);
const blurTargetRef = vue.ref(null);
const isActionable = vue.computed(() =>
props.clickable === true
|| hasLink.value === true
|| props.tag === 'label'
);
const isClickable = vue.computed(() =>
props.disable !== true && isActionable.value === true
);
const classes = vue.computed(() =>
'q-item q-item-type row no-wrap'
+ (props.dense === true ? ' q-item--dense' : '')
+ (isDark.value === true ? ' q-item--dark' : '')
+ (
hasLink.value === true && props.active === null
? linkClass.value
: (
props.active === true
? ` q-item--active${ props.activeClass !== void 0 ? ` ${ props.activeClass }` : '' }`
: ''
)
)
+ (props.disable === true ? ' disabled' : '')
+ (
isClickable.value === true
? ' q-item--clickable q-link cursor-pointer '
+ (props.manualFocus === true ? 'q-manual-focusable' : 'q-focusable q-hoverable')
+ (props.focused === true ? ' q-manual-focusable--focused' : '')
: ''
)
);
const style = vue.computed(() => {
if (props.insetLevel === void 0) {
return null
}
const dir = $q.lang.rtl === true ? 'Right' : 'Left';
return {
[ 'padding' + dir ]: (16 + props.insetLevel * 56) + 'px'
}
});
function onClick (e) {
if (isClickable.value === true) {
if (blurTargetRef.value !== null) {
if (e.qKeyEvent !== true && document.activeElement === rootRef.value) {
blurTargetRef.value.focus();
}
else if (document.activeElement === blurTargetRef.value) {
rootRef.value.focus();
}
}
navigateOnClick(e);
}
}
function onKeyup (e) {
if (isClickable.value === true && isKeyCode(e, 13) === true) {
stopAndPrevent(e);
// for ripple
e.qKeyEvent = true;
// for click trigger
const evt = new MouseEvent('click', e);
evt.qKeyEvent = true;
rootRef.value.dispatchEvent(evt);
}
emit('keyup', e);
}
function getContent () {
const child = hUniqueSlot(slots.default, []);
isClickable.value === true && child.unshift(
vue.h('div', { class: 'q-focus-helper', tabindex: -1, ref: blurTargetRef })
);
return child
}
return () => {
const data = {
ref: rootRef,
class: classes.value,
style: style.value,
role: 'listitem',
onClick,
onKeyup
};
if (isClickable.value === true) {
data.tabindex = props.tabindex || '0';
Object.assign(data, linkAttrs.value);
}
else if (isActionable.value === true) {
data[ 'aria-disabled' ] = 'true';
}
return vue.h(
linkTag.value,
data,
getContent()
)
}
}
});
var QItemSection = createComponent({
name: 'QItemSection',
props: {
avatar: Boolean,
thumbnail: Boolean,
side: Boolean,
top: Boolean,
noWrap: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() =>
'q-item__section column'
+ ` q-item__section--${ props.avatar === true || props.side === true || props.thumbnail === true ? 'side' : 'main' }`
+ (props.top === true ? ' q-item__section--top justify-start' : ' justify-center')
+ (props.avatar === true ? ' q-item__section--avatar' : '')
+ (props.thumbnail === true ? ' q-item__section--thumbnail' : '')
+ (props.noWrap === true ? ' q-item__section--nowrap' : '')
);
return () => vue.h('div', { class: classes.value }, hSlot(slots.default))
}
});
function run (e, btn, eVm) {
if (btn.handler) {
btn.handler(e, eVm, eVm.caret);
}
else {
eVm.runCmd(btn.cmd, btn.param);
}
}
function getGroup (children) {
return vue.h('div', { class: 'q-editor__toolbar-group' }, children)
}
function getBtn (eVm, btn, clickHandler, active = false) {
const
toggled = active || (btn.type === 'toggle'
? (btn.toggled ? btn.toggled(eVm) : btn.cmd && eVm.caret.is(btn.cmd, btn.param))
: false),
child = [];
if (btn.tip && eVm.$q.platform.is.desktop) {
const Key = btn.key
? vue.h('div', [
vue.h('small', `(CTRL + ${ String.fromCharCode(btn.key) })`)
])
: null;
child.push(
vue.h(QTooltip, { delay: 1000 }, () => [
vue.h('div', { innerHTML: btn.tip }),
Key
])
);
}
return vue.h(QBtn, {
...eVm.buttonProps.value,
icon: btn.icon !== null ? btn.icon : void 0,
color: toggled ? btn.toggleColor || eVm.props.toolbarToggleColor : btn.color || eVm.props.toolbarColor,
textColor: toggled && !eVm.props.toolbarPush ? null : btn.textColor || eVm.props.toolbarTextColor,
label: btn.label,
disable: btn.disable ? (typeof btn.disable === 'function' ? btn.disable(eVm) : true) : false,
size: 'sm',
onClick (e) {
clickHandler && clickHandler();
run(e, btn, eVm);
}
}, () => child)
}
function getDropdown (eVm, btn) {
const onlyIcons = btn.list === 'only-icons';
let
label = btn.label,
icon = btn.icon !== null ? btn.icon : void 0,
contentClass,
Items;
function closeDropdown () {
Dropdown.component.proxy.hide();
}
if (onlyIcons) {
Items = btn.options.map(btn => {
const active = btn.type === void 0
? eVm.caret.is(btn.cmd, btn.param)
: false;
if (active) {
label = btn.tip;
icon = btn.icon !== null ? btn.icon : void 0;
}
return getBtn(eVm, btn, closeDropdown, active)
});
contentClass = eVm.toolbarBackgroundClass.value;
Items = [
getGroup(Items)
];
}
else {
const activeClass = eVm.props.toolbarToggleColor !== void 0
? `text-${ eVm.props.toolbarToggleColor }`
: null;
const inactiveClass = eVm.props.toolbarTextColor !== void 0
? `text-${ eVm.props.toolbarTextColor }`
: null;
const noIcons = btn.list === 'no-icons';
Items = btn.options.map(btn => {
const disable = btn.disable ? btn.disable(eVm) : false;
const active = btn.type === void 0
? eVm.caret.is(btn.cmd, btn.param)
: false;
if (active) {
label = btn.tip;
icon = btn.icon !== null ? btn.icon : void 0;
}
const htmlTip = btn.htmlTip;
return vue.h(QItem, {
active,
activeClass,
clickable: true,
disable,
dense: true,
onClick (e) {
closeDropdown();
eVm.contentRef.value !== null && eVm.contentRef.value.focus();
eVm.caret.restore();
run(e, btn, eVm);
}
}, () => [
noIcons === true
? null
: vue.h(
QItemSection,
{
class: active ? activeClass : inactiveClass,
side: true
},
() => vue.h(QIcon, { name: btn.icon !== null ? btn.icon : void 0 })
),
vue.h(
QItemSection,
htmlTip
? () => vue.h('div', { class: 'text-no-wrap', innerHTML: btn.htmlTip })
: (btn.tip ? () => vue.h('div', { class: 'text-no-wrap' }, btn.tip) : void 0)
)
])
});
contentClass = [ eVm.toolbarBackgroundClass.value, inactiveClass ];
}
const highlight = btn.highlight && label !== btn.label;
const Dropdown = vue.h(QBtnDropdown, {
...eVm.buttonProps.value,
noCaps: true,
noWrap: true,
color: highlight ? eVm.props.toolbarToggleColor : eVm.props.toolbarColor,
textColor: highlight && !eVm.props.toolbarPush ? null : eVm.props.toolbarTextColor,
label: btn.fixedLabel ? btn.label : label,
icon: btn.fixedIcon ? (btn.icon !== null ? btn.icon : void 0) : icon,
contentClass,
onShow: evt => eVm.emit('dropdownShow', evt),
onHide: evt => eVm.emit('dropdownHide', evt),
onBeforeShow: evt => eVm.emit('dropdownBeforeShow', evt),
onBeforeHide: evt => eVm.emit('dropdownBeforeHide', evt)
}, () => Items);
return Dropdown
}
function getToolbar (eVm) {
if (eVm.caret) {
return eVm.buttons.value
.filter(f => {
return !eVm.isViewingSource.value || f.find(fb => fb.cmd === 'viewsource')
})
.map(group => getGroup(
group.map(btn => {
if (eVm.isViewingSource.value && btn.cmd !== 'viewsource') {
return false
}
if (btn.type === 'slot') {
return hSlot(eVm.slots[ btn.slot ])
}
if (btn.type === 'dropdown') {
return getDropdown(eVm, btn)
}
return getBtn(eVm, btn)
})
))
}
}
function getFonts (defaultFont, defaultFontLabel, defaultFontIcon, fonts = {}) {
const aliases = Object.keys(fonts);
if (aliases.length === 0) {
return {}
}
const def = {
default_font: {
cmd: 'fontName',
param: defaultFont,
icon: defaultFontIcon,
tip: defaultFontLabel
}
};
aliases.forEach(alias => {
const name = fonts[ alias ];
def[ alias ] = {
cmd: 'fontName',
param: name,
icon: defaultFontIcon,
tip: name,
htmlTip: `<font face="${ name }">${ name }</font>`
};
});
return def
}
function getLinkEditor (eVm) {
if (eVm.caret) {
const color = eVm.props.toolbarColor || eVm.props.toolbarTextColor;
let link = eVm.editLinkUrl.value;
const updateLink = () => {
eVm.caret.restore();
if (link !== eVm.editLinkUrl.value) {
document.execCommand('createLink', false, link === '' ? ' ' : link);
}
eVm.editLinkUrl.value = null;
};
return [
vue.h('div', { class: `q-mx-xs text-${ color }` }, `${ eVm.$q.lang.editor.url }: `),
vue.h('input', {
key: 'qedt_btm_input',
class: 'col q-editor__link-input',
value: link,
onInput: evt => {
stop(evt);
link = evt.target.value;
},
onKeydown: evt => {
if (shouldIgnoreKey(evt) === true) {
return
}
switch (evt.keyCode) {
case 13: // ENTER key
prevent(evt);
return updateLink()
case 27: // ESCAPE key
prevent(evt);
eVm.caret.restore();
if (!eVm.editLinkUrl.value || eVm.editLinkUrl.value === 'https://') {
document.execCommand('unlink');
}
eVm.editLinkUrl.value = null;
break
}
}
}),
getGroup([
vue.h(QBtn, {
key: 'qedt_btm_rem',
tabindex: -1,
...eVm.buttonProps.value,
label: eVm.$q.lang.label.remove,
noCaps: true,
onClick: () => {
eVm.caret.restore();
document.execCommand('unlink');
eVm.editLinkUrl.value = null;
}
}),
vue.h(QBtn, {
key: 'qedt_btm_upd',
...eVm.buttonProps.value,
label: eVm.$q.lang.label.update,
noCaps: true,
onClick: updateLink
})
])
]
}
}
const listenerRE = /^on[A-Z]/;
function useSplitAttrs (attrs, vnode) {
const acc = {
listeners: vue.ref({}),
attributes: vue.ref({})
};
function update () {
const attributes = {};
const listeners = {};
for (const key in attrs) {
if (key !== 'class' && key !== 'style' && listenerRE.test(key) === false) {
attributes[ key ] = attrs[ key ];
}
}
for (const key in vnode.props) {
if (listenerRE.test(key) === true) {
listeners[ key ] = vnode.props[ key ];
}
}
acc.attributes.value = attributes;
acc.listeners.value = listeners;
}
vue.onBeforeUpdate(update);
update();
return acc
}
const
toString = Object.prototype.toString,
hasOwn = Object.prototype.hasOwnProperty,
notPlainObject = new Set(
[ 'Boolean', 'Number', 'String', 'Function', 'Array', 'Date', 'RegExp' ]
.map(name => '[object ' + name + ']')
);
function isPlainObject (obj) {
if (obj !== Object(obj) || notPlainObject.has(toString.call(obj)) === true) {
return false
}
if (
obj.constructor
&& hasOwn.call(obj, 'constructor') === false
&& hasOwn.call(obj.constructor.prototype, 'isPrototypeOf') === false
) {
return false
}
let key;
for (key in obj) {} // eslint-disable-line
return key === void 0 || hasOwn.call(obj, key)
}
function extend () {
let
options, name, src, copy, copyIsArray, clone,
target = arguments[ 0 ] || {},
i = 1,
deep = false;
const length = arguments.length;
if (typeof target === 'boolean') {
deep = target;
target = arguments[ 1 ] || {};
i = 2;
}
if (Object(target) !== target && typeof target !== 'function') {
target = {};
}
if (length === i) {
target = this;
i--;
}
for (; i < length; i++) {
if ((options = arguments[ i ]) !== null) {
for (name in options) {
src = target[ name ];
copy = options[ name ];
if (target === copy) {
continue
}
if (
deep === true
&& copy
&& ((copyIsArray = Array.isArray(copy)) || isPlainObject(copy) === true)
) {
if (copyIsArray === true) {
clone = Array.isArray(src) === true ? src : [];
}
else {
clone = isPlainObject(src) === true ? src : {};
}
target[ name ] = extend(deep, clone, copy);
}
else if (copy !== void 0) {
target[ name ] = copy;
}
}
}
}
return target
}
var QEditor = createComponent({
name: 'QEditor',
props: {
...useDarkProps,
...useFullscreenProps,
modelValue: {
type: String,
required: true
},
readonly: Boolean,
disable: Boolean,
minHeight: {
type: String,
default: '10rem'
},
maxHeight: String,
height: String,
definitions: Object,
fonts: Object,
placeholder: String,
toolbar: {
type: Array,
validator: v => v.length === 0 || v.every(group => group.length),
default () {
return [
[ 'left', 'center', 'right', 'justify' ],
[ 'bold', 'italic', 'underline', 'strike' ],
[ 'undo', 'redo' ]
]
}
},
toolbarColor: String,
toolbarBg: String,
toolbarTextColor: String,
toolbarToggleColor: {
type: String,
default: 'primary'
},
toolbarOutline: Boolean,
toolbarPush: Boolean,
toolbarRounded: Boolean,
paragraphTag: {
type: String,
validator: v => [ 'div', 'p' ].includes(v),
default: 'div'
},
contentStyle: Object,
contentClass: [ Object, Array, String ],
square: Boolean,
flat: Boolean,
dense: Boolean
},
emits: [
...useFullscreenEmits,
'update:modelValue',
'keydown', 'click', 'mouseup', 'keyup', 'touchend',
'focus', 'blur',
'dropdownShow',
'dropdownHide',
'dropdownBeforeShow',
'dropdownBeforeHide',
'linkShow',
'linkHide'
],
setup (props, { slots, emit, attrs }) {
const { proxy, vnode } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const { inFullscreen, toggleFullscreen } = useFullscreen();
const splitAttrs = useSplitAttrs(attrs, vnode);
const rootRef = vue.ref(null);
const contentRef = vue.ref(null);
const editLinkUrl = vue.ref(null);
const isViewingSource = vue.ref(false);
const editable = vue.computed(() => !props.readonly && !props.disable);
let defaultFont, offsetBottom;
let lastEmit = props.modelValue;
{
document.execCommand('defaultParagraphSeparator', false, props.paragraphTag);
defaultFont = window.getComputedStyle(document.body).fontFamily;
}
const toolbarBackgroundClass = vue.computed(() => (
props.toolbarBg ? ` bg-${ props.toolbarBg }` : ''
));
const buttonProps = vue.computed(() => {
const flat = props.toolbarOutline !== true
&& props.toolbarPush !== true;
return {
type: 'a',
flat,
noWrap: true,
outline: props.toolbarOutline,
push: props.toolbarPush,
rounded: props.toolbarRounded,
dense: true,
color: props.toolbarColor,
disable: !editable.value,
size: 'sm'
}
});
const buttonDef = vue.computed(() => {
const
e = $q.lang.editor,
i = $q.iconSet.editor;
return {
bold: { cmd: 'bold', icon: i.bold, tip: e.bold, key: 66 },
italic: { cmd: 'italic', icon: i.italic, tip: e.italic, key: 73 },
strike: { cmd: 'strikeThrough', icon: i.strikethrough, tip: e.strikethrough, key: 83 },
underline: { cmd: 'underline', icon: i.underline, tip: e.underline, key: 85 },
unordered: { cmd: 'insertUnorderedList', icon: i.unorderedList, tip: e.unorderedList },
ordered: { cmd: 'insertOrderedList', icon: i.orderedList, tip: e.orderedList },
subscript: { cmd: 'subscript', icon: i.subscript, tip: e.subscript, htmlTip: 'x<subscript>2</subscript>' },
superscript: { cmd: 'superscript', icon: i.superscript, tip: e.superscript, htmlTip: 'x<superscript>2</superscript>' },
link: { cmd: 'link', disable: eVm => eVm.caret && !eVm.caret.can('link'), icon: i.hyperlink, tip: e.hyperlink, key: 76 },
fullscreen: { cmd: 'fullscreen', icon: i.toggleFullscreen, tip: e.toggleFullscreen, key: 70 },
viewsource: { cmd: 'viewsource', icon: i.viewSource, tip: e.viewSource },
quote: { cmd: 'formatBlock', param: 'BLOCKQUOTE', icon: i.quote, tip: e.quote, key: 81 },
left: { cmd: 'justifyLeft', icon: i.left, tip: e.left },
center: { cmd: 'justifyCenter', icon: i.center, tip: e.center },
right: { cmd: 'justifyRight', icon: i.right, tip: e.right },
justify: { cmd: 'justifyFull', icon: i.justify, tip: e.justify },
print: { type: 'no-state', cmd: 'print', icon: i.print, tip: e.print, key: 80 },
outdent: { type: 'no-state', disable: eVm => eVm.caret && !eVm.caret.can('outdent'), cmd: 'outdent', icon: i.outdent, tip: e.outdent },
indent: { type: 'no-state', disable: eVm => eVm.caret && !eVm.caret.can('indent'), cmd: 'indent', icon: i.indent, tip: e.indent },
removeFormat: { type: 'no-state', cmd: 'removeFormat', icon: i.removeFormat, tip: e.removeFormat },
hr: { type: 'no-state', cmd: 'insertHorizontalRule', icon: i.hr, tip: e.hr },
undo: { type: 'no-state', cmd: 'undo', icon: i.undo, tip: e.undo, key: 90 },
redo: { type: 'no-state', cmd: 'redo', icon: i.redo, tip: e.redo, key: 89 },
h1: { cmd: 'formatBlock', param: 'H1', icon: i.heading1 || i.heading, tip: e.heading1, htmlTip: `<h1 class="q-ma-none">${ e.heading1 }</h1>` },
h2: { cmd: 'formatBlock', param: 'H2', icon: i.heading2 || i.heading, tip: e.heading2, htmlTip: `<h2 class="q-ma-none">${ e.heading2 }</h2>` },
h3: { cmd: 'formatBlock', param: 'H3', icon: i.heading3 || i.heading, tip: e.heading3, htmlTip: `<h3 class="q-ma-none">${ e.heading3 }</h3>` },
h4: { cmd: 'formatBlock', param: 'H4', icon: i.heading4 || i.heading, tip: e.heading4, htmlTip: `<h4 class="q-ma-none">${ e.heading4 }</h4>` },
h5: { cmd: 'formatBlock', param: 'H5', icon: i.heading5 || i.heading, tip: e.heading5, htmlTip: `<h5 class="q-ma-none">${ e.heading5 }</h5>` },
h6: { cmd: 'formatBlock', param: 'H6', icon: i.heading6 || i.heading, tip: e.heading6, htmlTip: `<h6 class="q-ma-none">${ e.heading6 }</h6>` },
p: { cmd: 'formatBlock', param: props.paragraphTag, icon: i.heading, tip: e.paragraph },
code: { cmd: 'formatBlock', param: 'PRE', icon: i.code, htmlTip: `<code>${ e.code }</code>` },
'size-1': { cmd: 'fontSize', param: '1', icon: i.size1 || i.size, tip: e.size1, htmlTip: `<font size="1">${ e.size1 }</font>` },
'size-2': { cmd: 'fontSize', param: '2', icon: i.size2 || i.size, tip: e.size2, htmlTip: `<font size="2">${ e.size2 }</font>` },
'size-3': { cmd: 'fontSize', param: '3', icon: i.size3 || i.size, tip: e.size3, htmlTip: `<font size="3">${ e.size3 }</font>` },
'size-4': { cmd: 'fontSize', param: '4', icon: i.size4 || i.size, tip: e.size4, htmlTip: `<font size="4">${ e.size4 }</font>` },
'size-5': { cmd: 'fontSize', param: '5', icon: i.size5 || i.size, tip: e.size5, htmlTip: `<font size="5">${ e.size5 }</font>` },
'size-6': { cmd: 'fontSize', param: '6', icon: i.size6 || i.size, tip: e.size6, htmlTip: `<font size="6">${ e.size6 }</font>` },
'size-7': { cmd: 'fontSize', param: '7', icon: i.size7 || i.size, tip: e.size7, htmlTip: `<font size="7">${ e.size7 }</font>` }
}
});
const buttons = vue.computed(() => {
const userDef = props.definitions || {};
const def = props.definitions || props.fonts
? extend(
true,
{},
buttonDef.value,
userDef,
getFonts(
defaultFont,
$q.lang.editor.defaultFont,
$q.iconSet.editor.font,
props.fonts
)
)
: buttonDef.value;
return props.toolbar.map(
group => group.map(token => {
if (token.options) {
return {
type: 'dropdown',
icon: token.icon,
label: token.label,
size: 'sm',
dense: true,
fixedLabel: token.fixedLabel,
fixedIcon: token.fixedIcon,
highlight: token.highlight,
list: token.list,
options: token.options.map(item => def[ item ])
}
}
const obj = def[ token ];
if (obj) {
return obj.type === 'no-state' || (userDef[ token ] && (
obj.cmd === void 0 || (buttonDef.value[ obj.cmd ] && buttonDef.value[ obj.cmd ].type === 'no-state')
))
? obj
: Object.assign({ type: 'toggle' }, obj)
}
else {
return {
type: 'slot',
slot: token
}
}
})
)
});
const eVm = {
$q,
props,
slots,
emit,
// caret (will get injected after mount)
inFullscreen,
toggleFullscreen,
runCmd,
isViewingSource,
editLinkUrl,
toolbarBackgroundClass,
buttonProps,
contentRef,
buttons,
setContent
};
vue.watch(() => props.modelValue, v => {
if (lastEmit !== v) {
lastEmit = v;
setContent(v, true);
}
});
vue.watch(editLinkUrl, v => {
emit(`link${ v ? 'Show' : 'Hide' }`);
});
const hasToolbar = vue.computed(() => props.toolbar && props.toolbar.length !== 0);
const keys = vue.computed(() => {
const
k = {},
add = btn => {
if (btn.key) {
k[ btn.key ] = {
cmd: btn.cmd,
param: btn.param
};
}
};
buttons.value.forEach(group => {
group.forEach(token => {
if (token.options) {
token.options.forEach(add);
}
else {
add(token);
}
});
});
return k
});
const innerStyle = vue.computed(() => (
inFullscreen.value
? props.contentStyle
: [
{
minHeight: props.minHeight,
height: props.height,
maxHeight: props.maxHeight
},
props.contentStyle
]
));
const classes = vue.computed(() =>
`q-editor q-editor--${ isViewingSource.value === true ? 'source' : 'default' }`
+ (props.disable === true ? ' disabled' : '')
+ (inFullscreen.value === true ? ' fullscreen column' : '')
+ (props.square === true ? ' q-editor--square no-border-radius' : '')
+ (props.flat === true ? ' q-editor--flat' : '')
+ (props.dense === true ? ' q-editor--dense' : '')
+ (isDark.value === true ? ' q-editor--dark q-dark' : '')
);
const innerClass = vue.computed(() => ([
props.contentClass,
'q-editor__content',
{ col: inFullscreen.value, 'overflow-auto': inFullscreen.value || props.maxHeight }
]));
const attributes = vue.computed(() => (
props.disable === true
? { 'aria-disabled': 'true' }
: (props.readonly === true ? { 'aria-readonly': 'true' } : {})
));
function onInput () {
if (contentRef.value !== null) {
const prop = `inner${ isViewingSource.value === true ? 'Text' : 'HTML' }`;
const val = contentRef.value[ prop ];
if (val !== props.modelValue) {
lastEmit = val;
emit('update:modelValue', val);
}
}
}
function onKeydown (e) {
emit('keydown', e);
if (e.ctrlKey !== true || shouldIgnoreKey(e) === true) {
refreshToolbar();
return
}
const key = e.keyCode;
const target = keys.value[ key ];
if (target !== void 0) {
const { cmd, param } = target;
stopAndPrevent(e);
runCmd(cmd, param, false);
}
}
function onClick (e) {
refreshToolbar();
emit('click', e);
}
function onBlur (e) {
if (contentRef.value !== null) {
const { scrollTop, scrollHeight } = contentRef.value;
offsetBottom = scrollHeight - scrollTop;
}
eVm.caret.save();
emit('blur', e);
}
function onFocus (e) {
vue.nextTick(() => {
if (contentRef.value !== null && offsetBottom !== void 0) {
contentRef.value.scrollTop = contentRef.value.scrollHeight - offsetBottom;
}
});
emit('focus', e);
}
function onFocusin (e) {
const root = rootRef.value;
if (
root !== null
&& root.contains(e.target) === true
&& (
e.relatedTarget === null
|| root.contains(e.relatedTarget) !== true
)
) {
const prop = `inner${ isViewingSource.value === true ? 'Text' : 'HTML' }`;
eVm.caret.restorePosition(contentRef.value[ prop ].length);
refreshToolbar();
}
}
function onFocusout (e) {
const root = rootRef.value;
if (
root !== null
&& root.contains(e.target) === true
&& (
e.relatedTarget === null
|| root.contains(e.relatedTarget) !== true
)
) {
eVm.caret.savePosition();
refreshToolbar();
}
}
function onPointerStart () {
offsetBottom = void 0;
}
function onSelectionchange (e) {
eVm.caret.save();
}
function setContent (v, restorePosition) {
if (contentRef.value !== null) {
if (restorePosition === true) {
eVm.caret.savePosition();
}
const prop = `inner${ isViewingSource.value === true ? 'Text' : 'HTML' }`;
contentRef.value[ prop ] = v;
if (restorePosition === true) {
eVm.caret.restorePosition(contentRef.value[ prop ].length);
refreshToolbar();
}
}
}
function runCmd (cmd, param, update = true) {
focus();
eVm.caret.restore();
eVm.caret.apply(cmd, param, () => {
focus();
eVm.caret.save();
if (update) {
refreshToolbar();
}
});
}
function refreshToolbar () {
setTimeout(() => {
editLinkUrl.value = null;
proxy.$forceUpdate();
}, 1);
}
function focus () {
addFocusFn(() => {
contentRef.value !== null && contentRef.value.focus({ preventScroll: true });
});
}
function getContentEl () {
return contentRef.value
}
vue.onMounted(() => {
eVm.caret = proxy.caret = new Caret(contentRef.value, eVm);
setContent(props.modelValue);
refreshToolbar();
document.addEventListener('selectionchange', onSelectionchange);
});
vue.onBeforeUnmount(() => {
document.removeEventListener('selectionchange', onSelectionchange);
});
// expose public methods
Object.assign(proxy, {
runCmd, refreshToolbar, focus, getContentEl
});
return () => {
let toolbars;
if (hasToolbar.value) {
const bars = [
vue.h('div', {
key: 'qedt_top',
class: 'q-editor__toolbar row no-wrap scroll-x'
+ toolbarBackgroundClass.value
}, getToolbar(eVm))
];
editLinkUrl.value !== null && bars.push(
vue.h('div', {
key: 'qedt_btm',
class: 'q-editor__toolbar row no-wrap items-center scroll-x'
+ toolbarBackgroundClass.value
}, getLinkEditor(eVm))
);
toolbars = vue.h('div', {
key: 'toolbar_ctainer',
class: 'q-editor__toolbars-container'
}, bars);
}
return vue.h('div', {
ref: rootRef,
class: classes.value,
style: { height: inFullscreen.value === true ? '100%' : null },
...attributes.value,
onFocusin,
onFocusout
}, [
toolbars,
vue.h('div', {
ref: contentRef,
style: innerStyle.value,
class: innerClass.value,
contenteditable: editable.value,
placeholder: props.placeholder,
...({}),
...splitAttrs.listeners.value,
onInput,
onKeydown,
onClick,
onBlur,
onFocus,
// clean saved scroll position
onMousedown: onPointerStart,
onTouchstartPassive: onPointerStart
})
])
}
}
});
var QItemLabel = createComponent({
name: 'QItemLabel',
props: {
overline: Boolean,
caption: Boolean,
header: Boolean,
lines: [ Number, String ]
},
setup (props, { slots }) {
const parsedLines = vue.computed(() => parseInt(props.lines, 10));
const classes = vue.computed(() =>
'q-item__label'
+ (props.overline === true ? ' q-item__label--overline text-overline' : '')
+ (props.caption === true ? ' q-item__label--caption text-caption' : '')
+ (props.header === true ? ' q-item__label--header' : '')
+ (parsedLines.value === 1 ? ' ellipsis' : '')
);
const style = vue.computed(() => {
return props.lines !== void 0 && parsedLines.value > 1
? {
overflow: 'hidden',
display: '-webkit-box',
'-webkit-box-orient': 'vertical',
'-webkit-line-clamp': parsedLines.value
}
: null
});
return () => vue.h('div', {
style: style.value,
class: classes.value
}, hSlot(slots.default))
}
});
var QSlideTransition = createComponent({
name: 'QSlideTransition',
props: {
appear: Boolean,
duration: {
type: Number,
default: 300
}
},
emits: [ 'show', 'hide' ],
setup (props, { slots, emit }) {
let animating = false, doneFn, element;
let timer = null, timerFallback = null, animListener, lastEvent;
function cleanup () {
doneFn && doneFn();
doneFn = null;
animating = false;
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
if (timerFallback !== null) {
clearTimeout(timerFallback);
timerFallback = null;
}
element !== void 0 && element.removeEventListener('transitionend', animListener);
animListener = null;
}
function begin (el, height, done) {
// here overflowY is 'hidden'
if (height !== void 0) {
el.style.height = `${ height }px`;
}
el.style.transition = `height ${ props.duration }ms cubic-bezier(.25, .8, .50, 1)`;
animating = true;
doneFn = done;
}
function end (el, event) {
el.style.overflowY = null;
el.style.height = null;
el.style.transition = null;
cleanup();
event !== lastEvent && emit(event);
}
function onEnter (el, done) {
let pos = 0;
element = el;
// if animationg overflowY is already 'hidden'
if (animating === true) {
cleanup();
pos = el.offsetHeight === el.scrollHeight ? 0 : void 0;
}
else {
lastEvent = 'hide';
el.style.overflowY = 'hidden';
}
begin(el, pos, done);
timer = setTimeout(() => {
timer = null;
el.style.height = `${ el.scrollHeight }px`;
animListener = evt => {
timerFallback = null;
if (Object(evt) !== evt || evt.target === el) {
end(el, 'show');
}
};
el.addEventListener('transitionend', animListener);
timerFallback = setTimeout(animListener, props.duration * 1.1);
}, 100);
}
function onLeave (el, done) {
let pos;
element = el;
if (animating === true) {
cleanup();
}
else {
lastEvent = 'show';
// we need to set overflowY 'hidden' before calculating the height
// or else we get small differences
el.style.overflowY = 'hidden';
pos = el.scrollHeight;
}
begin(el, pos, done);
timer = setTimeout(() => {
timer = null;
el.style.height = 0;
animListener = evt => {
timerFallback = null;
if (Object(evt) !== evt || evt.target === el) {
end(el, 'hide');
}
};
el.addEventListener('transitionend', animListener);
timerFallback = setTimeout(animListener, props.duration * 1.1);
}, 100);
}
vue.onBeforeUnmount(() => {
animating === true && cleanup();
});
return () => vue.h(vue.Transition, {
css: false,
appear: props.appear,
onEnter,
onLeave
}, slots.default)
}
});
const insetMap = {
true: 'inset',
item: 'item-inset',
'item-thumbnail': 'item-thumbnail-inset'
};
const margins = {
xs: 2,
sm: 4,
md: 8,
lg: 16,
xl: 24
};
var QSeparator = createComponent({
name: 'QSeparator',
props: {
...useDarkProps,
spaced: [ Boolean, String ],
inset: [ Boolean, String ],
vertical: Boolean,
color: String,
size: String
},
setup (props) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const orientation = vue.computed(() => (
props.vertical === true
? 'vertical'
: 'horizontal'
));
const orientClass = vue.computed(() => ` q-separator--${ orientation.value }`);
const insetClass = vue.computed(() => (
props.inset !== false
? `${ orientClass.value }-${ insetMap[ props.inset ] }`
: ''
));
const classes = vue.computed(() =>
`q-separator${ orientClass.value }${ insetClass.value }`
+ (props.color !== void 0 ? ` bg-${ props.color }` : '')
+ (isDark.value === true ? ' q-separator--dark' : '')
);
const style = vue.computed(() => {
const acc = {};
if (props.size !== void 0) {
acc[ props.vertical === true ? 'width' : 'height' ] = props.size;
}
if (props.spaced !== false) {
const size = props.spaced === true
? `${ margins.md }px`
: props.spaced in margins ? `${ margins[ props.spaced ] }px` : props.spaced;
const dir = props.vertical === true
? [ 'Left', 'Right' ]
: [ 'Top', 'Bottom' ];
acc[ `margin${ dir[ 0 ] }` ] = acc[ `margin${ dir[ 1 ] }` ] = size;
}
return acc
});
return () => vue.h('hr', {
class: classes.value,
style: style.value,
'aria-orientation': orientation.value
})
}
});
const itemGroups = vue.shallowReactive({});
const LINK_PROPS = Object.keys(useRouterLinkProps);
var QExpansionItem = createComponent({
name: 'QExpansionItem',
props: {
...useRouterLinkProps,
...useModelToggleProps,
...useDarkProps,
icon: String,
label: String,
labelLines: [ Number, String ],
caption: String,
captionLines: [ Number, String ],
dense: Boolean,
toggleAriaLabel: String,
expandIcon: String,
expandedIcon: String,
expandIconClass: [ Array, String, Object ],
duration: Number,
headerInsetLevel: Number,
contentInsetLevel: Number,
expandSeparator: Boolean,
defaultOpened: Boolean,
hideExpandIcon: Boolean,
expandIconToggle: Boolean,
switchToggleSide: Boolean,
denseToggle: Boolean,
group: String,
popup: Boolean,
headerStyle: [ Array, String, Object ],
headerClass: [ Array, String, Object ]
},
emits: [
...useModelToggleEmits,
'click', 'afterShow', 'afterHide'
],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const showing = vue.ref(
props.modelValue !== null
? props.modelValue
: props.defaultOpened
);
const blurTargetRef = vue.ref(null);
const targetUid = uid$3();
const { show, hide, toggle } = useModelToggle({ showing });
let uniqueId, exitGroup;
const classes = vue.computed(() =>
'q-expansion-item q-item-type'
+ ` q-expansion-item--${ showing.value === true ? 'expanded' : 'collapsed' }`
+ ` q-expansion-item--${ props.popup === true ? 'popup' : 'standard' }`
);
const contentStyle = vue.computed(() => {
if (props.contentInsetLevel === void 0) {
return null
}
const dir = $q.lang.rtl === true ? 'Right' : 'Left';
return {
[ 'padding' + dir ]: (props.contentInsetLevel * 56) + 'px'
}
});
const hasLink = vue.computed(() =>
props.disable !== true && (
props.href !== void 0
|| (props.to !== void 0 && props.to !== null && props.to !== '')
)
);
const linkProps = vue.computed(() => {
const acc = {};
LINK_PROPS.forEach(key => {
acc[ key ] = props[ key ];
});
return acc
});
const isClickable = vue.computed(() =>
hasLink.value === true || props.expandIconToggle !== true
);
const expansionIcon = vue.computed(() => (
props.expandedIcon !== void 0 && showing.value === true
? props.expandedIcon
: props.expandIcon || $q.iconSet.expansionItem[ props.denseToggle === true ? 'denseIcon' : 'icon' ]
));
const activeToggleIcon = vue.computed(() =>
props.disable !== true && (hasLink.value === true || props.expandIconToggle === true)
);
const headerSlotScope = vue.computed(() => ({
expanded: showing.value === true,
detailsId: props.targetUid,
toggle,
show,
hide
}));
const toggleAriaAttrs = vue.computed(() => {
const toggleAriaLabel = props.toggleAriaLabel !== void 0
? props.toggleAriaLabel
: $q.lang.label[ showing.value === true ? 'collapse' : 'expand' ](props.label);
return {
role: 'button',
'aria-expanded': showing.value === true ? 'true' : 'false',
'aria-controls': targetUid,
'aria-label': toggleAriaLabel
}
});
vue.watch(() => props.group, name => {
exitGroup !== void 0 && exitGroup();
name !== void 0 && enterGroup();
});
function onHeaderClick (e) {
hasLink.value !== true && toggle(e);
emit('click', e);
}
function toggleIconKeyboard (e) {
e.keyCode === 13 && toggleIcon(e, true);
}
function toggleIcon (e, keyboard) {
keyboard !== true && blurTargetRef.value !== null && blurTargetRef.value.focus();
toggle(e);
stopAndPrevent(e);
}
function onShow () {
emit('afterShow');
}
function onHide () {
emit('afterHide');
}
function enterGroup () {
if (uniqueId === void 0) {
uniqueId = uid$3();
}
if (showing.value === true) {
itemGroups[ props.group ] = uniqueId;
}
const show = vue.watch(showing, val => {
if (val === true) {
itemGroups[ props.group ] = uniqueId;
}
else if (itemGroups[ props.group ] === uniqueId) {
delete itemGroups[ props.group ];
}
});
const group = vue.watch(
() => itemGroups[ props.group ],
(val, oldVal) => {
if (oldVal === uniqueId && val !== void 0 && val !== uniqueId) {
hide();
}
}
);
exitGroup = () => {
show();
group();
if (itemGroups[ props.group ] === uniqueId) {
delete itemGroups[ props.group ];
}
exitGroup = void 0;
};
}
function getToggleIcon () {
const data = {
class: [
'q-focusable relative-position cursor-pointer'
+ `${ props.denseToggle === true && props.switchToggleSide === true ? ' items-end' : '' }`,
props.expandIconClass
],
side: props.switchToggleSide !== true,
avatar: props.switchToggleSide
};
const child = [
vue.h(QIcon, {
class: 'q-expansion-item__toggle-icon'
+ (props.expandedIcon === void 0 && showing.value === true
? ' q-expansion-item__toggle-icon--rotated'
: ''),
name: expansionIcon.value
})
];
if (activeToggleIcon.value === true) {
Object.assign(data, {
tabindex: 0,
...toggleAriaAttrs.value,
onClick: toggleIcon,
onKeyup: toggleIconKeyboard
});
child.unshift(
vue.h('div', {
ref: blurTargetRef,
class: 'q-expansion-item__toggle-focus q-icon q-focus-helper q-focus-helper--rounded',
tabindex: -1
})
);
}
return vue.h(QItemSection, data, () => child)
}
function getHeaderChild () {
let child;
if (slots.header !== void 0) {
child = [].concat(slots.header(headerSlotScope.value));
}
else {
child = [
vue.h(QItemSection, () => [
vue.h(QItemLabel, { lines: props.labelLines }, () => props.label || ''),
props.caption
? vue.h(QItemLabel, { lines: props.captionLines, caption: true }, () => props.caption)
: null
])
];
props.icon && child[ props.switchToggleSide === true ? 'push' : 'unshift' ](
vue.h(QItemSection, {
side: props.switchToggleSide === true,
avatar: props.switchToggleSide !== true
}, () => vue.h(QIcon, { name: props.icon }))
);
}
if (props.disable !== true && props.hideExpandIcon !== true) {
child[ props.switchToggleSide === true ? 'unshift' : 'push' ](
getToggleIcon()
);
}
return child
}
function getHeader () {
const data = {
ref: 'item',
style: props.headerStyle,
class: props.headerClass,
dark: isDark.value,
disable: props.disable,
dense: props.dense,
insetLevel: props.headerInsetLevel
};
if (isClickable.value === true) {
data.clickable = true;
data.onClick = onHeaderClick;
Object.assign(
data,
hasLink.value === true ? linkProps.value : toggleAriaAttrs.value
);
}
return vue.h(QItem, data, getHeaderChild)
}
function getTransitionChild () {
return vue.withDirectives(
vue.h('div', {
key: 'e-content',
class: 'q-expansion-item__content relative-position',
style: contentStyle.value,
id: targetUid
}, hSlot(slots.default)),
[ [
vue.vShow,
showing.value
] ]
)
}
function getContent () {
const node = [
getHeader(),
vue.h(QSlideTransition, {
duration: props.duration,
onShow,
onHide
}, getTransitionChild)
];
if (props.expandSeparator === true) {
node.push(
vue.h(QSeparator, {
class: 'q-expansion-item__border q-expansion-item__border--top absolute-top',
dark: isDark.value
}),
vue.h(QSeparator, {
class: 'q-expansion-item__border q-expansion-item__border--bottom absolute-bottom',
dark: isDark.value
})
);
}
return node
}
props.group !== void 0 && enterGroup();
vue.onBeforeUnmount(() => {
exitGroup !== void 0 && exitGroup();
});
return () => vue.h('div', { class: classes.value }, [
vue.h('div', { class: 'q-expansion-item__container relative-position' }, getContent())
])
}
});
const labelPositions = [ 'top', 'right', 'bottom', 'left' ];
const useFabProps = {
type: {
type: String,
default: 'a'
},
outline: Boolean,
push: Boolean,
flat: Boolean,
unelevated: Boolean,
color: String,
textColor: String,
glossy: Boolean,
square: Boolean,
padding: String,
label: {
type: [ String, Number ],
default: ''
},
labelPosition: {
type: String,
default: 'right',
validator: v => labelPositions.includes(v)
},
externalLabel: Boolean,
hideLabel: {
type: Boolean
},
labelClass: [ Array, String, Object ],
labelStyle: [ Array, String, Object ],
disable: Boolean,
tabindex: [ Number, String ]
};
function useFab (props, showing) {
return {
formClass: vue.computed(() =>
`q-fab--form-${ props.square === true ? 'square' : 'rounded' }`
),
stacked: vue.computed(() =>
props.externalLabel === false
&& [ 'top', 'bottom' ].includes(props.labelPosition)
),
labelProps: vue.computed(() => {
if (props.externalLabel === true) {
const hideLabel = props.hideLabel === null
? showing.value === false
: props.hideLabel;
return {
action: 'push',
data: {
class: [
props.labelClass,
'q-fab__label q-tooltip--style q-fab__label--external'
+ ` q-fab__label--external-${ props.labelPosition }`
+ (hideLabel === true ? ' q-fab__label--external-hidden' : '')
],
style: props.labelStyle
}
}
}
return {
action: [ 'left', 'top' ].includes(props.labelPosition)
? 'unshift'
: 'push',
data: {
class: [
props.labelClass,
`q-fab__label q-fab__label--internal q-fab__label--internal-${ props.labelPosition }`
+ (props.hideLabel === true ? ' q-fab__label--internal-hidden' : '')
],
style: props.labelStyle
}
}
})
}
}
const directions = [ 'up', 'right', 'down', 'left' ];
const alignValues = [ 'left', 'center', 'right' ];
var QFab = createComponent({
name: 'QFab',
props: {
...useFabProps,
...useModelToggleProps,
icon: String,
activeIcon: String,
hideIcon: Boolean,
hideLabel: {
default: null
},
direction: {
type: String,
default: 'right',
validator: v => directions.includes(v)
},
persistent: Boolean,
verticalActionsAlign: {
type: String,
default: 'center',
validator: v => alignValues.includes(v)
}
},
emits: useModelToggleEmits,
setup (props, { slots }) {
const triggerRef = vue.ref(null);
const showing = vue.ref(props.modelValue === true);
const targetUid = uid$3();
const { proxy: { $q } } = vue.getCurrentInstance();
const { formClass, labelProps } = useFab(props, showing);
const hideOnRouteChange = vue.computed(() => props.persistent !== true);
const { hide, toggle } = useModelToggle({
showing,
hideOnRouteChange
});
const slotScope = vue.computed(() => ({ opened: showing.value }));
const classes = vue.computed(() =>
'q-fab z-fab row inline justify-center'
+ ` q-fab--align-${ props.verticalActionsAlign } ${ formClass.value }`
+ (showing.value === true ? ' q-fab--opened' : ' q-fab--closed')
);
const actionClass = vue.computed(() =>
'q-fab__actions flex no-wrap inline'
+ ` q-fab__actions--${ props.direction }`
+ ` q-fab__actions--${ showing.value === true ? 'opened' : 'closed' }`
);
const actionAttrs = vue.computed(() => {
const attrs = {
id: targetUid,
role: 'menu'
};
if (showing.value !== true) {
attrs[ 'aria-hidden' ] = 'true';
}
return attrs
});
const iconHolderClass = vue.computed(() =>
'q-fab__icon-holder '
+ ` q-fab__icon-holder--${ showing.value === true ? 'opened' : 'closed' }`
);
function getIcon (kebab, camel) {
const slotFn = slots[ kebab ];
const classes = `q-fab__${ kebab } absolute-full`;
return slotFn === void 0
? vue.h(QIcon, { class: classes, name: props[ camel ] || $q.iconSet.fab[ camel ] })
: vue.h('div', { class: classes }, slotFn(slotScope.value))
}
function getTriggerContent () {
const child = [];
props.hideIcon !== true && child.push(
vue.h('div', { class: iconHolderClass.value }, [
getIcon('icon', 'icon'),
getIcon('active-icon', 'activeIcon')
])
);
if (props.label !== '' || slots.label !== void 0) {
child[ labelProps.value.action ](
vue.h('div', labelProps.value.data, slots.label !== void 0 ? slots.label(slotScope.value) : [ props.label ])
);
}
return hMergeSlot(slots.tooltip, child)
}
vue.provide(fabKey, {
showing,
onChildClick (evt) {
hide(evt);
if (triggerRef.value !== null) {
triggerRef.value.$el.focus();
}
}
});
return () => vue.h('div', {
class: classes.value
}, [
vue.h(QBtn, {
ref: triggerRef,
class: formClass.value,
...props,
noWrap: true,
stack: props.stacked,
align: void 0,
icon: void 0,
label: void 0,
noCaps: true,
fab: true,
'aria-expanded': showing.value === true ? 'true' : 'false',
'aria-haspopup': 'true',
'aria-controls': targetUid,
onClick: toggle
}, getTriggerContent),
vue.h('div', { class: actionClass.value, ...actionAttrs.value }, hSlot(slots.default))
])
}
});
const anchorMap = {
start: 'self-end',
center: 'self-center',
end: 'self-start'
};
const anchorValues = Object.keys(anchorMap);
var QFabAction = createComponent({
name: 'QFabAction',
props: {
...useFabProps,
icon: {
type: String,
default: ''
},
anchor: {
type: String,
validator: v => anchorValues.includes(v)
},
to: [ String, Object ],
replace: Boolean
},
emits: [ 'click' ],
setup (props, { slots, emit }) {
const $fab = vue.inject(fabKey, () => ({
showing: { value: true },
onChildClick: noop
}));
const { formClass, labelProps } = useFab(props, $fab.showing);
const classes = vue.computed(() => {
const align = anchorMap[ props.anchor ];
return formClass.value + (align !== void 0 ? ` ${ align }` : '')
});
const isDisabled = vue.computed(() =>
props.disable === true
|| $fab.showing.value !== true
);
function click (e) {
$fab.onChildClick(e);
emit('click', e);
}
function getContent () {
const child = [];
if (slots.icon !== void 0) {
child.push(slots.icon());
}
else if (props.icon !== '') {
child.push(
vue.h(QIcon, { name: props.icon })
);
}
if (props.label !== '' || slots.label !== void 0) {
child[ labelProps.value.action ](
vue.h('div', labelProps.value.data, slots.label !== void 0 ? slots.label() : [ props.label ])
);
}
return hMergeSlot(slots.default, child)
}
// expose public methods
const vm = vue.getCurrentInstance();
Object.assign(vm.proxy, { click });
return () => vue.h(QBtn, {
class: classes.value,
...props,
noWrap: true,
stack: props.stacked,
icon: void 0,
label: void 0,
noCaps: true,
fabMini: true,
disable: isDisabled.value,
onClick: click
}, getContent)
}
});
function useFormChild ({ validate, resetValidation, requiresQForm }) {
const $form = vue.inject(formKey, false);
if ($form !== false) {
const { props, proxy } = vue.getCurrentInstance();
// export public method (so it can be used in QForm)
Object.assign(proxy, { validate, resetValidation });
vue.watch(() => props.disable, val => {
if (val === true) {
typeof resetValidation === 'function' && resetValidation();
$form.unbindComponent(proxy);
}
else {
$form.bindComponent(proxy);
}
});
vue.onMounted(() => {
// register to parent QForm
props.disable !== true && $form.bindComponent(proxy);
});
vue.onBeforeUnmount(() => {
// un-register from parent QForm
props.disable !== true && $form.unbindComponent(proxy);
});
}
else if (requiresQForm === true) {
console.error('Parent QForm not found on useFormChild()!');
}
}
const lazyRulesValues = [ true, false, 'ondemand' ];
const useValidateProps = {
modelValue: {},
error: {
type: Boolean,
default: null
},
errorMessage: String,
noErrorIcon: Boolean,
rules: Array,
reactiveRules: Boolean,
lazyRules: {
type: [ Boolean, String ],
validator: v => lazyRulesValues.includes(v)
}
};
function useValidate (focused, innerLoading) {
const { props, proxy } = vue.getCurrentInstance();
const innerError = vue.ref(false);
const innerErrorMessage = vue.ref(null);
const isDirtyModel = vue.ref(null);
useFormChild({ validate, resetValidation });
let validateIndex = 0, unwatchRules;
const hasRules = vue.computed(() =>
props.rules !== void 0
&& props.rules !== null
&& props.rules.length !== 0
);
const hasActiveRules = vue.computed(() =>
props.disable !== true
&& hasRules.value === true
);
const hasError = vue.computed(() =>
props.error === true || innerError.value === true
);
const errorMessage = vue.computed(() => (
typeof props.errorMessage === 'string' && props.errorMessage.length !== 0
? props.errorMessage
: innerErrorMessage.value
));
vue.watch(() => props.modelValue, () => {
validateIfNeeded();
});
vue.watch(() => props.reactiveRules, val => {
if (val === true) {
if (unwatchRules === void 0) {
unwatchRules = vue.watch(() => props.rules, () => {
validateIfNeeded(true);
});
}
}
else if (unwatchRules !== void 0) {
unwatchRules();
unwatchRules = void 0;
}
}, { immediate: true });
vue.watch(focused, val => {
if (val === true) {
if (isDirtyModel.value === null) {
isDirtyModel.value = false;
}
}
else if (isDirtyModel.value === false) {
isDirtyModel.value = true;
if (
hasActiveRules.value === true
&& props.lazyRules !== 'ondemand'
// Don't re-trigger if it's already in progress;
// It might mean that focus switched to submit btn and
// QForm's submit() has been called already (ENTER key)
&& innerLoading.value === false
) {
debouncedValidate();
}
}
});
function resetValidation () {
validateIndex++;
innerLoading.value = false;
isDirtyModel.value = null;
innerError.value = false;
innerErrorMessage.value = null;
debouncedValidate.cancel();
}
/*
* Return value
* - true (validation succeeded)
* - false (validation failed)
* - Promise (pending async validation)
*/
function validate (val = props.modelValue) {
if (hasActiveRules.value !== true) {
return true
}
const index = ++validateIndex;
const setDirty = innerLoading.value !== true
? () => { isDirtyModel.value = true; }
: () => {};
const update = (err, msg) => {
err === true && setDirty();
innerError.value = err;
innerErrorMessage.value = msg || null;
innerLoading.value = false;
};
const promises = [];
for (let i = 0; i < props.rules.length; i++) {
const rule = props.rules[ i ];
let res;
if (typeof rule === 'function') {
res = rule(val, testPattern);
}
else if (typeof rule === 'string' && testPattern[ rule ] !== void 0) {
res = testPattern[ rule ](val);
}
if (res === false || typeof res === 'string') {
update(true, res);
return false
}
else if (res !== true && res !== void 0) {
promises.push(res);
}
}
if (promises.length === 0) {
update(false);
return true
}
innerLoading.value = true;
return Promise.all(promises).then(
res => {
if (res === void 0 || Array.isArray(res) === false || res.length === 0) {
index === validateIndex && update(false);
return true
}
const msg = res.find(r => r === false || typeof r === 'string');
index === validateIndex && update(msg !== void 0, msg);
return msg === void 0
},
e => {
if (index === validateIndex) {
console.error(e);
update(true);
}
return false
}
)
}
function validateIfNeeded (changedRules) {
if (
hasActiveRules.value === true
&& props.lazyRules !== 'ondemand'
&& (isDirtyModel.value === true || (props.lazyRules !== true && changedRules !== true))
) {
debouncedValidate();
}
}
const debouncedValidate = debounce(validate, 0);
vue.onBeforeUnmount(() => {
unwatchRules !== void 0 && unwatchRules();
debouncedValidate.cancel();
});
// expose public methods & props
Object.assign(proxy, { resetValidation, validate });
injectProp(proxy, 'hasError', () => hasError.value);
return {
isDirtyModel,
hasRules,
hasError,
errorMessage,
validate,
resetValidation
}
}
function getTargetUid (val) {
return val === void 0 ? `f_${ uid$3() }` : val
}
function fieldValueIsFilled (val) {
return val !== void 0
&& val !== null
&& ('' + val).length !== 0
}
const useFieldProps = {
...useDarkProps,
...useValidateProps,
label: String,
stackLabel: Boolean,
hint: String,
hideHint: Boolean,
prefix: String,
suffix: String,
labelColor: String,
color: String,
bgColor: String,
filled: Boolean,
outlined: Boolean,
borderless: Boolean,
standout: [ Boolean, String ],
square: Boolean,
loading: Boolean,
labelSlot: Boolean,
bottomSlots: Boolean,
hideBottomSpace: Boolean,
rounded: Boolean,
dense: Boolean,
itemAligned: Boolean,
counter: Boolean,
clearable: Boolean,
clearIcon: String,
disable: Boolean,
readonly: Boolean,
autofocus: Boolean,
for: String,
maxlength: [ Number, String ]
};
const useFieldEmits = [ 'update:modelValue', 'clear', 'focus', 'blur', 'popupShow', 'popupHide' ];
function useFieldState () {
const { props, attrs, proxy, vnode } = vue.getCurrentInstance();
const isDark = useDark(props, proxy.$q);
return {
isDark,
editable: vue.computed(() =>
props.disable !== true && props.readonly !== true
),
innerLoading: vue.ref(false),
focused: vue.ref(false),
hasPopupOpen: false,
splitAttrs: useSplitAttrs(attrs, vnode),
targetUid: vue.ref(getTargetUid(props.for)),
rootRef: vue.ref(null),
targetRef: vue.ref(null),
controlRef: vue.ref(null)
/**
* user supplied additionals:
* innerValue - computed
* floatingLabel - computed
* inputRef - computed
* fieldClass - computed
* hasShadow - computed
* controlEvents - Object with fn(e)
* getControl - fn
* getInnerAppend - fn
* getControlChild - fn
* getShadowControl - fn
* showPopup - fn
*/
}
}
function useField (state) {
const { props, emit, slots, attrs, proxy } = vue.getCurrentInstance();
const { $q } = proxy;
let focusoutTimer = null;
if (state.hasValue === void 0) {
state.hasValue = vue.computed(() => fieldValueIsFilled(props.modelValue));
}
if (state.emitValue === void 0) {
state.emitValue = value => {
emit('update:modelValue', value);
};
}
if (state.controlEvents === void 0) {
state.controlEvents = {
onFocusin: onControlFocusin,
onFocusout: onControlFocusout
};
}
Object.assign(state, {
clearValue,
onControlFocusin,
onControlFocusout,
focus
});
if (state.computedCounter === void 0) {
state.computedCounter = vue.computed(() => {
if (props.counter !== false) {
const len = typeof props.modelValue === 'string' || typeof props.modelValue === 'number'
? ('' + props.modelValue).length
: (Array.isArray(props.modelValue) === true ? props.modelValue.length : 0);
const max = props.maxlength !== void 0
? props.maxlength
: props.maxValues;
return len + (max !== void 0 ? ' / ' + max : '')
}
});
}
const {
isDirtyModel,
hasRules,
hasError,
errorMessage,
resetValidation
} = useValidate(state.focused, state.innerLoading);
const floatingLabel = state.floatingLabel !== void 0
? vue.computed(() => props.stackLabel === true || state.focused.value === true || state.floatingLabel.value === true)
: vue.computed(() => props.stackLabel === true || state.focused.value === true || state.hasValue.value === true);
const shouldRenderBottom = vue.computed(() =>
props.bottomSlots === true
|| props.hint !== void 0
|| hasRules.value === true
|| props.counter === true
|| props.error !== null
);
const styleType = vue.computed(() => {
if (props.filled === true) { return 'filled' }
if (props.outlined === true) { return 'outlined' }
if (props.borderless === true) { return 'borderless' }
if (props.standout) { return 'standout' }
return 'standard'
});
const classes = vue.computed(() =>
`q-field row no-wrap items-start q-field--${ styleType.value }`
+ (state.fieldClass !== void 0 ? ` ${ state.fieldClass.value }` : '')
+ (props.rounded === true ? ' q-field--rounded' : '')
+ (props.square === true ? ' q-field--square' : '')
+ (floatingLabel.value === true ? ' q-field--float' : '')
+ (hasLabel.value === true ? ' q-field--labeled' : '')
+ (props.dense === true ? ' q-field--dense' : '')
+ (props.itemAligned === true ? ' q-field--item-aligned q-item-type' : '')
+ (state.isDark.value === true ? ' q-field--dark' : '')
+ (state.getControl === void 0 ? ' q-field--auto-height' : '')
+ (state.focused.value === true ? ' q-field--focused' : '')
+ (hasError.value === true ? ' q-field--error' : '')
+ (hasError.value === true || state.focused.value === true ? ' q-field--highlighted' : '')
+ (props.hideBottomSpace !== true && shouldRenderBottom.value === true ? ' q-field--with-bottom' : '')
+ (props.disable === true ? ' q-field--disabled' : (props.readonly === true ? ' q-field--readonly' : ''))
);
const contentClass = vue.computed(() =>
'q-field__control relative-position row no-wrap'
+ (props.bgColor !== void 0 ? ` bg-${ props.bgColor }` : '')
+ (
hasError.value === true
? ' text-negative'
: (
typeof props.standout === 'string' && props.standout.length !== 0 && state.focused.value === true
? ` ${ props.standout }`
: (props.color !== void 0 ? ` text-${ props.color }` : '')
)
)
);
const hasLabel = vue.computed(() =>
props.labelSlot === true || props.label !== void 0
);
const labelClass = vue.computed(() =>
'q-field__label no-pointer-events absolute ellipsis'
+ (props.labelColor !== void 0 && hasError.value !== true ? ` text-${ props.labelColor }` : '')
);
const controlSlotScope = vue.computed(() => ({
id: state.targetUid.value,
editable: state.editable.value,
focused: state.focused.value,
floatingLabel: floatingLabel.value,
modelValue: props.modelValue,
emitValue: state.emitValue
}));
const attributes = vue.computed(() => {
const acc = {
for: state.targetUid.value
};
if (props.disable === true) {
acc[ 'aria-disabled' ] = 'true';
}
else if (props.readonly === true) {
acc[ 'aria-readonly' ] = 'true';
}
return acc
});
vue.watch(() => props.for, val => {
// don't transform targetUid into a computed
// prop as it will break SSR
state.targetUid.value = getTargetUid(val);
});
function focusHandler () {
const el = document.activeElement;
let target = state.targetRef !== void 0 && state.targetRef.value;
if (target && (el === null || el.id !== state.targetUid.value)) {
target.hasAttribute('tabindex') === true || (target = target.querySelector('[tabindex]'));
if (target && target !== el) {
target.focus({ preventScroll: true });
}
}
}
function focus () {
addFocusFn(focusHandler);
}
function blur () {
removeFocusFn(focusHandler);
const el = document.activeElement;
if (el !== null && state.rootRef.value.contains(el)) {
el.blur();
}
}
function onControlFocusin (e) {
if (focusoutTimer !== null) {
clearTimeout(focusoutTimer);
focusoutTimer = null;
}
if (state.editable.value === true && state.focused.value === false) {
state.focused.value = true;
emit('focus', e);
}
}
function onControlFocusout (e, then) {
focusoutTimer !== null && clearTimeout(focusoutTimer);
focusoutTimer = setTimeout(() => {
focusoutTimer = null;
if (
document.hasFocus() === true && (
state.hasPopupOpen === true
|| state.controlRef === void 0
|| state.controlRef.value === null
|| state.controlRef.value.contains(document.activeElement) !== false
)
) {
return
}
if (state.focused.value === true) {
state.focused.value = false;
emit('blur', e);
}
then !== void 0 && then();
});
}
function clearValue (e) {
// prevent activating the field but keep focus on desktop
stopAndPrevent(e);
if ($q.platform.is.mobile !== true) {
const el = (state.targetRef !== void 0 && state.targetRef.value) || state.rootRef.value;
el.focus();
}
else if (state.rootRef.value.contains(document.activeElement) === true) {
document.activeElement.blur();
}
if (props.type === 'file') { // TODO vue3
// do not let focus be triggered
// as it will make the native file dialog
// appear for another selection
state.inputRef.value.value = null;
}
emit('update:modelValue', null);
emit('clear', props.modelValue);
vue.nextTick(() => {
resetValidation();
if ($q.platform.is.mobile !== true) {
isDirtyModel.value = false;
}
});
}
function getContent () {
const node = [];
slots.prepend !== void 0 && node.push(
vue.h('div', {
class: 'q-field__prepend q-field__marginal row no-wrap items-center',
key: 'prepend',
onClick: prevent
}, slots.prepend())
);
node.push(
vue.h('div', {
class: 'q-field__control-container col relative-position row no-wrap q-anchor--skip'
}, getControlContainer())
);
hasError.value === true && props.noErrorIcon === false && node.push(
getInnerAppendNode('error', [
vue.h(QIcon, { name: $q.iconSet.field.error, color: 'negative' })
])
);
if (props.loading === true || state.innerLoading.value === true) {
node.push(
getInnerAppendNode(
'inner-loading-append',
slots.loading !== void 0
? slots.loading()
: [ vue.h(QSpinner, { color: props.color }) ]
)
);
}
else if (props.clearable === true && state.hasValue.value === true && state.editable.value === true) {
node.push(
getInnerAppendNode('inner-clearable-append', [
vue.h(QIcon, {
class: 'q-field__focusable-action',
tag: 'button',
name: props.clearIcon || $q.iconSet.field.clear,
tabindex: 0,
type: 'button',
'aria-hidden': null,
role: null,
onClick: clearValue
})
])
);
}
slots.append !== void 0 && node.push(
vue.h('div', {
class: 'q-field__append q-field__marginal row no-wrap items-center',
key: 'append',
onClick: prevent
}, slots.append())
);
state.getInnerAppend !== void 0 && node.push(
getInnerAppendNode('inner-append', state.getInnerAppend())
);
state.getControlChild !== void 0 && node.push(
state.getControlChild()
);
return node
}
function getControlContainer () {
const node = [];
props.prefix !== void 0 && props.prefix !== null && node.push(
vue.h('div', {
class: 'q-field__prefix no-pointer-events row items-center'
}, props.prefix)
);
if (state.getShadowControl !== void 0 && state.hasShadow.value === true) {
node.push(
state.getShadowControl()
);
}
if (state.getControl !== void 0) {
node.push(state.getControl());
}
// internal usage only:
else if (slots.rawControl !== void 0) {
node.push(slots.rawControl());
}
else if (slots.control !== void 0) {
node.push(
vue.h('div', {
ref: state.targetRef,
class: 'q-field__native row',
tabindex: -1,
...state.splitAttrs.attributes.value,
'data-autofocus': props.autofocus === true || void 0
}, slots.control(controlSlotScope.value))
);
}
hasLabel.value === true && node.push(
vue.h('div', {
class: labelClass.value
}, hSlot(slots.label, props.label))
);
props.suffix !== void 0 && props.suffix !== null && node.push(
vue.h('div', {
class: 'q-field__suffix no-pointer-events row items-center'
}, props.suffix)
);
return node.concat(hSlot(slots.default))
}
function getBottom () {
let msg, key;
if (hasError.value === true) {
if (errorMessage.value !== null) {
msg = [ vue.h('div', { role: 'alert' }, errorMessage.value) ];
key = `q--slot-error-${ errorMessage.value }`;
}
else {
msg = hSlot(slots.error);
key = 'q--slot-error';
}
}
else if (props.hideHint !== true || state.focused.value === true) {
if (props.hint !== void 0) {
msg = [ vue.h('div', props.hint) ];
key = `q--slot-hint-${ props.hint }`;
}
else {
msg = hSlot(slots.hint);
key = 'q--slot-hint';
}
}
const hasCounter = props.counter === true || slots.counter !== void 0;
if (props.hideBottomSpace === true && hasCounter === false && msg === void 0) {
return
}
const main = vue.h('div', {
key,
class: 'q-field__messages col'
}, msg);
return vue.h('div', {
class: 'q-field__bottom row items-start q-field__bottom--'
+ (props.hideBottomSpace !== true ? 'animated' : 'stale'),
onClick: prevent
}, [
props.hideBottomSpace === true
? main
: vue.h(vue.Transition, { name: 'q-transition--field-message' }, () => main),
hasCounter === true
? vue.h('div', {
class: 'q-field__counter'
}, slots.counter !== void 0 ? slots.counter() : state.computedCounter.value)
: null
])
}
function getInnerAppendNode (key, content) {
return content === null
? null
: vue.h('div', {
key,
class: 'q-field__append q-field__marginal row no-wrap items-center q-anchor--skip'
}, content)
}
let shouldActivate = false;
vue.onDeactivated(() => {
shouldActivate = true;
});
vue.onActivated(() => {
shouldActivate === true && props.autofocus === true && proxy.focus();
});
vue.onMounted(() => {
if (isRuntimeSsrPreHydration.value === true && props.for === void 0) {
state.targetUid.value = getTargetUid();
}
props.autofocus === true && proxy.focus();
});
vue.onBeforeUnmount(() => {
focusoutTimer !== null && clearTimeout(focusoutTimer);
});
// expose public methods
Object.assign(proxy, { focus, blur });
return function renderField () {
const labelAttrs = state.getControl === void 0 && slots.control === void 0
? {
...state.splitAttrs.attributes.value,
'data-autofocus': props.autofocus === true || void 0,
...attributes.value
}
: attributes.value;
return vue.h('label', {
ref: state.rootRef,
class: [
classes.value,
attrs.class
],
style: attrs.style,
...labelAttrs
}, [
slots.before !== void 0
? vue.h('div', {
class: 'q-field__before q-field__marginal row no-wrap items-center',
onClick: prevent
}, slots.before())
: null,
vue.h('div', {
class: 'q-field__inner relative-position col self-stretch'
}, [
vue.h('div', {
ref: state.controlRef,
class: contentClass.value,
tabindex: -1,
...state.controlEvents
}, getContent()),
shouldRenderBottom.value === true
? getBottom()
: null
]),
slots.after !== void 0
? vue.h('div', {
class: 'q-field__after q-field__marginal row no-wrap items-center',
onClick: prevent
}, slots.after())
: null
])
}
}
var QField = createComponent({
name: 'QField',
inheritAttrs: false,
props: useFieldProps,
emits: useFieldEmits,
setup () {
return useField(useFieldState())
}
});
function filterFiles (files, rejectedFiles, failedPropValidation, filterFn) {
const acceptedFiles = [];
files.forEach(file => {
if (filterFn(file) === true) {
acceptedFiles.push(file);
}
else {
rejectedFiles.push({ failedPropValidation, file });
}
});
return acceptedFiles
}
function stopAndPreventDrag (e) {
e && e.dataTransfer && (e.dataTransfer.dropEffect = 'copy');
stopAndPrevent(e);
}
const useFileProps = {
multiple: Boolean,
accept: String,
capture: String,
maxFileSize: [ Number, String ],
maxTotalSize: [ Number, String ],
maxFiles: [ Number, String ],
filter: Function
};
const useFileEmits = [ 'rejected' ];
function useFile ({
editable,
dnd,
getFileInput,
addFilesToQueue
}) {
const { props, emit, proxy } = vue.getCurrentInstance();
const dndRef = vue.ref(null);
const extensions = vue.computed(() => (
props.accept !== void 0
? props.accept.split(',').map(ext => {
ext = ext.trim();
if (ext === '*') { // support "*"
return '*/'
}
else if (ext.endsWith('/*')) { // support "image/*" or "*/*"
ext = ext.slice(0, ext.length - 1);
}
return ext.toUpperCase()
})
: null
));
const maxFilesNumber = vue.computed(() => parseInt(props.maxFiles, 10));
const maxTotalSizeNumber = vue.computed(() => parseInt(props.maxTotalSize, 10));
function pickFiles (e) {
if (editable.value) {
if (e !== Object(e)) {
e = { target: null };
}
if (e.target !== null && e.target.matches('input[type="file"]') === true) {
// stop propagation if it's not a real pointer event
e.clientX === 0 && e.clientY === 0 && stop(e);
}
else {
const input = getFileInput();
input && input !== e.target && input.click(e);
}
}
}
function addFiles (files) {
if (editable.value && files) {
addFilesToQueue(null, files);
}
}
function processFiles (e, filesToProcess, currentFileList, append) {
let files = Array.from(filesToProcess || e.target.files);
const rejectedFiles = [];
const done = () => {
if (rejectedFiles.length !== 0) {
emit('rejected', rejectedFiles);
}
};
// filter file types
if (props.accept !== void 0 && extensions.value.indexOf('*/') === -1) {
files = filterFiles(files, rejectedFiles, 'accept', file => {
return extensions.value.some(ext => (
file.type.toUpperCase().startsWith(ext)
|| file.name.toUpperCase().endsWith(ext)
))
});
if (files.length === 0) { return done() }
}
// filter max file size
if (props.maxFileSize !== void 0) {
const maxFileSize = parseInt(props.maxFileSize, 10);
files = filterFiles(files, rejectedFiles, 'max-file-size', file => {
return file.size <= maxFileSize
});
if (files.length === 0) { return done() }
}
// Cordova/iOS allows selecting multiple files even when the
// multiple attribute is not specified. We also normalize drag'n'dropped
// files here:
if (props.multiple !== true && files.length !== 0) {
files = [ files[ 0 ] ];
}
// Compute key to use for each file
files.forEach(file => {
file.__key = file.webkitRelativePath + file.lastModified + file.name + file.size;
});
if (append === true) {
// Avoid duplicate files
const filenameMap = currentFileList.map(entry => entry.__key);
files = filterFiles(files, rejectedFiles, 'duplicate', file => {
return filenameMap.includes(file.__key) === false
});
}
if (files.length === 0) { return done() }
if (props.maxTotalSize !== void 0) {
let size = append === true
? currentFileList.reduce((total, file) => total + file.size, 0)
: 0;
files = filterFiles(files, rejectedFiles, 'max-total-size', file => {
size += file.size;
return size <= maxTotalSizeNumber.value
});
if (files.length === 0) { return done() }
}
// do we have custom filter function?
if (typeof props.filter === 'function') {
const filteredFiles = props.filter(files);
files = filterFiles(files, rejectedFiles, 'filter', file => {
return filteredFiles.includes(file)
});
}
if (props.maxFiles !== void 0) {
let filesNumber = append === true
? currentFileList.length
: 0;
files = filterFiles(files, rejectedFiles, 'max-files', () => {
filesNumber++;
return filesNumber <= maxFilesNumber.value
});
if (files.length === 0) { return done() }
}
done();
if (files.length !== 0) {
return files
}
}
function onDragover (e) {
stopAndPreventDrag(e);
dnd.value !== true && (dnd.value = true);
}
function onDragleave (e) {
stopAndPrevent(e);
// Safari bug: relatedTarget is null for over 10 years
// https://bugs.webkit.org/show_bug.cgi?id=66547
const gone = e.relatedTarget !== null || client.is.safari !== true
? e.relatedTarget !== dndRef.value
: document.elementsFromPoint(e.clientX, e.clientY).includes(dndRef.value) === false;
gone === true && (dnd.value = false);
}
function onDrop (e) {
stopAndPreventDrag(e);
const files = e.dataTransfer.files;
if (files.length !== 0) {
addFilesToQueue(null, files);
}
dnd.value = false;
}
function getDndNode (type) {
if (dnd.value === true) {
return vue.h('div', {
ref: dndRef,
class: `q-${ type }__dnd absolute-full`,
onDragenter: stopAndPreventDrag,
onDragover: stopAndPreventDrag,
onDragleave,
onDrop
})
}
}
// expose public methods
Object.assign(proxy, { pickFiles, addFiles });
return {
pickFiles,
addFiles,
onDragover,
onDragleave,
processFiles,
getDndNode,
maxFilesNumber,
maxTotalSizeNumber
}
}
function useFileFormDomProps (props, typeGuard) {
function getFormDomProps () {
const model = props.modelValue;
try {
const dt = 'DataTransfer' in window
? new DataTransfer()
: ('ClipboardEvent' in window
? new ClipboardEvent('').clipboardData
: void 0
);
if (Object(model) === model) {
('length' in model
? Array.from(model)
: [ model ]
).forEach(file => {
dt.items.add(file);
});
}
return {
files: dt.files
}
}
catch (e) {
return {
files: void 0
}
}
}
return typeGuard === true
? vue.computed(() => {
if (props.type !== 'file') {
return
}
return getFormDomProps()
})
: vue.computed(getFormDomProps)
}
var QFile = createComponent({
name: 'QFile',
inheritAttrs: false,
props: {
...useFieldProps,
...useFormProps,
...useFileProps,
/* SSR does not know about File & FileList */
modelValue: [ File, FileList, Array ],
append: Boolean,
useChips: Boolean,
displayValue: [ String, Number ],
tabindex: {
type: [ String, Number ],
default: 0
},
counterLabel: Function,
inputClass: [ Array, String, Object ],
inputStyle: [ Array, String, Object ]
},
emits: [
...useFieldEmits,
...useFileEmits
],
setup (props, { slots, emit, attrs }) {
const { proxy } = vue.getCurrentInstance();
const state = useFieldState();
const inputRef = vue.ref(null);
const dnd = vue.ref(false);
const nameProp = useFormInputNameAttr(props);
const {
pickFiles,
onDragover,
onDragleave,
processFiles,
getDndNode
} = useFile({ editable: state.editable, dnd, getFileInput, addFilesToQueue });
const formDomProps = useFileFormDomProps(props);
const innerValue = vue.computed(() => (
Object(props.modelValue) === props.modelValue
? ('length' in props.modelValue ? Array.from(props.modelValue) : [ props.modelValue ])
: []
));
const hasValue = vue.computed(() => fieldValueIsFilled(innerValue.value));
const selectedString = vue.computed(() =>
innerValue.value
.map(file => file.name)
.join(', ')
);
const totalSize = vue.computed(() =>
humanStorageSize(
innerValue.value.reduce((acc, file) => acc + file.size, 0)
)
);
const counterProps = vue.computed(() => ({
totalSize: totalSize.value,
filesNumber: innerValue.value.length,
maxFiles: props.maxFiles
}));
const inputAttrs = vue.computed(() => ({
tabindex: -1,
type: 'file',
title: '', // try to remove default tooltip,
accept: props.accept,
capture: props.capture,
name: nameProp.value,
...attrs,
id: state.targetUid.value,
disabled: state.editable.value !== true
}));
const fieldClass = vue.computed(() =>
'q-file q-field--auto-height'
+ (dnd.value === true ? ' q-file--dnd' : '')
);
const isAppending = vue.computed(() =>
props.multiple === true && props.append === true
);
function removeAtIndex (index) {
const files = innerValue.value.slice();
files.splice(index, 1);
emitValue(files);
}
function removeFile (file) {
const index = innerValue.value.indexOf(file);
if (index > -1) {
removeAtIndex(index);
}
}
function emitValue (files) {
emit('update:modelValue', props.multiple === true ? files : files[ 0 ]);
}
function onKeydown (e) {
// prevent form submit if ENTER is pressed
e.keyCode === 13 && prevent(e);
}
function onKeyup (e) {
// only on ENTER and SPACE to match native input field
if (e.keyCode === 13 || e.keyCode === 32) {
pickFiles(e);
}
}
function getFileInput () {
return inputRef.value
}
function addFilesToQueue (e, fileList) {
const files = processFiles(e, fileList, innerValue.value, isAppending.value);
const fileInput = getFileInput();
if (fileInput !== void 0 && fileInput !== null) {
fileInput.value = '';
}
// if nothing to do...
if (files === void 0) { return }
// protect against input @change being called in a loop
// like it happens on Safari, so don't emit same thing:
if (
props.multiple === true
? props.modelValue && files.every(f => innerValue.value.includes(f))
: props.modelValue === files[ 0 ]
) {
return
}
emitValue(
isAppending.value === true
? innerValue.value.concat(files)
: files
);
}
function getFiller () {
return [
vue.h('input', {
class: [ props.inputClass, 'q-file__filler' ],
style: props.inputStyle
})
]
}
function getSelection () {
if (slots.file !== void 0) {
return innerValue.value.length === 0
? getFiller()
: innerValue.value.map(
(file, index) => slots.file({ index, file, ref: this })
)
}
if (slots.selected !== void 0) {
return innerValue.value.length === 0
? getFiller()
: slots.selected({ files: innerValue.value, ref: this })
}
if (props.useChips === true) {
return innerValue.value.length === 0
? getFiller()
: innerValue.value.map((file, i) => vue.h(QChip, {
key: 'file-' + i,
removable: state.editable.value,
dense: true,
textColor: props.color,
tabindex: props.tabindex,
onRemove: () => { removeAtIndex(i); }
}, () => vue.h('span', {
class: 'ellipsis',
textContent: file.name
})))
}
const textContent = props.displayValue !== void 0
? props.displayValue
: selectedString.value;
return textContent.length !== 0
? [
vue.h('div', {
class: props.inputClass,
style: props.inputStyle,
textContent
})
]
: getFiller()
}
function getInput () {
const data = {
ref: inputRef,
...inputAttrs.value,
...formDomProps.value,
class: 'q-field__input fit absolute-full cursor-pointer',
onChange: addFilesToQueue
};
if (props.multiple === true) {
data.multiple = true;
}
return vue.h('input', data)
}
Object.assign(state, {
fieldClass,
emitValue,
hasValue,
inputRef,
innerValue,
floatingLabel: vue.computed(() =>
hasValue.value === true
|| fieldValueIsFilled(props.displayValue)
),
computedCounter: vue.computed(() => {
if (props.counterLabel !== void 0) {
return props.counterLabel(counterProps.value)
}
const max = props.maxFiles;
return `${ innerValue.value.length }${ max !== void 0 ? ' / ' + max : '' } (${ totalSize.value })`
}),
getControlChild: () => getDndNode('file'),
getControl: () => {
const data = {
ref: state.targetRef,
class: 'q-field__native row items-center cursor-pointer',
tabindex: props.tabindex
};
if (state.editable.value === true) {
Object.assign(data, { onDragover, onDragleave, onKeydown, onKeyup });
}
return vue.h('div', data, [ getInput() ].concat(getSelection()))
}
});
// expose public methods
Object.assign(proxy, {
removeAtIndex,
removeFile,
getNativeElement: () => inputRef.value // deprecated
});
injectProp(proxy, 'nativeEl', () => inputRef.value);
return useField(state)
}
});
var QFooter = createComponent({
name: 'QFooter',
props: {
modelValue: {
type: Boolean,
default: true
},
reveal: Boolean,
bordered: Boolean,
elevated: Boolean,
heightHint: {
type: [ String, Number ],
default: 50
}
},
emits: [ 'reveal', 'focusin' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QFooter needs to be child of QLayout');
return emptyRenderFn
}
const size = vue.ref(parseInt(props.heightHint, 10));
const revealed = vue.ref(true);
const windowHeight = vue.ref(
isRuntimeSsrPreHydration.value === true || $layout.isContainer.value === true
? 0
: window.innerHeight
);
const fixed = vue.computed(() =>
props.reveal === true
|| $layout.view.value.indexOf('F') > -1
|| ($q.platform.is.ios && $layout.isContainer.value === true)
);
const containerHeight = vue.computed(() => (
$layout.isContainer.value === true
? $layout.containerHeight.value
: windowHeight.value
));
const offset = vue.computed(() => {
if (props.modelValue !== true) {
return 0
}
if (fixed.value === true) {
return revealed.value === true ? size.value : 0
}
const offset = $layout.scroll.value.position + containerHeight.value + size.value - $layout.height.value;
return offset > 0 ? offset : 0
});
const hidden = vue.computed(() =>
props.modelValue !== true || (fixed.value === true && revealed.value !== true)
);
const revealOnFocus = vue.computed(() =>
props.modelValue === true && hidden.value === true && props.reveal === true
);
const classes = vue.computed(() =>
'q-footer q-layout__section--marginal '
+ (fixed.value === true ? 'fixed' : 'absolute') + '-bottom'
+ (props.bordered === true ? ' q-footer--bordered' : '')
+ (hidden.value === true ? ' q-footer--hidden' : '')
+ (
props.modelValue !== true
? ' q-layout--prevent-focus' + (fixed.value !== true ? ' hidden' : '')
: ''
)
);
const style = vue.computed(() => {
const
view = $layout.rows.value.bottom,
css = {};
if (view[ 0 ] === 'l' && $layout.left.space === true) {
css[ $q.lang.rtl === true ? 'right' : 'left' ] = `${ $layout.left.size }px`;
}
if (view[ 2 ] === 'r' && $layout.right.space === true) {
css[ $q.lang.rtl === true ? 'left' : 'right' ] = `${ $layout.right.size }px`;
}
return css
});
function updateLayout (prop, val) {
$layout.update('footer', prop, val);
}
function updateLocal (prop, val) {
if (prop.value !== val) {
prop.value = val;
}
}
function onResize ({ height }) {
updateLocal(size, height);
updateLayout('size', height);
}
function updateRevealed () {
if (props.reveal !== true) { return }
const { direction, position, inflectionPoint } = $layout.scroll.value;
updateLocal(revealed, (
direction === 'up'
|| position - inflectionPoint < 100
|| $layout.height.value - containerHeight.value - position - size.value < 300
));
}
function onFocusin (evt) {
if (revealOnFocus.value === true) {
updateLocal(revealed, true);
}
emit('focusin', evt);
}
vue.watch(() => props.modelValue, val => {
updateLayout('space', val);
updateLocal(revealed, true);
$layout.animate();
});
vue.watch(offset, val => {
updateLayout('offset', val);
});
vue.watch(() => props.reveal, val => {
val === false && updateLocal(revealed, props.modelValue);
});
vue.watch(revealed, val => {
$layout.animate();
emit('reveal', val);
});
vue.watch([ size, $layout.scroll, $layout.height ], updateRevealed);
vue.watch(() => $q.screen.height, val => {
$layout.isContainer.value !== true && updateLocal(windowHeight, val);
});
const instance = {};
$layout.instances.footer = instance;
props.modelValue === true && updateLayout('size', size.value);
updateLayout('space', props.modelValue);
updateLayout('offset', offset.value);
vue.onBeforeUnmount(() => {
if ($layout.instances.footer === instance) {
$layout.instances.footer = void 0;
updateLayout('size', 0);
updateLayout('offset', 0);
updateLayout('space', false);
}
});
return () => {
const child = hMergeSlot(slots.default, [
vue.h(QResizeObserver, {
debounce: 0,
onResize
})
]);
props.elevated === true && child.push(
vue.h('div', {
class: 'q-layout__shadow absolute-full overflow-hidden no-pointer-events'
})
);
return vue.h('footer', {
class: classes.value,
style: style.value,
onFocusin
}, child)
}
}
});
var QForm = createComponent({
name: 'QForm',
props: {
autofocus: Boolean,
noErrorFocus: Boolean,
noResetFocus: Boolean,
greedy: Boolean,
onSubmit: Function
},
emits: [ 'reset', 'validationSuccess', 'validationError' ],
setup (props, { slots, emit }) {
const vm = vue.getCurrentInstance();
const rootRef = vue.ref(null);
let validateIndex = 0;
const registeredComponents = [];
function validate (shouldFocus) {
const focus = typeof shouldFocus === 'boolean'
? shouldFocus
: props.noErrorFocus !== true;
const index = ++validateIndex;
const emitEvent = (res, ref) => {
emit('validation' + (res === true ? 'Success' : 'Error'), ref);
};
const validateComponent = comp => {
const valid = comp.validate();
return typeof valid.then === 'function'
? valid.then(
valid => ({ valid, comp }),
err => ({ valid: false, comp, err })
)
: Promise.resolve({ valid, comp })
};
const errorsPromise = props.greedy === true
? Promise
.all(registeredComponents.map(validateComponent))
.then(res => res.filter(r => r.valid !== true))
: registeredComponents
.reduce(
(acc, comp) => acc.then(() => {
return validateComponent(comp).then(r => {
if (r.valid === false) { return Promise.reject(r) }
})
}),
Promise.resolve()
)
.catch(error => [ error ]);
return errorsPromise.then(errors => {
if (errors === void 0 || errors.length === 0) {
index === validateIndex && emitEvent(true);
return true
}
// if not outdated already
if (index === validateIndex) {
const { comp, err } = errors[ 0 ];
err !== void 0 && console.error(err);
emitEvent(false, comp);
if (focus === true) {
// Try to focus first mounted and active component
const activeError = errors.find(({ comp }) => (
typeof comp.focus === 'function'
&& vmIsDestroyed(comp.$) === false
));
if (activeError !== void 0) {
activeError.comp.focus();
}
}
}
return false
})
}
function resetValidation () {
validateIndex++;
registeredComponents.forEach(comp => {
typeof comp.resetValidation === 'function' && comp.resetValidation();
});
}
function submit (evt) {
evt !== void 0 && stopAndPrevent(evt);
const index = validateIndex + 1;
validate().then(val => {
// if not outdated && validation succeeded
if (index === validateIndex && val === true) {
if (props.onSubmit !== void 0) {
emit('submit', evt);
}
else if (evt !== void 0 && evt.target !== void 0 && typeof evt.target.submit === 'function') {
evt.target.submit();
}
}
});
}
function reset (evt) {
evt !== void 0 && stopAndPrevent(evt);
emit('reset');
vue.nextTick(() => { // allow userland to reset values before
resetValidation();
if (props.autofocus === true && props.noResetFocus !== true) {
focus();
}
});
}
function focus () {
addFocusFn(() => {
if (rootRef.value === null) { return }
const target = rootRef.value.querySelector('[autofocus][tabindex], [data-autofocus][tabindex]')
|| rootRef.value.querySelector('[autofocus] [tabindex], [data-autofocus] [tabindex]')
|| rootRef.value.querySelector('[autofocus], [data-autofocus]')
|| Array.prototype.find.call(rootRef.value.querySelectorAll('[tabindex]'), el => el.tabIndex > -1);
target !== null && target !== void 0 && target.focus({ preventScroll: true });
});
}
vue.provide(formKey, {
bindComponent (vmProxy) {
registeredComponents.push(vmProxy);
},
unbindComponent (vmProxy) {
const index = registeredComponents.indexOf(vmProxy);
if (index > -1) {
registeredComponents.splice(index, 1);
}
}
});
let shouldActivate = false;
vue.onDeactivated(() => {
shouldActivate = true;
});
vue.onActivated(() => {
shouldActivate === true && props.autofocus === true && focus();
});
vue.onMounted(() => {
props.autofocus === true && focus();
});
// expose public methods
Object.assign(vm.proxy, {
validate,
resetValidation,
submit,
reset,
focus,
getValidationComponents: () => registeredComponents
});
return () => vue.h('form', {
class: 'q-form',
ref: rootRef,
onSubmit: submit,
onReset: reset
}, hSlot(slots.default))
}
});
var QFormChildMixin = {
inject: {
[ formKey ]: {
default: noop
}
},
watch: {
disable (val) {
const $form = this.$.provides[ formKey ];
if ($form !== void 0) {
if (val === true) {
this.resetValidation();
$form.unbindComponent(this);
}
else {
$form.bindComponent(this);
}
}
}
},
methods: {
validate () {},
resetValidation () {}
},
mounted () {
// register to parent QForm
const $form = this.$.provides[ formKey ];
$form !== void 0 && this.disable !== true && $form.bindComponent(this);
},
beforeUnmount () {
// un-register from parent QForm
const $form = this.$.provides[ formKey ];
$form !== void 0 && this.disable !== true && $form.unbindComponent(this);
}
};
var QHeader = createComponent({
name: 'QHeader',
props: {
modelValue: {
type: Boolean,
default: true
},
reveal: Boolean,
revealOffset: {
type: Number,
default: 250
},
bordered: Boolean,
elevated: Boolean,
heightHint: {
type: [ String, Number ],
default: 50
}
},
emits: [ 'reveal', 'focusin' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QHeader needs to be child of QLayout');
return emptyRenderFn
}
const size = vue.ref(parseInt(props.heightHint, 10));
const revealed = vue.ref(true);
const fixed = vue.computed(() =>
props.reveal === true
|| $layout.view.value.indexOf('H') > -1
|| ($q.platform.is.ios && $layout.isContainer.value === true)
);
const offset = vue.computed(() => {
if (props.modelValue !== true) {
return 0
}
if (fixed.value === true) {
return revealed.value === true ? size.value : 0
}
const offset = size.value - $layout.scroll.value.position;
return offset > 0 ? offset : 0
});
const hidden = vue.computed(() => props.modelValue !== true
|| (fixed.value === true && revealed.value !== true)
);
const revealOnFocus = vue.computed(() =>
props.modelValue === true && hidden.value === true && props.reveal === true
);
const classes = vue.computed(() =>
'q-header q-layout__section--marginal '
+ (fixed.value === true ? 'fixed' : 'absolute') + '-top'
+ (props.bordered === true ? ' q-header--bordered' : '')
+ (hidden.value === true ? ' q-header--hidden' : '')
+ (props.modelValue !== true ? ' q-layout--prevent-focus' : '')
);
const style = vue.computed(() => {
const
view = $layout.rows.value.top,
css = {};
if (view[ 0 ] === 'l' && $layout.left.space === true) {
css[ $q.lang.rtl === true ? 'right' : 'left' ] = `${ $layout.left.size }px`;
}
if (view[ 2 ] === 'r' && $layout.right.space === true) {
css[ $q.lang.rtl === true ? 'left' : 'right' ] = `${ $layout.right.size }px`;
}
return css
});
function updateLayout (prop, val) {
$layout.update('header', prop, val);
}
function updateLocal (prop, val) {
if (prop.value !== val) {
prop.value = val;
}
}
function onResize ({ height }) {
updateLocal(size, height);
updateLayout('size', height);
}
function onFocusin (evt) {
if (revealOnFocus.value === true) {
updateLocal(revealed, true);
}
emit('focusin', evt);
}
vue.watch(() => props.modelValue, val => {
updateLayout('space', val);
updateLocal(revealed, true);
$layout.animate();
});
vue.watch(offset, val => {
updateLayout('offset', val);
});
vue.watch(() => props.reveal, val => {
val === false && updateLocal(revealed, props.modelValue);
});
vue.watch(revealed, val => {
$layout.animate();
emit('reveal', val);
});
vue.watch($layout.scroll, scroll => {
props.reveal === true && updateLocal(revealed,
scroll.direction === 'up'
|| scroll.position <= props.revealOffset
|| scroll.position - scroll.inflectionPoint < 100
);
});
const instance = {};
$layout.instances.header = instance;
props.modelValue === true && updateLayout('size', size.value);
updateLayout('space', props.modelValue);
updateLayout('offset', offset.value);
vue.onBeforeUnmount(() => {
if ($layout.instances.header === instance) {
$layout.instances.header = void 0;
updateLayout('size', 0);
updateLayout('offset', 0);
updateLayout('space', false);
}
});
return () => {
const child = hUniqueSlot(slots.default, []);
props.elevated === true && child.push(
vue.h('div', {
class: 'q-layout__shadow absolute-full overflow-hidden no-pointer-events'
})
);
child.push(
vue.h(QResizeObserver, {
debounce: 0,
onResize
})
);
return vue.h('header', {
class: classes.value,
style: style.value,
onFocusin
}, child)
}
}
});
const useRatioProps = {
ratio: [ String, Number ]
};
function useRatio (props, naturalRatio) {
// return ratioStyle
return vue.computed(() => {
const ratio = Number(
props.ratio || (naturalRatio !== void 0 ? naturalRatio.value : void 0)
);
return isNaN(ratio) !== true && ratio > 0
? { paddingBottom: `${ 100 / ratio }%` }
: null
})
}
const defaultRatio = 16 / 9;
var QImg = createComponent({
name: 'QImg',
props: {
...useRatioProps,
src: String,
srcset: String,
sizes: String,
alt: String,
crossorigin: String,
decoding: String,
referrerpolicy: String,
draggable: Boolean,
loading: {
type: String,
default: 'lazy'
},
fetchpriority: {
type: String,
default: 'auto'
},
width: String,
height: String,
initialRatio: {
type: [ Number, String ],
default: defaultRatio
},
placeholderSrc: String,
fit: {
type: String,
default: 'cover'
},
position: {
type: String,
default: '50% 50%'
},
imgClass: String,
imgStyle: Object,
noSpinner: Boolean,
noNativeMenu: Boolean,
noTransition: Boolean,
spinnerColor: String,
spinnerSize: String
},
emits: [ 'load', 'error' ],
setup (props, { slots, emit }) {
const naturalRatio = vue.ref(props.initialRatio);
const ratioStyle = useRatio(props, naturalRatio);
let loadTimer = null, isDestroyed = false;
const images = [
vue.ref(null),
vue.ref(getPlaceholderSrc())
];
const position = vue.ref(0);
const isLoading = vue.ref(false);
const hasError = vue.ref(false);
const classes = vue.computed(() =>
`q-img q-img--${ props.noNativeMenu === true ? 'no-' : '' }menu`
);
const style = vue.computed(() => ({
width: props.width,
height: props.height
}));
const imgClass = vue.computed(() =>
`q-img__image ${ props.imgClass !== void 0 ? props.imgClass + ' ' : '' }`
+ `q-img__image--with${ props.noTransition === true ? 'out' : '' }-transition`
);
const imgStyle = vue.computed(() => ({
...props.imgStyle,
objectFit: props.fit,
objectPosition: props.position
}));
vue.watch(() => getCurrentSrc(), addImage);
function getCurrentSrc () {
return props.src || props.srcset || props.sizes
? {
src: props.src,
srcset: props.srcset,
sizes: props.sizes
}
: null
}
function getPlaceholderSrc () {
return props.placeholderSrc !== void 0
? { src: props.placeholderSrc }
: null
}
function addImage (imgProps) {
if (loadTimer !== null) {
clearTimeout(loadTimer);
loadTimer = null;
}
hasError.value = false;
if (imgProps === null) {
isLoading.value = false;
images[ position.value ^ 1 ].value = getPlaceholderSrc();
}
else {
isLoading.value = true;
}
images[ position.value ].value = imgProps;
}
function onLoad ({ target }) {
if (isDestroyed === true) { return }
if (loadTimer !== null) {
clearTimeout(loadTimer);
loadTimer = null;
}
naturalRatio.value = target.naturalHeight === 0
? 0.5
: target.naturalWidth / target.naturalHeight;
waitForCompleteness(target, 1);
}
function waitForCompleteness (target, count) {
// protect against running forever
if (isDestroyed === true || count === 1000) { return }
if (target.complete === true) {
onReady(target);
}
else {
loadTimer = setTimeout(() => {
loadTimer = null;
waitForCompleteness(target, count + 1);
}, 50);
}
}
function onReady (img) {
if (isDestroyed === true) { return }
position.value = position.value ^ 1;
images[ position.value ].value = null;
isLoading.value = false;
hasError.value = false;
emit('load', img.currentSrc || img.src);
}
function onError (err) {
if (loadTimer !== null) {
clearTimeout(loadTimer);
loadTimer = null;
}
isLoading.value = false;
hasError.value = true;
images[ position.value ].value = null;
images[ position.value ^ 1 ].value = getPlaceholderSrc();
emit('error', err);
}
function getImage (index) {
const img = images[ index ].value;
const data = {
key: 'img_' + index,
class: imgClass.value,
style: imgStyle.value,
crossorigin: props.crossorigin,
decoding: props.decoding,
referrerpolicy: props.referrerpolicy,
height: props.height,
width: props.width,
loading: props.loading,
fetchpriority: props.fetchpriority,
'aria-hidden': 'true',
draggable: props.draggable,
...img
};
if (position.value === index) {
data.class += ' q-img__image--waiting';
Object.assign(data, { onLoad, onError });
}
else {
data.class += ' q-img__image--loaded';
}
return vue.h(
'div',
{ class: 'q-img__container absolute-full', key: 'img' + index },
vue.h('img', data)
)
}
function getContent () {
if (isLoading.value !== true) {
return vue.h('div', {
key: 'content',
class: 'q-img__content absolute-full q-anchor--skip'
}, hSlot(slots[ hasError.value === true ? 'error' : 'default' ]))
}
return vue.h('div', {
key: 'loading',
class: 'q-img__loading absolute-full flex flex-center'
}, (
slots.loading !== void 0
? slots.loading()
: (
props.noSpinner === true
? void 0
: [
vue.h(QSpinner, {
color: props.spinnerColor,
size: props.spinnerSize
})
]
)
))
}
{
{
addImage(getCurrentSrc());
}
vue.onBeforeUnmount(() => {
isDestroyed = true;
if (loadTimer !== null) {
clearTimeout(loadTimer);
loadTimer = null;
}
});
}
return () => {
const content = [];
if (ratioStyle.value !== null) {
content.push(
vue.h('div', { key: 'filler', style: ratioStyle.value })
);
}
if (hasError.value !== true) {
if (images[ 0 ].value !== null) {
content.push(getImage(0));
}
if (images[ 1 ].value !== null) {
content.push(getImage(1));
}
}
content.push(
vue.h(vue.Transition, { name: 'q-transition--fade' }, getContent)
);
return vue.h('div', {
class: classes.value,
style: style.value,
role: 'img',
'aria-label': props.alt
}, content)
}
}
});
const { passive: passive$3 } = listenOpts;
var QInfiniteScroll = createComponent({
name: 'QInfiniteScroll',
props: {
offset: {
type: Number,
default: 500
},
debounce: {
type: [ String, Number ],
default: 100
},
scrollTarget: {
default: void 0
},
initialIndex: Number,
disable: Boolean,
reverse: Boolean
},
emits: [ 'load' ],
setup (props, { slots, emit }) {
const isFetching = vue.ref(false);
const isWorking = vue.ref(true);
const rootRef = vue.ref(null);
const loadingRef = vue.ref(null);
let index = props.initialIndex || 0;
let localScrollTarget, poll;
const classes = vue.computed(() =>
'q-infinite-scroll__loading'
+ (isFetching.value === true ? '' : ' invisible')
);
function immediatePoll () {
if (props.disable === true || isFetching.value === true || isWorking.value === false) {
return
}
const
scrollHeight = getScrollHeight(localScrollTarget),
scrollPosition = getVerticalScrollPosition(localScrollTarget),
containerHeight = height(localScrollTarget);
if (props.reverse === false) {
if (Math.round(scrollPosition + containerHeight + props.offset) >= Math.round(scrollHeight)) {
trigger();
}
}
else if (Math.round(scrollPosition) <= props.offset) {
trigger();
}
}
function trigger () {
if (props.disable === true || isFetching.value === true || isWorking.value === false) {
return
}
index++;
isFetching.value = true;
const heightBefore = getScrollHeight(localScrollTarget);
emit('load', index, isDone => {
if (isWorking.value === true) {
isFetching.value = false;
vue.nextTick(() => {
if (props.reverse === true) {
const
heightAfter = getScrollHeight(localScrollTarget),
scrollPosition = getVerticalScrollPosition(localScrollTarget),
heightDifference = heightAfter - heightBefore;
setVerticalScrollPosition(localScrollTarget, scrollPosition + heightDifference);
}
if (isDone === true) {
stop();
}
else if (rootRef.value) {
rootRef.value.closest('body') && poll();
}
});
}
});
}
function reset () {
index = 0;
}
function resume () {
if (isWorking.value === false) {
isWorking.value = true;
localScrollTarget.addEventListener('scroll', poll, passive$3);
}
immediatePoll();
}
function stop () {
if (isWorking.value === true) {
isWorking.value = false;
isFetching.value = false;
localScrollTarget.removeEventListener('scroll', poll, passive$3);
if (poll !== void 0 && poll.cancel !== void 0) {
poll.cancel();
}
}
}
function updateScrollTarget () {
if (localScrollTarget && isWorking.value === true) {
localScrollTarget.removeEventListener('scroll', poll, passive$3);
}
localScrollTarget = getScrollTarget(rootRef.value, props.scrollTarget);
if (isWorking.value === true) {
localScrollTarget.addEventListener('scroll', poll, passive$3);
if (props.reverse === true) {
const
scrollHeight = getScrollHeight(localScrollTarget),
containerHeight = height(localScrollTarget);
setVerticalScrollPosition(localScrollTarget, scrollHeight - containerHeight);
}
immediatePoll();
}
}
function setIndex (newIndex) {
index = newIndex;
}
function setDebounce (val) {
val = parseInt(val, 10);
const oldPoll = poll;
poll = val <= 0
? immediatePoll
: debounce(immediatePoll, isNaN(val) === true ? 100 : val);
if (localScrollTarget && isWorking.value === true) {
if (oldPoll !== void 0) {
localScrollTarget.removeEventListener('scroll', oldPoll, passive$3);
}
localScrollTarget.addEventListener('scroll', poll, passive$3);
}
}
function updateSvgAnimations (isRetry) {
if (renderLoadingSlot.value === true) {
if (loadingRef.value === null) {
isRetry !== true && vue.nextTick(() => { updateSvgAnimations(true); });
return
}
// we need to pause svg animations (if any) when hiding
// otherwise the browser will keep on recalculating the style
const action = `${ isFetching.value === true ? 'un' : '' }pauseAnimations`;
Array.from(loadingRef.value.getElementsByTagName('svg')).forEach(el => {
el[ action ]();
});
}
}
const renderLoadingSlot = vue.computed(() => props.disable !== true && isWorking.value === true);
vue.watch([ isFetching, renderLoadingSlot ], () => { updateSvgAnimations(); });
vue.watch(() => props.disable, val => {
if (val === true) { stop(); }
else { resume(); }
});
vue.watch(() => props.reverse, () => {
if (isFetching.value === false && isWorking.value === true) {
immediatePoll();
}
});
vue.watch(() => props.scrollTarget, updateScrollTarget);
vue.watch(() => props.debounce, setDebounce);
let scrollPos = false;
vue.onActivated(() => {
if (scrollPos !== false && localScrollTarget) {
setVerticalScrollPosition(localScrollTarget, scrollPos);
}
});
vue.onDeactivated(() => {
scrollPos = localScrollTarget
? getVerticalScrollPosition(localScrollTarget)
: false;
});
vue.onBeforeUnmount(() => {
if (isWorking.value === true) {
localScrollTarget.removeEventListener('scroll', poll, passive$3);
}
});
vue.onMounted(() => {
setDebounce(props.debounce);
updateScrollTarget();
isFetching.value === false && updateSvgAnimations();
});
// expose public methods
const vm = vue.getCurrentInstance();
Object.assign(vm.proxy, {
poll: () => { poll !== void 0 && poll(); },
trigger, stop, reset, resume, setIndex
});
return () => {
const child = hUniqueSlot(slots.default, []);
if (renderLoadingSlot.value === true) {
child[ props.reverse === false ? 'push' : 'unshift' ](
vue.h('div', { ref: loadingRef, class: classes.value }, hSlot(slots.loading))
);
}
return vue.h('div', {
class: 'q-infinite-scroll',
ref: rootRef
}, child)
}
}
});
var QInnerLoading = createComponent({
name: 'QInnerLoading',
props: {
...useDarkProps,
...useTransitionProps,
showing: Boolean,
color: String,
size: {
type: [ String, Number ],
default: 42
},
label: String,
labelClass: String,
labelStyle: [ String, Array, Object ]
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const { transitionProps, transitionStyle } = useTransition(props);
const classes = vue.computed(() =>
'q-inner-loading absolute-full column flex-center'
+ (isDark.value === true ? ' q-inner-loading--dark' : '')
);
const labelClass = vue.computed(() =>
'q-inner-loading__label'
+ (props.labelClass !== void 0 ? ` ${ props.labelClass }` : '')
);
function getInner () {
const child = [
vue.h(QSpinner, {
size: props.size,
color: props.color
})
];
if (props.label !== void 0) {
child.push(
vue.h('div', {
class: labelClass.value,
style: props.labelStyle
}, [ props.label ])
);
}
return child
}
function getContent () {
return props.showing === true
? vue.h(
'div',
{ class: classes.value, style: transitionStyle.value },
slots.default !== void 0
? slots.default()
: getInner()
)
: null
}
return () => vue.h(vue.Transition, transitionProps.value, getContent)
}
});
// leave NAMED_MASKS at top of file (code referenced from docs)
const NAMED_MASKS = {
date: '####/##/##',
datetime: '####/##/## ##:##',
time: '##:##',
fulltime: '##:##:##',
phone: '(###) ### - ####',
card: '#### #### #### ####'
};
const TOKENS = {
'#': { pattern: '[\\d]', negate: '[^\\d]' },
S: { pattern: '[a-zA-Z]', negate: '[^a-zA-Z]' },
N: { pattern: '[0-9a-zA-Z]', negate: '[^0-9a-zA-Z]' },
A: { pattern: '[a-zA-Z]', negate: '[^a-zA-Z]', transform: v => v.toLocaleUpperCase() },
a: { pattern: '[a-zA-Z]', negate: '[^a-zA-Z]', transform: v => v.toLocaleLowerCase() },
X: { pattern: '[0-9a-zA-Z]', negate: '[^0-9a-zA-Z]', transform: v => v.toLocaleUpperCase() },
x: { pattern: '[0-9a-zA-Z]', negate: '[^0-9a-zA-Z]', transform: v => v.toLocaleLowerCase() }
};
const KEYS = Object.keys(TOKENS);
KEYS.forEach(key => {
TOKENS[ key ].regex = new RegExp(TOKENS[ key ].pattern);
});
const
tokenRegexMask = new RegExp('\\\\([^.*+?^${}()|([\\]])|([.*+?^${}()|[\\]])|([' + KEYS.join('') + '])|(.)', 'g'),
escRegex = /[.*+?^${}()|[\]\\]/g;
const MARKER = String.fromCharCode(1);
const useMaskProps = {
mask: String,
reverseFillMask: Boolean,
fillMask: [ Boolean, String ],
unmaskedValue: Boolean
};
function useMask (props, emit, emitValue, inputRef) {
let maskMarked, maskReplaced, computedMask, computedUnmask, pastedTextStart, selectionAnchor;
const hasMask = vue.ref(null);
const innerValue = vue.ref(getInitialMaskedValue());
function getIsTypeText () {
return props.autogrow === true
|| [ 'textarea', 'text', 'search', 'url', 'tel', 'password' ].includes(props.type)
}
vue.watch(() => props.type + props.autogrow, updateMaskInternals);
vue.watch(() => props.mask, v => {
if (v !== void 0) {
updateMaskValue(innerValue.value, true);
}
else {
const val = unmaskValue(innerValue.value);
updateMaskInternals();
props.modelValue !== val && emit('update:modelValue', val);
}
});
vue.watch(() => props.fillMask + props.reverseFillMask, () => {
hasMask.value === true && updateMaskValue(innerValue.value, true);
});
vue.watch(() => props.unmaskedValue, () => {
hasMask.value === true && updateMaskValue(innerValue.value);
});
function getInitialMaskedValue () {
updateMaskInternals();
if (hasMask.value === true) {
const masked = maskValue(unmaskValue(props.modelValue));
return props.fillMask !== false
? fillWithMask(masked)
: masked
}
return props.modelValue
}
function getPaddedMaskMarked (size) {
if (size < maskMarked.length) {
return maskMarked.slice(-size)
}
let pad = '', localMaskMarked = maskMarked;
const padPos = localMaskMarked.indexOf(MARKER);
if (padPos > -1) {
for (let i = size - localMaskMarked.length; i > 0; i--) {
pad += MARKER;
}
localMaskMarked = localMaskMarked.slice(0, padPos) + pad + localMaskMarked.slice(padPos);
}
return localMaskMarked
}
function updateMaskInternals () {
hasMask.value = props.mask !== void 0
&& props.mask.length !== 0
&& getIsTypeText();
if (hasMask.value === false) {
computedUnmask = void 0;
maskMarked = '';
maskReplaced = '';
return
}
const
localComputedMask = NAMED_MASKS[ props.mask ] === void 0
? props.mask
: NAMED_MASKS[ props.mask ],
fillChar = typeof props.fillMask === 'string' && props.fillMask.length !== 0
? props.fillMask.slice(0, 1)
: '_',
fillCharEscaped = fillChar.replace(escRegex, '\\$&'),
unmask = [],
extract = [],
mask = [];
let
firstMatch = props.reverseFillMask === true,
unmaskChar = '',
negateChar = '';
localComputedMask.replace(tokenRegexMask, (_, char1, esc, token, char2) => {
if (token !== void 0) {
const c = TOKENS[ token ];
mask.push(c);
negateChar = c.negate;
if (firstMatch === true) {
extract.push('(?:' + negateChar + '+)?(' + c.pattern + '+)?(?:' + negateChar + '+)?(' + c.pattern + '+)?');
firstMatch = false;
}
extract.push('(?:' + negateChar + '+)?(' + c.pattern + ')?');
}
else if (esc !== void 0) {
unmaskChar = '\\' + (esc === '\\' ? '' : esc);
mask.push(esc);
unmask.push('([^' + unmaskChar + ']+)?' + unmaskChar + '?');
}
else {
const c = char1 !== void 0 ? char1 : char2;
unmaskChar = c === '\\' ? '\\\\\\\\' : c.replace(escRegex, '\\\\$&');
mask.push(c);
unmask.push('([^' + unmaskChar + ']+)?' + unmaskChar + '?');
}
});
const
unmaskMatcher = new RegExp(
'^'
+ unmask.join('')
+ '(' + (unmaskChar === '' ? '.' : '[^' + unmaskChar + ']') + '+)?'
+ (unmaskChar === '' ? '' : '[' + unmaskChar + ']*') + '$'
),
extractLast = extract.length - 1,
extractMatcher = extract.map((re, index) => {
if (index === 0 && props.reverseFillMask === true) {
return new RegExp('^' + fillCharEscaped + '*' + re)
}
else if (index === extractLast) {
return new RegExp(
'^' + re
+ '(' + (negateChar === '' ? '.' : negateChar) + '+)?'
+ (props.reverseFillMask === true ? '$' : fillCharEscaped + '*')
)
}
return new RegExp('^' + re)
});
computedMask = mask;
computedUnmask = val => {
const unmaskMatch = unmaskMatcher.exec(props.reverseFillMask === true ? val : val.slice(0, mask.length + 1));
if (unmaskMatch !== null) {
val = unmaskMatch.slice(1).join('');
}
const
extractMatch = [],
extractMatcherLength = extractMatcher.length;
for (let i = 0, str = val; i < extractMatcherLength; i++) {
const m = extractMatcher[ i ].exec(str);
if (m === null) {
break
}
str = str.slice(m.shift().length);
extractMatch.push(...m);
}
if (extractMatch.length !== 0) {
return extractMatch.join('')
}
return val
};
maskMarked = mask.map(v => (typeof v === 'string' ? v : MARKER)).join('');
maskReplaced = maskMarked.split(MARKER).join(fillChar);
}
function updateMaskValue (rawVal, updateMaskInternalsFlag, inputType) {
const
inp = inputRef.value,
end = inp.selectionEnd,
endReverse = inp.value.length - end,
unmasked = unmaskValue(rawVal);
// Update here so unmask uses the original fillChar
updateMaskInternalsFlag === true && updateMaskInternals();
const
preMasked = maskValue(unmasked),
masked = props.fillMask !== false
? fillWithMask(preMasked)
: preMasked,
changed = innerValue.value !== masked;
// We want to avoid "flickering" so we set value immediately
inp.value !== masked && (inp.value = masked);
changed === true && (innerValue.value = masked);
document.activeElement === inp && vue.nextTick(() => {
if (masked === maskReplaced) {
const cursor = props.reverseFillMask === true ? maskReplaced.length : 0;
inp.setSelectionRange(cursor, cursor, 'forward');
return
}
if (inputType === 'insertFromPaste' && props.reverseFillMask !== true) {
const maxEnd = inp.selectionEnd;
let cursor = end - 1;
// each non-marker char means we move once to right
for (let i = pastedTextStart; i <= cursor && i < maxEnd; i++) {
if (maskMarked[ i ] !== MARKER) {
cursor++;
}
}
moveCursor.right(inp, cursor);
return
}
if ([ 'deleteContentBackward', 'deleteContentForward' ].indexOf(inputType) > -1) {
const cursor = props.reverseFillMask === true
? (
end === 0
? (masked.length > preMasked.length ? 1 : 0)
: Math.max(0, masked.length - (masked === maskReplaced ? 0 : Math.min(preMasked.length, endReverse) + 1)) + 1
)
: end;
inp.setSelectionRange(cursor, cursor, 'forward');
return
}
if (props.reverseFillMask === true) {
if (changed === true) {
const cursor = Math.max(0, masked.length - (masked === maskReplaced ? 0 : Math.min(preMasked.length, endReverse + 1)));
if (cursor === 1 && end === 1) {
inp.setSelectionRange(cursor, cursor, 'forward');
}
else {
moveCursor.rightReverse(inp, cursor);
}
}
else {
const cursor = masked.length - endReverse;
inp.setSelectionRange(cursor, cursor, 'backward');
}
}
else {
if (changed === true) {
const cursor = Math.max(0, maskMarked.indexOf(MARKER), Math.min(preMasked.length, end) - 1);
moveCursor.right(inp, cursor);
}
else {
const cursor = end - 1;
moveCursor.right(inp, cursor);
}
}
});
const val = props.unmaskedValue === true
? unmaskValue(masked)
: masked;
String(props.modelValue) !== val && emitValue(val, true);
}
function moveCursorForPaste (inp, start, end) {
const preMasked = maskValue(unmaskValue(inp.value));
start = Math.max(0, maskMarked.indexOf(MARKER), Math.min(preMasked.length, start));
pastedTextStart = start;
inp.setSelectionRange(start, end, 'forward');
}
const moveCursor = {
left (inp, cursor) {
const noMarkBefore = maskMarked.slice(cursor - 1).indexOf(MARKER) === -1;
let i = Math.max(0, cursor - 1);
for (; i >= 0; i--) {
if (maskMarked[ i ] === MARKER) {
cursor = i;
noMarkBefore === true && cursor++;
break
}
}
if (
i < 0
&& maskMarked[ cursor ] !== void 0
&& maskMarked[ cursor ] !== MARKER
) {
return moveCursor.right(inp, 0)
}
cursor >= 0 && inp.setSelectionRange(cursor, cursor, 'backward');
},
right (inp, cursor) {
const limit = inp.value.length;
let i = Math.min(limit, cursor + 1);
for (; i <= limit; i++) {
if (maskMarked[ i ] === MARKER) {
cursor = i;
break
}
else if (maskMarked[ i - 1 ] === MARKER) {
cursor = i;
}
}
if (
i > limit
&& maskMarked[ cursor - 1 ] !== void 0
&& maskMarked[ cursor - 1 ] !== MARKER
) {
return moveCursor.left(inp, limit)
}
inp.setSelectionRange(cursor, cursor, 'forward');
},
leftReverse (inp, cursor) {
const
localMaskMarked = getPaddedMaskMarked(inp.value.length);
let i = Math.max(0, cursor - 1);
for (; i >= 0; i--) {
if (localMaskMarked[ i - 1 ] === MARKER) {
cursor = i;
break
}
else if (localMaskMarked[ i ] === MARKER) {
cursor = i;
if (i === 0) {
break
}
}
}
if (
i < 0
&& localMaskMarked[ cursor ] !== void 0
&& localMaskMarked[ cursor ] !== MARKER
) {
return moveCursor.rightReverse(inp, 0)
}
cursor >= 0 && inp.setSelectionRange(cursor, cursor, 'backward');
},
rightReverse (inp, cursor) {
const
limit = inp.value.length,
localMaskMarked = getPaddedMaskMarked(limit),
noMarkBefore = localMaskMarked.slice(0, cursor + 1).indexOf(MARKER) === -1;
let i = Math.min(limit, cursor + 1);
for (; i <= limit; i++) {
if (localMaskMarked[ i - 1 ] === MARKER) {
cursor = i;
cursor > 0 && noMarkBefore === true && cursor--;
break
}
}
if (
i > limit
&& localMaskMarked[ cursor - 1 ] !== void 0
&& localMaskMarked[ cursor - 1 ] !== MARKER
) {
return moveCursor.leftReverse(inp, limit)
}
inp.setSelectionRange(cursor, cursor, 'forward');
}
};
function onMaskedClick (e) {
emit('click', e);
selectionAnchor = void 0;
}
function onMaskedKeydown (e) {
emit('keydown', e);
if (shouldIgnoreKey(e) === true) {
return
}
const
inp = inputRef.value,
start = inp.selectionStart,
end = inp.selectionEnd;
if (!e.shiftKey) {
selectionAnchor = void 0;
}
if (e.keyCode === 37 || e.keyCode === 39) { // Left / Right
if (e.shiftKey && selectionAnchor === void 0) {
selectionAnchor = inp.selectionDirection === 'forward' ? start : end;
}
const fn = moveCursor[ (e.keyCode === 39 ? 'right' : 'left') + (props.reverseFillMask === true ? 'Reverse' : '') ];
e.preventDefault();
fn(inp, selectionAnchor === start ? end : start);
if (e.shiftKey) {
const cursor = inp.selectionStart;
inp.setSelectionRange(Math.min(selectionAnchor, cursor), Math.max(selectionAnchor, cursor), 'forward');
}
}
else if (
e.keyCode === 8 // Backspace
&& props.reverseFillMask !== true
&& start === end
) {
moveCursor.left(inp, start);
inp.setSelectionRange(inp.selectionStart, end, 'backward');
}
else if (
e.keyCode === 46 // Delete
&& props.reverseFillMask === true
&& start === end
) {
moveCursor.rightReverse(inp, end);
inp.setSelectionRange(start, inp.selectionEnd, 'forward');
}
}
function maskValue (val) {
if (val === void 0 || val === null || val === '') { return '' }
if (props.reverseFillMask === true) {
return maskValueReverse(val)
}
const mask = computedMask;
let valIndex = 0, output = '';
for (let maskIndex = 0; maskIndex < mask.length; maskIndex++) {
const
valChar = val[ valIndex ],
maskDef = mask[ maskIndex ];
if (typeof maskDef === 'string') {
output += maskDef;
valChar === maskDef && valIndex++;
}
else if (valChar !== void 0 && maskDef.regex.test(valChar)) {
output += maskDef.transform !== void 0
? maskDef.transform(valChar)
: valChar;
valIndex++;
}
else {
return output
}
}
return output
}
function maskValueReverse (val) {
const
mask = computedMask,
firstTokenIndex = maskMarked.indexOf(MARKER);
let valIndex = val.length - 1, output = '';
for (let maskIndex = mask.length - 1; maskIndex >= 0 && valIndex > -1; maskIndex--) {
const maskDef = mask[ maskIndex ];
let valChar = val[ valIndex ];
if (typeof maskDef === 'string') {
output = maskDef + output;
valChar === maskDef && valIndex--;
}
else if (valChar !== void 0 && maskDef.regex.test(valChar)) {
do {
output = (maskDef.transform !== void 0 ? maskDef.transform(valChar) : valChar) + output;
valIndex--;
valChar = val[ valIndex ];
// eslint-disable-next-line no-unmodified-loop-condition
} while (firstTokenIndex === maskIndex && valChar !== void 0 && maskDef.regex.test(valChar))
}
else {
return output
}
}
return output
}
function unmaskValue (val) {
return typeof val !== 'string' || computedUnmask === void 0
? (typeof val === 'number' ? computedUnmask('' + val) : val)
: computedUnmask(val)
}
function fillWithMask (val) {
if (maskReplaced.length - val.length <= 0) {
return val
}
return props.reverseFillMask === true && val.length !== 0
? maskReplaced.slice(0, -val.length) + val
: val + maskReplaced.slice(val.length)
}
return {
innerValue,
hasMask,
moveCursorForPaste,
updateMaskValue,
onMaskedKeydown,
onMaskedClick
}
}
const isJapanese = /[\u3000-\u303f\u3040-\u309f\u30a0-\u30ff\uff00-\uff9f\u4e00-\u9faf\u3400-\u4dbf]/;
const isChinese = /[\u4e00-\u9fff\u3400-\u4dbf\u{20000}-\u{2a6df}\u{2a700}-\u{2b73f}\u{2b740}-\u{2b81f}\u{2b820}-\u{2ceaf}\uf900-\ufaff\u3300-\u33ff\ufe30-\ufe4f\uf900-\ufaff\u{2f800}-\u{2fa1f}]/u;
const isKorean = /[\u3131-\u314e\u314f-\u3163\uac00-\ud7a3]/;
const isPlainText = /[a-z0-9_ -]$/i;
function useKeyComposition (onInput) {
return function onComposition (e) {
if (e.type === 'compositionend' || e.type === 'change') {
if (e.target.qComposing !== true) { return }
e.target.qComposing = false;
onInput(e);
}
else if (
e.type === 'compositionupdate'
&& e.target.qComposing !== true
&& typeof e.data === 'string'
) {
const isComposing = client.is.firefox === true
? isPlainText.test(e.data) === false
: isJapanese.test(e.data) === true || isChinese.test(e.data) === true || isKorean.test(e.data) === true;
if (isComposing === true) {
e.target.qComposing = true;
}
}
}
}
var QInput = createComponent({
name: 'QInput',
inheritAttrs: false,
props: {
...useFieldProps,
...useMaskProps,
...useFormProps,
modelValue: { required: false },
shadowText: String,
type: {
type: String,
default: 'text'
},
debounce: [ String, Number ],
autogrow: Boolean, // makes a textarea
inputClass: [ Array, String, Object ],
inputStyle: [ Array, String, Object ]
},
emits: [
...useFieldEmits,
'paste', 'change',
'keydown', 'click', 'animationend'
],
setup (props, { emit, attrs }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const temp = {};
let emitCachedValue = NaN, typedNumber, stopValueWatcher, emitTimer = null, emitValueFn;
const inputRef = vue.ref(null);
const nameProp = useFormInputNameAttr(props);
const {
innerValue,
hasMask,
moveCursorForPaste,
updateMaskValue,
onMaskedKeydown,
onMaskedClick
} = useMask(props, emit, emitValue, inputRef);
const formDomProps = useFileFormDomProps(props, /* type guard */ true);
const hasValue = vue.computed(() => fieldValueIsFilled(innerValue.value));
const onComposition = useKeyComposition(onInput);
const state = useFieldState();
const isTextarea = vue.computed(() =>
props.type === 'textarea' || props.autogrow === true
);
const isTypeText = vue.computed(() =>
isTextarea.value === true
|| [ 'text', 'search', 'url', 'tel', 'password' ].includes(props.type)
);
const onEvents = vue.computed(() => {
const evt = {
...state.splitAttrs.listeners.value,
onInput,
onPaste,
// Safari < 10.2 & UIWebView doesn't fire compositionend when
// switching focus before confirming composition choice
// this also fixes the issue where some browsers e.g. iOS Chrome
// fires "change" instead of "input" on autocomplete.
onChange,
onBlur: onFinishEditing,
onFocus: stop
};
evt.onCompositionstart = evt.onCompositionupdate = evt.onCompositionend = onComposition;
if (hasMask.value === true) {
evt.onKeydown = onMaskedKeydown;
// reset selection anchor on pointer selection
evt.onClick = onMaskedClick;
}
if (props.autogrow === true) {
evt.onAnimationend = onAnimationend;
}
return evt
});
const inputAttrs = vue.computed(() => {
const attrs = {
tabindex: 0,
'data-autofocus': props.autofocus === true || void 0,
rows: props.type === 'textarea' ? 6 : void 0,
'aria-label': props.label,
name: nameProp.value,
...state.splitAttrs.attributes.value,
id: state.targetUid.value,
maxlength: props.maxlength,
disabled: props.disable === true,
readonly: props.readonly === true
};
if (isTextarea.value === false) {
attrs.type = props.type;
}
if (props.autogrow === true) {
attrs.rows = 1;
}
return attrs
});
// some browsers lose the native input value
// so we need to reattach it dynamically
// (like type="password" <-> type="text"; see #12078)
vue.watch(() => props.type, () => {
if (inputRef.value) {
inputRef.value.value = props.modelValue;
}
});
vue.watch(() => props.modelValue, v => {
if (hasMask.value === true) {
if (stopValueWatcher === true) {
stopValueWatcher = false;
if (String(v) === emitCachedValue) {
return
}
}
updateMaskValue(v);
}
else if (innerValue.value !== v) {
innerValue.value = v;
if (
props.type === 'number'
&& temp.hasOwnProperty('value') === true
) {
if (typedNumber === true) {
typedNumber = false;
}
else {
delete temp.value;
}
}
}
// textarea only
props.autogrow === true && vue.nextTick(adjustHeight);
});
vue.watch(() => props.autogrow, val => {
// textarea only
if (val === true) {
vue.nextTick(adjustHeight);
}
// if it has a number of rows set respect it
else if (inputRef.value !== null && attrs.rows > 0) {
inputRef.value.style.height = 'auto';
}
});
vue.watch(() => props.dense, () => {
props.autogrow === true && vue.nextTick(adjustHeight);
});
function focus () {
addFocusFn(() => {
const el = document.activeElement;
if (
inputRef.value !== null
&& inputRef.value !== el
&& (el === null || el.id !== state.targetUid.value)
) {
inputRef.value.focus({ preventScroll: true });
}
});
}
function select () {
inputRef.value !== null && inputRef.value.select();
}
function onPaste (e) {
if (hasMask.value === true && props.reverseFillMask !== true) {
const inp = e.target;
moveCursorForPaste(inp, inp.selectionStart, inp.selectionEnd);
}
emit('paste', e);
}
function onInput (e) {
if (!e || !e.target) {
return
}
if (props.type === 'file') {
emit('update:modelValue', e.target.files);
return
}
const val = e.target.value;
if (e.target.qComposing === true) {
temp.value = val;
return
}
if (hasMask.value === true) {
updateMaskValue(val, false, e.inputType);
}
else {
emitValue(val);
if (isTypeText.value === true && e.target === document.activeElement) {
const { selectionStart, selectionEnd } = e.target;
if (selectionStart !== void 0 && selectionEnd !== void 0) {
vue.nextTick(() => {
if (e.target === document.activeElement && val.indexOf(e.target.value) === 0) {
e.target.setSelectionRange(selectionStart, selectionEnd);
}
});
}
}
}
// we need to trigger it immediately too,
// to avoid "flickering"
props.autogrow === true && adjustHeight();
}
function onAnimationend (e) {
emit('animationend', e);
adjustHeight();
}
function emitValue (val, stopWatcher) {
emitValueFn = () => {
emitTimer = null;
if (
props.type !== 'number'
&& temp.hasOwnProperty('value') === true
) {
delete temp.value;
}
if (props.modelValue !== val && emitCachedValue !== val) {
emitCachedValue = val;
stopWatcher === true && (stopValueWatcher = true);
emit('update:modelValue', val);
vue.nextTick(() => {
emitCachedValue === val && (emitCachedValue = NaN);
});
}
emitValueFn = void 0;
};
if (props.type === 'number') {
typedNumber = true;
temp.value = val;
}
if (props.debounce !== void 0) {
emitTimer !== null && clearTimeout(emitTimer);
temp.value = val;
emitTimer = setTimeout(emitValueFn, props.debounce);
}
else {
emitValueFn();
}
}
// textarea only
function adjustHeight () {
requestAnimationFrame(() => {
const inp = inputRef.value;
if (inp !== null) {
const parentStyle = inp.parentNode.style;
// chrome does not keep scroll #15498
const { scrollTop } = inp;
// chrome calculates a smaller scrollHeight when in a .column container
const { overflowY, maxHeight } = $q.platform.is.firefox === true
? {}
: window.getComputedStyle(inp);
// on firefox or if overflowY is specified as scroll #14263, #14344
// we don't touch overflow
// firefox is not so bad in the end
const changeOverflow = overflowY !== void 0 && overflowY !== 'scroll';
// reset height of textarea to a small size to detect the real height
// but keep the total control size the same
changeOverflow === true && (inp.style.overflowY = 'hidden');
parentStyle.marginBottom = (inp.scrollHeight - 1) + 'px';
inp.style.height = '1px';
inp.style.height = inp.scrollHeight + 'px';
// we should allow scrollbars only
// if there is maxHeight and content is taller than maxHeight
changeOverflow === true && (inp.style.overflowY = parseInt(maxHeight, 10) < inp.scrollHeight ? 'auto' : 'hidden');
parentStyle.marginBottom = '';
inp.scrollTop = scrollTop;
}
});
}
function onChange (e) {
onComposition(e);
if (emitTimer !== null) {
clearTimeout(emitTimer);
emitTimer = null;
}
emitValueFn !== void 0 && emitValueFn();
emit('change', e.target.value);
}
function onFinishEditing (e) {
e !== void 0 && stop(e);
if (emitTimer !== null) {
clearTimeout(emitTimer);
emitTimer = null;
}
emitValueFn !== void 0 && emitValueFn();
typedNumber = false;
stopValueWatcher = false;
delete temp.value;
// we need to use setTimeout instead of this.$nextTick
// to avoid a bug where focusout is not emitted for type date/time/week/...
props.type !== 'file' && setTimeout(() => {
if (inputRef.value !== null) {
inputRef.value.value = innerValue.value !== void 0 ? innerValue.value : '';
}
});
}
function getCurValue () {
return temp.hasOwnProperty('value') === true
? temp.value
: (innerValue.value !== void 0 ? innerValue.value : '')
}
vue.onBeforeUnmount(() => {
onFinishEditing();
});
vue.onMounted(() => {
// textarea only
props.autogrow === true && adjustHeight();
});
Object.assign(state, {
innerValue,
fieldClass: vue.computed(() =>
`q-${ isTextarea.value === true ? 'textarea' : 'input' }`
+ (props.autogrow === true ? ' q-textarea--autogrow' : '')
),
hasShadow: vue.computed(() =>
props.type !== 'file'
&& typeof props.shadowText === 'string'
&& props.shadowText.length !== 0
),
inputRef,
emitValue,
hasValue,
floatingLabel: vue.computed(() =>
(
hasValue.value === true
&& (props.type !== 'number' || isNaN(innerValue.value) === false)
)
|| fieldValueIsFilled(props.displayValue)
),
getControl: () => {
return vue.h(isTextarea.value === true ? 'textarea' : 'input', {
ref: inputRef,
class: [
'q-field__native q-placeholder',
props.inputClass
],
style: props.inputStyle,
...inputAttrs.value,
...onEvents.value,
...(
props.type !== 'file'
? { value: getCurValue() }
: formDomProps.value
)
})
},
getShadowControl: () => {
return vue.h('div', {
class: 'q-field__native q-field__shadow absolute-bottom no-pointer-events'
+ (isTextarea.value === true ? '' : ' text-no-wrap')
}, [
vue.h('span', { class: 'invisible' }, getCurValue()),
vue.h('span', props.shadowText)
])
}
});
const renderFn = useField(state);
// expose public methods
Object.assign(proxy, {
focus,
select,
getNativeElement: () => inputRef.value // deprecated
});
injectProp(proxy, 'nativeEl', () => inputRef.value);
return renderFn
}
});
const defaultCfg$1 = {
threshold: 0,
root: null,
rootMargin: '0px'
};
function update$3 (el, ctx, value) {
let handler, cfg, changed;
if (typeof value === 'function') {
handler = value;
cfg = defaultCfg$1;
changed = ctx.cfg === void 0;
}
else {
handler = value.handler;
cfg = Object.assign({}, defaultCfg$1, value.cfg);
changed = ctx.cfg === void 0 || isDeepEqual(ctx.cfg, cfg) === false;
}
if (ctx.handler !== handler) {
ctx.handler = handler;
}
if (changed === true) {
ctx.cfg = cfg;
ctx.observer !== void 0 && ctx.observer.unobserve(el);
ctx.observer = new IntersectionObserver(([ entry ]) => {
if (typeof ctx.handler === 'function') {
// if observed element is part of a vue transition
// then we need to be careful...
if (
entry.rootBounds === null
&& document.body.contains(el) === true
) {
ctx.observer.unobserve(el);
ctx.observer.observe(el);
return
}
const res = ctx.handler(entry, ctx.observer);
if (
res === false
|| (ctx.once === true && entry.isIntersecting === true)
) {
destroy$1(el);
}
}
}, cfg);
ctx.observer.observe(el);
}
}
function destroy$1 (el) {
const ctx = el.__qvisible;
if (ctx !== void 0) {
ctx.observer !== void 0 && ctx.observer.unobserve(el);
delete el.__qvisible;
}
}
var Intersection = createDirective({
name: 'intersection',
mounted (el, { modifiers, value }) {
const ctx = {
once: modifiers.once === true
};
update$3(el, ctx, value);
el.__qvisible = ctx;
},
updated (el, binding) {
const ctx = el.__qvisible;
ctx !== void 0 && update$3(el, ctx, binding.value);
},
beforeUnmount: destroy$1
}
);
var QIntersection = createComponent({
name: 'QIntersection',
props: {
tag: {
type: String,
default: 'div'
},
once: Boolean,
transition: String,
transitionDuration: {
type: [ String, Number ],
default: 300
},
ssrPrerender: Boolean,
margin: String,
threshold: [ Number, Array ],
root: {
default: null
},
disable: Boolean,
onVisibility: Function
},
setup (props, { slots, emit }) {
const showing = vue.ref(isRuntimeSsrPreHydration.value === true ? props.ssrPrerender : false);
const intersectionProps = vue.computed(() => (
props.root !== void 0 || props.margin !== void 0 || props.threshold !== void 0
? {
handler: trigger,
cfg: {
root: props.root,
rootMargin: props.margin,
threshold: props.threshold
}
}
: trigger
));
const hasDirective = vue.computed(() =>
props.disable !== true
&& (isRuntimeSsrPreHydration.value !== true || props.once !== true || props.ssrPrerender !== true)
);
const directives = vue.computed(() => {
// if hasDirective.value === true
return [ [
Intersection,
intersectionProps.value,
void 0,
{ once: props.once }
] ]
});
const transitionStyle = vue.computed(
() => `--q-transition-duration: ${ props.transitionDuration }ms`
);
function trigger (entry) {
if (showing.value !== entry.isIntersecting) {
showing.value = entry.isIntersecting;
props.onVisibility !== void 0 && emit('visibility', showing.value);
}
}
function getContent () {
if (showing.value === true) {
return [ vue.h('div', { key: 'content', style: transitionStyle.value }, hSlot(slots.default)) ]
}
if (slots.hidden !== void 0) {
return [ vue.h('div', { key: 'hidden', style: transitionStyle.value }, slots.hidden()) ]
}
}
return () => {
const child = props.transition
? [
vue.h(vue.Transition, {
name: 'q-transition--' + props.transition
}, getContent)
]
: getContent();
return hDir(
props.tag,
{ class: 'q-intersection' },
child,
'main',
hasDirective.value,
() => directives.value
)
}
}
});
var QList = createComponent({
name: 'QList',
props: {
...useDarkProps,
bordered: Boolean,
dense: Boolean,
separator: Boolean,
padding: Boolean,
tag: {
type: String,
default: 'div'
}
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const classes = vue.computed(() =>
'q-list'
+ (props.bordered === true ? ' q-list--bordered' : '')
+ (props.dense === true ? ' q-list--dense' : '')
+ (props.separator === true ? ' q-list--separator' : '')
+ (isDark.value === true ? ' q-list--dark' : '')
+ (props.padding === true ? ' q-list--padding' : '')
);
return () => vue.h(props.tag, { class: classes.value }, hSlot(slots.default))
}
});
// PGDOWN, LEFT, DOWN, PGUP, RIGHT, UP
const keyCodes$1 = [ 34, 37, 40, 33, 39, 38 ];
const commonPropsName = Object.keys(useCircularCommonProps);
var QKnob = createComponent({
name: 'QKnob',
props: {
...useFormProps,
...useCircularCommonProps,
modelValue: {
type: Number,
required: true
},
innerMin: Number,
innerMax: Number,
step: {
type: Number,
default: 1,
validator: v => v >= 0
},
tabindex: {
type: [ Number, String ],
default: 0
},
disable: Boolean,
readonly: Boolean
},
emits: [ 'update:modelValue', 'change', 'dragValue' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const model = vue.ref(props.modelValue);
const dragging = vue.ref(false);
const innerMin = vue.computed(() => (
isNaN(props.innerMin) === true || props.innerMin < props.min
? props.min
: props.innerMin
));
const innerMax = vue.computed(() => (
isNaN(props.innerMax) === true || props.innerMax > props.max
? props.max
: props.innerMax
));
let centerPosition;
function normalizeModel () {
model.value = props.modelValue === null
? innerMin.value
: between(props.modelValue, innerMin.value, innerMax.value);
updateValue(true);
}
vue.watch(
() => `${ props.modelValue }|${ innerMin.value }|${ innerMax.value }`,
normalizeModel
);
normalizeModel();
const editable = vue.computed(() => props.disable === false && props.readonly === false);
const classes = vue.computed(() =>
'q-knob non-selectable' + (
editable.value === true
? ' q-knob--editable'
: (props.disable === true ? ' disabled' : '')
)
);
const decimals = vue.computed(() => (String(props.step).trim().split('.')[ 1 ] || '').length);
const step = vue.computed(() => (props.step === 0 ? 1 : props.step));
const instantFeedback = vue.computed(() => props.instantFeedback === true || dragging.value === true);
const onEvents = $q.platform.is.mobile === true
? vue.computed(() => (editable.value === true ? { onClick } : {}))
: vue.computed(() => (
editable.value === true
? {
onMousedown,
onClick,
onKeydown,
onKeyup
}
: {}
));
const attrs = vue.computed(() => (
editable.value === true
? { tabindex: props.tabindex }
: { [ `aria-${ props.disable === true ? 'disabled' : 'readonly' }` ]: 'true' }
));
const circularProps = vue.computed(() => {
const agg = {};
commonPropsName.forEach(name => {
agg[ name ] = props[ name ];
});
return agg
});
function pan (event) {
if (event.isFinal) {
updatePosition(event.evt, true);
dragging.value = false;
}
else if (event.isFirst) {
updateCenterPosition();
dragging.value = true;
updatePosition(event.evt);
}
else {
updatePosition(event.evt);
}
}
const directives = vue.computed(() => {
return [ [
TouchPan,
pan,
void 0,
{ prevent: true, stop: true, mouse: true }
] ]
});
function updateCenterPosition () {
const { top, left, width, height } = proxy.$el.getBoundingClientRect();
centerPosition = {
top: top + height / 2,
left: left + width / 2
};
}
function onMousedown (evt) {
updateCenterPosition();
updatePosition(evt);
}
function onClick (evt) {
updateCenterPosition();
updatePosition(evt, true);
}
function onKeydown (evt) {
if (!keyCodes$1.includes(evt.keyCode)) {
return
}
stopAndPrevent(evt);
const
stepVal = ([ 34, 33 ].includes(evt.keyCode) ? 10 : 1) * step.value,
offset = [ 34, 37, 40 ].includes(evt.keyCode) ? -stepVal : stepVal;
model.value = between(
parseFloat((model.value + offset).toFixed(decimals.value)),
innerMin.value,
innerMax.value
);
updateValue();
}
function updatePosition (evt, change) {
const
pos = position(evt),
height = Math.abs(pos.top - centerPosition.top),
distance = Math.sqrt(
height ** 2
+ Math.abs(pos.left - centerPosition.left) ** 2
);
let angle = Math.asin(height / distance) * (180 / Math.PI);
if (pos.top < centerPosition.top) {
angle = centerPosition.left < pos.left ? 90 - angle : 270 + angle;
}
else {
angle = centerPosition.left < pos.left ? angle + 90 : 270 - angle;
}
if ($q.lang.rtl === true) {
angle = normalizeToInterval(-angle - props.angle, 0, 360);
}
else if (props.angle) {
angle = normalizeToInterval(angle - props.angle, 0, 360);
}
if (props.reverse === true) {
angle = 360 - angle;
}
let newModel = props.min + (angle / 360) * (props.max - props.min);
if (step.value !== 0) {
const modulo = newModel % step.value;
newModel = newModel - modulo
+ (Math.abs(modulo) >= step.value / 2 ? (modulo < 0 ? -1 : 1) * step.value : 0);
newModel = parseFloat(newModel.toFixed(decimals.value));
}
newModel = between(newModel, innerMin.value, innerMax.value);
emit('dragValue', newModel);
if (model.value !== newModel) {
model.value = newModel;
}
updateValue(change);
}
function onKeyup (evt) {
if (keyCodes$1.includes(evt.keyCode)) {
updateValue(true);
}
}
function updateValue (change) {
props.modelValue !== model.value && emit('update:modelValue', model.value);
change === true && emit('change', model.value);
}
const formAttrs = useFormAttrs(props);
function getNameInput () {
return vue.h('input', formAttrs.value)
}
return () => {
const data = {
class: classes.value,
role: 'slider',
'aria-valuemin': innerMin.value,
'aria-valuemax': innerMax.value,
'aria-valuenow': props.modelValue,
...attrs.value,
...circularProps.value,
value: model.value,
instantFeedback: instantFeedback.value,
...onEvents.value
};
const child = {
default: slots.default
};
if (editable.value === true && props.name !== void 0) {
child.internal = getNameInput;
}
return hDir(
QCircularProgress,
data,
child,
'knob',
editable.value,
() => directives.value
)
}
}
});
const { passive: passive$2 } = listenOpts;
const axisValues = [ 'both', 'horizontal', 'vertical' ];
var QScrollObserver = createComponent({
name: 'QScrollObserver',
props: {
axis: {
type: String,
validator: v => axisValues.includes(v),
default: 'vertical'
},
debounce: [ String, Number ],
scrollTarget: {
default: void 0
}
},
emits: [ 'scroll' ],
setup (props, { emit }) {
const scroll = {
position: {
top: 0,
left: 0
},
direction: 'down',
directionChanged: false,
delta: {
top: 0,
left: 0
},
inflectionPoint: {
top: 0,
left: 0
}
};
let clearTimer = null, localScrollTarget, parentEl;
vue.watch(() => props.scrollTarget, () => {
unconfigureScrollTarget();
configureScrollTarget();
});
function emitEvent () {
clearTimer !== null && clearTimer();
const top = Math.max(0, getVerticalScrollPosition(localScrollTarget));
const left = getHorizontalScrollPosition(localScrollTarget);
const delta = {
top: top - scroll.position.top,
left: left - scroll.position.left
};
if (
(props.axis === 'vertical' && delta.top === 0)
|| (props.axis === 'horizontal' && delta.left === 0)
) {
return
}
const curDir = Math.abs(delta.top) >= Math.abs(delta.left)
? (delta.top < 0 ? 'up' : 'down')
: (delta.left < 0 ? 'left' : 'right');
scroll.position = { top, left };
scroll.directionChanged = scroll.direction !== curDir;
scroll.delta = delta;
if (scroll.directionChanged === true) {
scroll.direction = curDir;
scroll.inflectionPoint = scroll.position;
}
emit('scroll', { ...scroll });
}
function configureScrollTarget () {
localScrollTarget = getScrollTarget(parentEl, props.scrollTarget);
localScrollTarget.addEventListener('scroll', trigger, passive$2);
trigger(true);
}
function unconfigureScrollTarget () {
if (localScrollTarget !== void 0) {
localScrollTarget.removeEventListener('scroll', trigger, passive$2);
localScrollTarget = void 0;
}
}
function trigger (immediately) {
if (immediately === true || props.debounce === 0 || props.debounce === '0') {
emitEvent();
}
else if (clearTimer === null) {
const [ timer, fn ] = props.debounce
? [ setTimeout(emitEvent, props.debounce), clearTimeout ]
: [ requestAnimationFrame(emitEvent), cancelAnimationFrame ];
clearTimer = () => {
fn(timer);
clearTimer = null;
};
}
}
const { proxy } = vue.getCurrentInstance();
vue.watch(() => proxy.$q.lang.rtl, emitEvent);
vue.onMounted(() => {
parentEl = proxy.$el.parentNode;
configureScrollTarget();
});
vue.onBeforeUnmount(() => {
clearTimer !== null && clearTimer();
unconfigureScrollTarget();
});
// expose public methods
Object.assign(proxy, {
trigger,
getPosition: () => scroll
});
return noop
}
});
var QLayout = createComponent({
name: 'QLayout',
props: {
container: Boolean,
view: {
type: String,
default: 'hhh lpr fff',
validator: v => /^(h|l)h(h|r) lpr (f|l)f(f|r)$/.test(v.toLowerCase())
},
onScroll: Function,
onScrollHeight: Function,
onResize: Function
},
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const rootRef = vue.ref(null);
// page related
const height = vue.ref($q.screen.height);
const width = vue.ref(props.container === true ? 0 : $q.screen.width);
const scroll = vue.ref({ position: 0, direction: 'down', inflectionPoint: 0 });
// container only prop
const containerHeight = vue.ref(0);
const scrollbarWidth = vue.ref(isRuntimeSsrPreHydration.value === true ? 0 : getScrollbarWidth());
const classes = vue.computed(() =>
'q-layout q-layout--'
+ (props.container === true ? 'containerized' : 'standard')
);
const style = vue.computed(() => (
props.container === false
? { minHeight: $q.screen.height + 'px' }
: null
));
// used by container only
const targetStyle = vue.computed(() => (
scrollbarWidth.value !== 0
? { [ $q.lang.rtl === true ? 'left' : 'right' ]: `${ scrollbarWidth.value }px` }
: null
));
const targetChildStyle = vue.computed(() => (
scrollbarWidth.value !== 0
? {
[ $q.lang.rtl === true ? 'right' : 'left' ]: 0,
[ $q.lang.rtl === true ? 'left' : 'right' ]: `-${ scrollbarWidth.value }px`,
width: `calc(100% + ${ scrollbarWidth.value }px)`
}
: null
));
function onPageScroll (data) {
if (props.container === true || document.qScrollPrevented !== true) {
const info = {
position: data.position.top,
direction: data.direction,
directionChanged: data.directionChanged,
inflectionPoint: data.inflectionPoint.top,
delta: data.delta.top
};
scroll.value = info;
props.onScroll !== void 0 && emit('scroll', info);
}
}
function onPageResize (data) {
const { height: newHeight, width: newWidth } = data;
let resized = false;
if (height.value !== newHeight) {
resized = true;
height.value = newHeight;
props.onScrollHeight !== void 0 && emit('scrollHeight', newHeight);
updateScrollbarWidth();
}
if (width.value !== newWidth) {
resized = true;
width.value = newWidth;
}
if (resized === true && props.onResize !== void 0) {
emit('resize', data);
}
}
function onContainerResize ({ height }) {
if (containerHeight.value !== height) {
containerHeight.value = height;
updateScrollbarWidth();
}
}
function updateScrollbarWidth () {
if (props.container === true) {
const width = height.value > containerHeight.value
? getScrollbarWidth()
: 0;
if (scrollbarWidth.value !== width) {
scrollbarWidth.value = width;
}
}
}
let animateTimer = null;
const $layout = {
instances: {},
view: vue.computed(() => props.view),
isContainer: vue.computed(() => props.container),
rootRef,
height,
containerHeight,
scrollbarWidth,
totalWidth: vue.computed(() => width.value + scrollbarWidth.value),
rows: vue.computed(() => {
const rows = props.view.toLowerCase().split(' ');
return {
top: rows[ 0 ].split(''),
middle: rows[ 1 ].split(''),
bottom: rows[ 2 ].split('')
}
}),
header: vue.reactive({ size: 0, offset: 0, space: false }),
right: vue.reactive({ size: 300, offset: 0, space: false }),
footer: vue.reactive({ size: 0, offset: 0, space: false }),
left: vue.reactive({ size: 300, offset: 0, space: false }),
scroll,
animate () {
if (animateTimer !== null) {
clearTimeout(animateTimer);
}
else {
document.body.classList.add('q-body--layout-animate');
}
animateTimer = setTimeout(() => {
animateTimer = null;
document.body.classList.remove('q-body--layout-animate');
}, 155);
},
update (part, prop, val) {
$layout[ part ][ prop ] = val;
}
};
vue.provide(layoutKey, $layout);
// prevent scrollbar flicker while resizing window height
// if no page scrollbar is already present
if (getScrollbarWidth() > 0) {
let timer = null;
const el = document.body;
function restoreScrollbar () {
timer = null;
el.classList.remove('hide-scrollbar');
}
function hideScrollbar () {
if (timer === null) {
// if it has no scrollbar then there's nothing to do
if (el.scrollHeight > $q.screen.height) {
return
}
el.classList.add('hide-scrollbar');
}
else {
clearTimeout(timer);
}
timer = setTimeout(restoreScrollbar, 300);
}
function updateScrollEvent (action) {
if (timer !== null && action === 'remove') {
clearTimeout(timer);
restoreScrollbar();
}
window[ `${ action }EventListener` ]('resize', hideScrollbar);
}
vue.watch(
() => (props.container !== true ? 'add' : 'remove'),
updateScrollEvent
);
props.container !== true && updateScrollEvent('add');
vue.onUnmounted(() => {
updateScrollEvent('remove');
});
}
return () => {
const content = hMergeSlot(slots.default, [
vue.h(QScrollObserver, { onScroll: onPageScroll }),
vue.h(QResizeObserver, { onResize: onPageResize })
]);
const layout = vue.h('div', {
class: classes.value,
style: style.value,
ref: props.container === true ? void 0 : rootRef,
tabindex: -1
}, content);
if (props.container === true) {
return vue.h('div', {
class: 'q-layout-container overflow-hidden',
ref: rootRef
}, [
vue.h(QResizeObserver, { onResize: onContainerResize }),
vue.h('div', {
class: 'absolute-full',
style: targetStyle.value
}, [
vue.h('div', {
class: 'scroll',
style: targetChildStyle.value
}, [ layout ])
])
])
}
return layout
}
}
});
const separatorValues = [ 'horizontal', 'vertical', 'cell', 'none' ];
var QMarkupTable = createComponent({
name: 'QMarkupTable',
props: {
...useDarkProps,
dense: Boolean,
flat: Boolean,
bordered: Boolean,
square: Boolean,
wrapCells: Boolean,
separator: {
type: String,
default: 'horizontal',
validator: v => separatorValues.includes(v)
}
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const classes = vue.computed(() =>
'q-markup-table q-table__container q-table__card'
+ ` q-table--${ props.separator }-separator`
+ (isDark.value === true ? ' q-table--dark q-table__card--dark q-dark' : '')
+ (props.dense === true ? ' q-table--dense' : '')
+ (props.flat === true ? ' q-table--flat' : '')
+ (props.bordered === true ? ' q-table--bordered' : '')
+ (props.square === true ? ' q-table--square' : '')
+ (props.wrapCells === false ? ' q-table--no-wrap' : '')
);
return () => vue.h('div', {
class: classes.value
}, [
vue.h('table', { class: 'q-table' }, hSlot(slots.default))
])
}
});
var QNoSsr = createComponent({
name: 'QNoSsr',
props: {
tag: {
type: String,
default: 'div'
},
placeholder: String
},
setup (props, { slots }) {
const canRender = useCanRender();
return () => {
const data = {};
if (canRender.value === true) {
const node = hSlot(slots.default);
return node === void 0
? node
: (node.length > 1 ? vue.h(props.tag, data, node) : node[ 0 ])
}
data.class = 'q-no-ssr-placeholder';
const node = hSlot(slots.placeholder);
if (node !== void 0) {
return node.length > 1
? vue.h(props.tag, data, node)
: node[ 0 ]
}
if (props.placeholder !== void 0) {
return vue.h(props.tag, data, props.placeholder)
}
}
}
});
const svg$m = vue.h('svg', {
key: 'svg',
class: 'q-radio__bg absolute non-selectable',
viewBox: '0 0 24 24'
}, [
vue.h('path', {
d: 'M12,22a10,10 0 0 1 -10,-10a10,10 0 0 1 10,-10a10,10 0 0 1 10,10a10,10 0 0 1 -10,10m0,-22a12,12 0 0 0 -12,12a12,12 0 0 0 12,12a12,12 0 0 0 12,-12a12,12 0 0 0 -12,-12'
}),
vue.h('path', {
class: 'q-radio__check',
d: 'M12,6a6,6 0 0 0 -6,6a6,6 0 0 0 6,6a6,6 0 0 0 6,-6a6,6 0 0 0 -6,-6'
})
]);
var QRadio = createComponent({
name: 'QRadio',
props: {
...useDarkProps,
...useSizeProps,
...useFormProps,
modelValue: { required: true },
val: { required: true },
label: String,
leftLabel: Boolean,
checkedIcon: String,
uncheckedIcon: String,
color: String,
keepColor: Boolean,
dense: Boolean,
disable: Boolean,
tabindex: [ String, Number ]
},
emits: [ 'update:modelValue' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const isDark = useDark(props, proxy.$q);
const sizeStyle = useSize(props, optionSizes);
const rootRef = vue.ref(null);
const { refocusTargetEl, refocusTarget } = useRefocusTarget(props, rootRef);
const isTrue = vue.computed(() => vue.toRaw(props.modelValue) === vue.toRaw(props.val));
const classes = vue.computed(() =>
'q-radio cursor-pointer no-outline row inline no-wrap items-center'
+ (props.disable === true ? ' disabled' : '')
+ (isDark.value === true ? ' q-radio--dark' : '')
+ (props.dense === true ? ' q-radio--dense' : '')
+ (props.leftLabel === true ? ' reverse' : '')
);
const innerClass = vue.computed(() => {
const color = props.color !== void 0 && (
props.keepColor === true
|| isTrue.value === true
)
? ` text-${ props.color }`
: '';
return 'q-radio__inner relative-position '
+ `q-radio__inner--${ isTrue.value === true ? 'truthy' : 'falsy' }${ color }`
});
const icon = vue.computed(() =>
(isTrue.value === true
? props.checkedIcon
: props.uncheckedIcon
) || null
);
const tabindex = vue.computed(() => (
props.disable === true ? -1 : props.tabindex || 0
));
const formAttrs = vue.computed(() => {
const prop = { type: 'radio' };
props.name !== void 0 && Object.assign(prop, {
// see https://vuejs.org/guide/extras/render-function.html#creating-vnodes (.prop)
'.checked': isTrue.value === true,
'^checked': isTrue.value === true ? 'checked' : void 0,
name: props.name,
value: props.val
});
return prop
});
const injectFormInput = useFormInject(formAttrs);
function onClick (e) {
if (e !== void 0) {
stopAndPrevent(e);
refocusTarget(e);
}
if (props.disable !== true && isTrue.value !== true) {
emit('update:modelValue', props.val, e);
}
}
function onKeydown (e) {
if (e.keyCode === 13 || e.keyCode === 32) {
stopAndPrevent(e);
}
}
function onKeyup (e) {
if (e.keyCode === 13 || e.keyCode === 32) {
onClick(e);
}
}
// expose public methods
Object.assign(proxy, { set: onClick });
return () => {
const content = icon.value !== null
? [
vue.h('div', {
key: 'icon',
class: 'q-radio__icon-container absolute-full flex flex-center no-wrap'
}, [
vue.h(QIcon, {
class: 'q-radio__icon',
name: icon.value
})
])
]
: [ svg$m ];
props.disable !== true && injectFormInput(
content,
'unshift',
' q-radio__native q-ma-none q-pa-none'
);
const child = [
vue.h('div', {
class: innerClass.value,
style: sizeStyle.value,
'aria-hidden': 'true'
}, content)
];
if (refocusTargetEl.value !== null) {
child.push(refocusTargetEl.value);
}
const label = props.label !== void 0
? hMergeSlot(slots.default, [ props.label ])
: hSlot(slots.default);
label !== void 0 && child.push(
vue.h('div', {
class: 'q-radio__label q-anchor--skip'
}, label)
);
return vue.h('div', {
ref: rootRef,
class: classes.value,
tabindex: tabindex.value,
role: 'radio',
'aria-label': props.label,
'aria-checked': isTrue.value === true ? 'true' : 'false',
'aria-disabled': props.disable === true ? 'true' : void 0,
onClick,
onKeydown,
onKeyup
}, child)
}
}
});
var QToggle = createComponent({
name: 'QToggle',
props: {
...useCheckboxProps,
icon: String,
iconColor: String
},
emits: useCheckboxEmits,
setup (props) {
function getInner (isTrue, isIndeterminate) {
const icon = vue.computed(() =>
(isTrue.value === true
? props.checkedIcon
: (isIndeterminate.value === true ? props.indeterminateIcon : props.uncheckedIcon)
) || props.icon
);
const color = vue.computed(() => (isTrue.value === true ? props.iconColor : null));
return () => [
vue.h('div', { class: 'q-toggle__track' }),
vue.h('div', {
class: 'q-toggle__thumb absolute flex flex-center no-wrap'
}, icon.value !== void 0
? [
vue.h(QIcon, {
name: icon.value,
color: color.value
})
]
: void 0
)
]
}
return useCheckbox('toggle', getInner)
}
});
const components$1 = {
radio: QRadio,
checkbox: QCheckbox,
toggle: QToggle
};
const typeValues = Object.keys(components$1);
var QOptionGroup = createComponent({
name: 'QOptionGroup',
props: {
...useDarkProps,
modelValue: {
required: true
},
options: {
type: Array,
validator: opts => opts.every(opt => 'value' in opt && 'label' in opt)
},
name: String,
type: {
default: 'radio',
validator: v => typeValues.includes(v)
},
color: String,
keepColor: Boolean,
dense: Boolean,
size: String,
leftLabel: Boolean,
inline: Boolean,
disable: Boolean
},
emits: [ 'update:modelValue' ],
setup (props, { emit, slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const arrayModel = Array.isArray(props.modelValue);
if (props.type === 'radio') {
if (arrayModel === true) {
console.error('q-option-group: model should not be array');
}
}
else if (arrayModel === false) {
console.error('q-option-group: model should be array in your case');
}
const isDark = useDark(props, $q);
const component = vue.computed(() => components$1[ props.type ]);
const classes = vue.computed(() =>
'q-option-group q-gutter-x-sm'
+ (props.inline === true ? ' q-option-group--inline' : '')
);
const attrs = vue.computed(() => {
const attrs = { role: 'group' };
if (props.type === 'radio') {
attrs.role = 'radiogroup';
if (props.disable === true) {
attrs[ 'aria-disabled' ] = 'true';
}
}
return attrs
});
function onUpdateModelValue (value) {
emit('update:modelValue', value);
}
return () => vue.h('div', {
class: classes.value,
...attrs.value
}, props.options.map((opt, i) => {
// TODO: (Qv3) Make the 'opt' a separate property instead of
// the whole scope for consistency and flexibility
// (e.g. { opt } instead of opt)
const child = slots[ 'label-' + i ] !== void 0
? () => slots[ 'label-' + i ](opt)
: (
slots.label !== void 0
? () => slots.label(opt)
: void 0
);
return vue.h('div', [
vue.h(component.value, {
modelValue: props.modelValue,
val: opt.value,
name: opt.name === void 0 ? props.name : opt.name,
disable: props.disable || opt.disable,
label: child === void 0 ? opt.label : null,
leftLabel: opt.leftLabel === void 0 ? props.leftLabel : opt.leftLabel,
color: opt.color === void 0 ? props.color : opt.color,
checkedIcon: opt.checkedIcon,
uncheckedIcon: opt.uncheckedIcon,
dark: opt.dark || isDark.value,
size: opt.size === void 0 ? props.size : opt.size,
dense: props.dense,
keepColor: opt.keepColor === void 0 ? props.keepColor : opt.keepColor,
'onUpdate:modelValue': onUpdateModelValue
}, child)
])
}))
}
});
var QPage = createComponent({
name: 'QPage',
props: {
padding: Boolean,
styleFn: Function
},
setup (props, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QPage needs to be a deep child of QLayout');
return emptyRenderFn
}
const $pageContainer = vue.inject(pageContainerKey, emptyRenderFn);
if ($pageContainer === emptyRenderFn) {
console.error('QPage needs to be child of QPageContainer');
return emptyRenderFn
}
const style = vue.computed(() => {
const offset
= ($layout.header.space === true ? $layout.header.size : 0)
+ ($layout.footer.space === true ? $layout.footer.size : 0);
if (typeof props.styleFn === 'function') {
const height = $layout.isContainer.value === true
? $layout.containerHeight.value
: $q.screen.height;
return props.styleFn(offset, height)
}
return {
minHeight: $layout.isContainer.value === true
? ($layout.containerHeight.value - offset) + 'px'
: (
$q.screen.height === 0
? (offset !== 0 ? `calc(100vh - ${ offset }px)` : '100vh')
: ($q.screen.height - offset) + 'px'
)
}
});
const classes = vue.computed(() =>
`q-page${ props.padding === true ? ' q-layout-padding' : '' }`
);
return () => vue.h('main', {
class: classes.value,
style: style.value
}, hSlot(slots.default))
}
});
var QPageContainer = createComponent({
name: 'QPageContainer',
setup (_, { slots }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QPageContainer needs to be child of QLayout');
return emptyRenderFn
}
vue.provide(pageContainerKey, true);
const style = vue.computed(() => {
const css = {};
if ($layout.header.space === true) {
css.paddingTop = `${ $layout.header.size }px`;
}
if ($layout.right.space === true) {
css[ `padding${ $q.lang.rtl === true ? 'Left' : 'Right' }` ] = `${ $layout.right.size }px`;
}
if ($layout.footer.space === true) {
css.paddingBottom = `${ $layout.footer.size }px`;
}
if ($layout.left.space === true) {
css[ `padding${ $q.lang.rtl === true ? 'Right' : 'Left' }` ] = `${ $layout.left.size }px`;
}
return css
});
return () => vue.h('div', {
class: 'q-page-container',
style: style.value
}, hSlot(slots.default))
}
});
const usePageStickyProps = {
position: {
type: String,
default: 'bottom-right',
validator: v => [
'top-right', 'top-left',
'bottom-right', 'bottom-left',
'top', 'right', 'bottom', 'left'
].includes(v)
},
offset: {
type: Array,
validator: v => v.length === 2
},
expand: Boolean
};
function usePageSticky () {
const { props, proxy: { $q } } = vue.getCurrentInstance();
const $layout = vue.inject(layoutKey, emptyRenderFn);
if ($layout === emptyRenderFn) {
console.error('QPageSticky needs to be child of QLayout');
return emptyRenderFn
}
const attach = vue.computed(() => {
const pos = props.position;
return {
top: pos.indexOf('top') > -1,
right: pos.indexOf('right') > -1,
bottom: pos.indexOf('bottom') > -1,
left: pos.indexOf('left') > -1,
vertical: pos === 'top' || pos === 'bottom',
horizontal: pos === 'left' || pos === 'right'
}
});
const top = vue.computed(() => $layout.header.offset);
const right = vue.computed(() => $layout.right.offset);
const bottom = vue.computed(() => $layout.footer.offset);
const left = vue.computed(() => $layout.left.offset);
const style = vue.computed(() => {
let posX = 0, posY = 0;
const side = attach.value;
const dir = $q.lang.rtl === true ? -1 : 1;
if (side.top === true && top.value !== 0) {
posY = `${ top.value }px`;
}
else if (side.bottom === true && bottom.value !== 0) {
posY = `${ -bottom.value }px`;
}
if (side.left === true && left.value !== 0) {
posX = `${ dir * left.value }px`;
}
else if (side.right === true && right.value !== 0) {
posX = `${ -dir * right.value }px`;
}
const css = { transform: `translate(${ posX }, ${ posY })` };
if (props.offset) {
css.margin = `${ props.offset[ 1 ] }px ${ props.offset[ 0 ] }px`;
}
if (side.vertical === true) {
if (left.value !== 0) {
css[ $q.lang.rtl === true ? 'right' : 'left' ] = `${ left.value }px`;
}
if (right.value !== 0) {
css[ $q.lang.rtl === true ? 'left' : 'right' ] = `${ right.value }px`;
}
}
else if (side.horizontal === true) {
if (top.value !== 0) {
css.top = `${ top.value }px`;
}
if (bottom.value !== 0) {
css.bottom = `${ bottom.value }px`;
}
}
return css
});
const classes = vue.computed(() =>
`q-page-sticky row flex-center fixed-${ props.position }`
+ ` q-page-sticky--${ props.expand === true ? 'expand' : 'shrink' }`
);
function getStickyContent (slots) {
const content = hSlot(slots.default);
return vue.h('div', {
class: classes.value,
style: style.value
},
props.expand === true
? content
: [ vue.h('div', content) ]
)
}
return {
$layout,
getStickyContent
}
}
var QPageScroller = createComponent({
name: 'QPageScroller',
props: {
...usePageStickyProps,
scrollOffset: {
type: Number,
default: 1000
},
reverse: Boolean,
duration: {
type: Number,
default: 300
},
offset: {
default: () => [ 18, 18 ]
}
},
emits: [ 'click' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const { $layout, getStickyContent } = usePageSticky();
const rootRef = vue.ref(null);
let heightWatcher;
const scrollHeight = vue.computed(() => $layout.height.value - (
$layout.isContainer.value === true
? $layout.containerHeight.value
: $q.screen.height
));
function isVisible () {
return props.reverse === true
? scrollHeight.value - $layout.scroll.value.position > props.scrollOffset
: $layout.scroll.value.position > props.scrollOffset
}
const showing = vue.ref(isVisible());
function updateVisibility () {
const newVal = isVisible();
if (showing.value !== newVal) {
showing.value = newVal;
}
}
function updateReverse () {
if (props.reverse === true) {
if (heightWatcher === void 0) {
heightWatcher = vue.watch(scrollHeight, updateVisibility);
}
}
else {
cleanup();
}
}
vue.watch($layout.scroll, updateVisibility);
vue.watch(() => props.reverse, updateReverse);
function cleanup () {
if (heightWatcher !== void 0) {
heightWatcher();
heightWatcher = void 0;
}
}
function onClick (e) {
const target = getScrollTarget(
$layout.isContainer.value === true
? rootRef.value
: $layout.rootRef.value
);
setVerticalScrollPosition(
target,
props.reverse === true ? $layout.height.value : 0,
props.duration
);
emit('click', e);
}
function getContent () {
return showing.value === true
? vue.h('div', {
ref: rootRef,
class: 'q-page-scroller',
onClick
}, getStickyContent(slots))
: null
}
updateReverse();
vue.onBeforeUnmount(cleanup);
return () => vue.h(
vue.Transition,
{ name: 'q-transition--fade' },
getContent
)
}
});
var QPageSticky = createComponent({
name: 'QPageSticky',
props: usePageStickyProps,
setup (_, { slots }) {
const { getStickyContent } = usePageSticky();
return () => getStickyContent(slots)
}
});
function getBool (val, otherwise) {
return [ true, false ].includes(val)
? val
: otherwise
}
var QPagination = createComponent({
name: 'QPagination',
props: {
...useDarkProps,
modelValue: {
type: Number,
required: true
},
min: {
type: [ Number, String ],
default: 1
},
max: {
type: [ Number, String ],
required: true
},
maxPages: {
type: [ Number, String ],
default: 0,
validator: v => (
(typeof v === 'string' ? parseInt(v, 10) : v) >= 0
)
},
inputStyle: [ Array, String, Object ],
inputClass: [ Array, String, Object ],
size: String,
disable: Boolean,
input: Boolean,
iconPrev: String,
iconNext: String,
iconFirst: String,
iconLast: String,
toFn: Function,
boundaryLinks: {
type: Boolean,
default: null
},
boundaryNumbers: {
type: Boolean,
default: null
},
directionLinks: {
type: Boolean,
default: null
},
ellipses: {
type: Boolean,
default: null
},
ripple: {
type: [ Boolean, Object ],
default: null
},
round: Boolean,
rounded: Boolean,
flat: Boolean,
outline: Boolean,
unelevated: Boolean,
push: Boolean,
glossy: Boolean,
color: {
type: String,
default: 'primary'
},
textColor: String,
activeDesign: {
type: String,
default: '',
values: v => v === '' || btnDesignOptions.includes(v)
},
activeColor: String,
activeTextColor: String,
gutter: String,
padding: {
type: String,
default: '3px 2px'
}
},
emits: [ 'update:modelValue' ],
setup (props, { emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const minProp = vue.computed(() => parseInt(props.min, 10));
const maxProp = vue.computed(() => parseInt(props.max, 10));
const maxPagesProp = vue.computed(() => parseInt(props.maxPages, 10));
const inputPlaceholder = vue.computed(() => model.value + ' / ' + maxProp.value);
const boundaryLinksProp = vue.computed(() => getBool(props.boundaryLinks, props.input));
const boundaryNumbersProp = vue.computed(() => getBool(props.boundaryNumbers, !props.input));
const directionLinksProp = vue.computed(() => getBool(props.directionLinks, props.input));
const ellipsesProp = vue.computed(() => getBool(props.ellipses, !props.input));
const newPage = vue.ref(null);
const model = vue.computed({
get: () => props.modelValue,
set: val => {
val = parseInt(val, 10);
if (props.disable || isNaN(val)) {
return
}
const value = between(val, minProp.value, maxProp.value);
if (props.modelValue !== value) {
emit('update:modelValue', value);
}
}
});
vue.watch(() => `${ minProp.value }|${ maxProp.value }`, () => {
model.value = props.modelValue;
});
const classes = vue.computed(() =>
'q-pagination row no-wrap items-center'
+ (props.disable === true ? ' disabled' : '')
);
const gutterProp = vue.computed(() => (
props.gutter in btnPadding
? `${ btnPadding[ props.gutter ] }px`
: props.gutter || null
));
const gutterStyle = vue.computed(() => (
gutterProp.value !== null
? `--q-pagination-gutter-parent:-${ gutterProp.value };--q-pagination-gutter-child:${ gutterProp.value }`
: null
));
const icons = vue.computed(() => {
const ico = [
props.iconFirst || $q.iconSet.pagination.first,
props.iconPrev || $q.iconSet.pagination.prev,
props.iconNext || $q.iconSet.pagination.next,
props.iconLast || $q.iconSet.pagination.last
];
return $q.lang.rtl === true ? ico.reverse() : ico
});
const attrs = vue.computed(() => ({
'aria-disabled': props.disable === true ? 'true' : 'false',
role: 'navigation'
}));
const btnDesignProp = vue.computed(() => getBtnDesign(props, 'flat'));
const btnProps = vue.computed(() => ({
[ btnDesignProp.value ]: true,
round: props.round,
rounded: props.rounded,
padding: props.padding,
color: props.color,
textColor: props.textColor,
size: props.size,
ripple: props.ripple !== null
? props.ripple
: true
}));
const btnActiveDesignProp = vue.computed(() => {
// we also reset non-active design
const acc = { [ btnDesignProp.value ]: false };
if (props.activeDesign !== '') {
acc[ props.activeDesign ] = true;
}
return acc
});
const activeBtnProps = vue.computed(() => ({
...btnActiveDesignProp.value,
color: props.activeColor || props.color,
textColor: props.activeTextColor || props.textColor
}));
const btnConfig = vue.computed(() => {
let maxPages = Math.max(
maxPagesProp.value,
1 + (ellipsesProp.value ? 2 : 0) + (boundaryNumbersProp.value ? 2 : 0)
);
const acc = {
pgFrom: minProp.value,
pgTo: maxProp.value,
ellipsesStart: false,
ellipsesEnd: false,
boundaryStart: false,
boundaryEnd: false,
marginalStyle: {
minWidth: `${ Math.max(2, String(maxProp.value).length) }em`
}
};
if (maxPagesProp.value && maxPages < (maxProp.value - minProp.value + 1)) {
maxPages = 1 + Math.floor(maxPages / 2) * 2;
acc.pgFrom = Math.max(minProp.value, Math.min(maxProp.value - maxPages + 1, props.modelValue - Math.floor(maxPages / 2)));
acc.pgTo = Math.min(maxProp.value, acc.pgFrom + maxPages - 1);
if (boundaryNumbersProp.value) {
acc.boundaryStart = true;
acc.pgFrom++;
}
if (ellipsesProp.value && acc.pgFrom > (minProp.value + (boundaryNumbersProp.value ? 1 : 0))) {
acc.ellipsesStart = true;
acc.pgFrom++;
}
if (boundaryNumbersProp.value) {
acc.boundaryEnd = true;
acc.pgTo--;
}
if (ellipsesProp.value && acc.pgTo < (maxProp.value - (boundaryNumbersProp.value ? 1 : 0))) {
acc.ellipsesEnd = true;
acc.pgTo--;
}
}
return acc
});
function set (value) {
model.value = value;
}
function setByOffset (offset) {
model.value = model.value + offset;
}
const inputEvents = vue.computed(() => {
function updateModel () {
model.value = newPage.value;
newPage.value = null;
}
return {
'onUpdate:modelValue': val => { newPage.value = val; },
onKeyup: e => { isKeyCode(e, 13) === true && updateModel(); },
onBlur: updateModel
}
});
function getBtn (cfg, page, active) {
const data = {
'aria-label': page,
'aria-current': 'false',
...btnProps.value,
...cfg
};
if (active === true) {
Object.assign(data, {
'aria-current': 'true',
...activeBtnProps.value
});
}
if (page !== void 0) {
if (props.toFn !== void 0) {
data.to = props.toFn(page);
}
else {
data.onClick = () => { set(page); };
}
}
return vue.h(QBtn, data)
}
// expose public methods
Object.assign(proxy, { set, setByOffset });
return () => {
const contentStart = [];
const contentEnd = [];
let contentMiddle;
if (boundaryLinksProp.value === true) {
contentStart.push(
getBtn({
key: 'bls',
disable: props.disable || props.modelValue <= minProp.value,
icon: icons.value[ 0 ]
}, minProp.value)
);
contentEnd.unshift(
getBtn({
key: 'ble',
disable: props.disable || props.modelValue >= maxProp.value,
icon: icons.value[ 3 ]
}, maxProp.value)
);
}
if (directionLinksProp.value === true) {
contentStart.push(
getBtn({
key: 'bdp',
disable: props.disable || props.modelValue <= minProp.value,
icon: icons.value[ 1 ]
}, props.modelValue - 1)
);
contentEnd.unshift(
getBtn({
key: 'bdn',
disable: props.disable || props.modelValue >= maxProp.value,
icon: icons.value[ 2 ]
}, props.modelValue + 1)
);
}
if (props.input !== true) { // has buttons instead of inputbox
contentMiddle = [];
const { pgFrom, pgTo, marginalStyle: style } = btnConfig.value;
if (btnConfig.value.boundaryStart === true) {
const active = minProp.value === props.modelValue;
contentStart.push(
getBtn({
key: 'bns',
style,
disable: props.disable,
label: minProp.value
}, minProp.value, active)
);
}
if (btnConfig.value.boundaryEnd === true) {
const active = maxProp.value === props.modelValue;
contentEnd.unshift(
getBtn({
key: 'bne',
style,
disable: props.disable,
label: maxProp.value
}, maxProp.value, active)
);
}
if (btnConfig.value.ellipsesStart === true) {
contentStart.push(
getBtn({
key: 'bes',
style,
disable: props.disable,
label: '…',
ripple: false
}, pgFrom - 1)
);
}
if (btnConfig.value.ellipsesEnd === true) {
contentEnd.unshift(
getBtn({
key: 'bee',
style,
disable: props.disable,
label: '…',
ripple: false
}, pgTo + 1)
);
}
for (let i = pgFrom; i <= pgTo; i++) {
contentMiddle.push(
getBtn({
key: `bpg${ i }`,
style,
disable: props.disable,
label: i
}, i, i === props.modelValue)
);
}
}
return vue.h('div', {
class: classes.value,
...attrs.value
}, [
vue.h('div', {
class: 'q-pagination__content row no-wrap items-center',
style: gutterStyle.value
}, [
...contentStart,
props.input === true
? vue.h(QInput, {
class: 'inline',
style: { width: `${ inputPlaceholder.value.length / 1.5 }em` },
type: 'number',
dense: true,
value: newPage.value,
disable: props.disable,
dark: isDark.value,
borderless: true,
inputClass: props.inputClass,
inputStyle: props.inputStyle,
placeholder: inputPlaceholder.value,
min: minProp.value,
max: maxProp.value,
...inputEvents.value
})
: vue.h('div', {
class: 'q-pagination__middle row justify-center'
}, contentMiddle),
...contentEnd
])
])
}
}
});
function frameDebounce (fn) {
let wait = false, frame, callArgs;
function debounced (/* ...args */) {
callArgs = arguments;
if (wait === true) { return }
wait = true;
frame = requestAnimationFrame(() => {
fn.apply(this, callArgs);
callArgs = void 0;
wait = false;
});
}
debounced.cancel = () => {
window.cancelAnimationFrame(frame);
wait = false;
};
return debounced
}
const { passive: passive$1 } = listenOpts;
var QParallax = createComponent({
name: 'QParallax',
props: {
src: String,
height: {
type: Number,
default: 500
},
speed: {
type: Number,
default: 1,
validator: v => v >= 0 && v <= 1
},
scrollTarget: {
default: void 0
},
onScroll: Function
},
setup (props, { slots, emit }) {
const percentScrolled = vue.ref(0);
const rootRef = vue.ref(null);
const mediaParentRef = vue.ref(null);
const mediaRef = vue.ref(null);
let isWorking, mediaEl, mediaHeight, resizeHandler, observer, localScrollTarget;
vue.watch(() => props.height, () => {
isWorking === true && updatePos();
});
vue.watch(() => props.scrollTarget, () => {
if (isWorking === true) {
stop();
start();
}
});
let update = percentage => {
percentScrolled.value = percentage;
props.onScroll !== void 0 && emit('scroll', percentage);
};
function updatePos () {
let containerTop, containerHeight, containerBottom;
if (localScrollTarget === window) {
containerTop = 0;
containerBottom = containerHeight = window.innerHeight;
}
else {
containerTop = offset(localScrollTarget).top;
containerHeight = height(localScrollTarget);
containerBottom = containerTop + containerHeight;
}
const top = offset(rootRef.value).top;
const bottom = top + props.height;
if (observer !== void 0 || (bottom > containerTop && top < containerBottom)) {
const percent = (containerBottom - top) / (props.height + containerHeight);
setPos((mediaHeight - props.height) * percent * props.speed);
update(percent);
}
}
let setPos = offset => {
// apply it immediately without any delay
mediaEl.style.transform = `translate3d(-50%,${ Math.round(offset) }px,0)`;
};
function onResize () {
mediaHeight = mediaEl.naturalHeight || mediaEl.videoHeight || height(mediaEl);
isWorking === true && updatePos();
}
function start () {
isWorking = true;
localScrollTarget = getScrollTarget(rootRef.value, props.scrollTarget);
localScrollTarget.addEventListener('scroll', updatePos, passive$1);
window.addEventListener('resize', resizeHandler, passive$1);
updatePos();
}
function stop () {
if (isWorking === true) {
isWorking = false;
localScrollTarget.removeEventListener('scroll', updatePos, passive$1);
window.removeEventListener('resize', resizeHandler, passive$1);
localScrollTarget = void 0;
setPos.cancel();
update.cancel();
resizeHandler.cancel();
}
}
vue.onMounted(() => {
setPos = frameDebounce(setPos);
update = frameDebounce(update);
resizeHandler = frameDebounce(onResize);
mediaEl = slots.media !== void 0
? mediaParentRef.value.children[ 0 ]
: mediaRef.value;
mediaEl.onload = mediaEl.onloadstart = mediaEl.loadedmetadata = onResize;
onResize();
mediaEl.style.display = 'initial';
if (window.IntersectionObserver !== void 0) {
observer = new IntersectionObserver(entries => {
const fn = entries[ 0 ].isIntersecting === true ? start : stop;
fn();
});
observer.observe(rootRef.value);
}
else {
start();
}
});
vue.onBeforeUnmount(() => {
stop();
observer !== void 0 && observer.disconnect();
mediaEl.onload = mediaEl.onloadstart = mediaEl.loadedmetadata = null;
});
return () => {
return vue.h('div', {
ref: rootRef,
class: 'q-parallax',
style: { height: `${ props.height }px` }
}, [
vue.h('div', {
ref: mediaParentRef,
class: 'q-parallax__media absolute-full'
}, slots.media !== void 0 ? slots.media() : [
vue.h('img', {
ref: mediaRef,
src: props.src
})
]),
vue.h(
'div',
{ class: 'q-parallax__content absolute-full column flex-center' },
slots.content !== void 0
? slots.content({ percentScrolled: percentScrolled.value })
: hSlot(slots.default)
)
])
}
}
});
// adapted from https://stackoverflow.com/a/40294058
function cloneDeep (data, hash = new WeakMap()) {
if (Object(data) !== data) return data
if (hash.has(data)) return hash.get(data)
const result = data instanceof Date
? new Date(data)
: (data instanceof RegExp
? new RegExp(data.source, data.flags)
: (data instanceof Set
? new Set()
: (data instanceof Map
? new Map()
: (typeof data.constructor !== 'function'
? Object.create(null)
: (data.prototype !== void 0 && typeof data.prototype.constructor === 'function'
? data
: new data.constructor()
)
)
)
)
);
if (typeof data.constructor === 'function' && typeof data.valueOf === 'function') {
const val = data.valueOf();
if (Object(val) !== val) {
const result = new data.constructor(val);
hash.set(data, result);
return result
}
}
hash.set(data, result);
if (data instanceof Set) {
data.forEach(val => {
result.add(cloneDeep(val, hash));
});
}
else if (data instanceof Map) {
data.forEach((val, key) => {
result.set(key, cloneDeep(val, hash));
});
}
return Object.assign(
result,
...Object.keys(data).map(key => ({ [ key ]: cloneDeep(data[ key ], hash) }))
)
}
var QPopupEdit = createComponent({
name: 'QPopupEdit',
props: {
modelValue: {
required: true
},
title: String,
buttons: Boolean,
labelSet: String,
labelCancel: String,
color: {
type: String,
default: 'primary'
},
validate: {
type: Function,
default: () => true
},
autoSave: Boolean,
/* menu props overrides */
cover: {
type: Boolean,
default: true
},
/* end of menu props */
disable: Boolean
},
emits: [
'update:modelValue', 'save', 'cancel',
'beforeShow', 'show', 'beforeHide', 'hide'
],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const menuRef = vue.ref(null);
const initialValue = vue.ref('');
const currentModel = vue.ref('');
let validated = false;
const scope = vue.computed(() => {
return injectProp({
initialValue: initialValue.value,
validate: props.validate,
set,
cancel,
updatePosition
}, 'value', () => currentModel.value, val => { currentModel.value = val; })
});
function set () {
if (props.validate(currentModel.value) === false) {
return
}
if (hasModelChanged() === true) {
emit('save', currentModel.value, initialValue.value);
emit('update:modelValue', currentModel.value);
}
closeMenu();
}
function cancel () {
if (hasModelChanged() === true) {
emit('cancel', currentModel.value, initialValue.value);
}
closeMenu();
}
function updatePosition () {
vue.nextTick(() => {
menuRef.value.updatePosition();
});
}
function hasModelChanged () {
return isDeepEqual(currentModel.value, initialValue.value) === false
}
function closeMenu () {
validated = true;
menuRef.value.hide();
}
function onBeforeShow () {
validated = false;
initialValue.value = cloneDeep(props.modelValue);
currentModel.value = cloneDeep(props.modelValue);
emit('beforeShow');
}
function onShow () {
emit('show');
}
function onBeforeHide () {
if (validated === false && hasModelChanged() === true) {
if (props.autoSave === true && props.validate(currentModel.value) === true) {
emit('save', currentModel.value, initialValue.value);
emit('update:modelValue', currentModel.value);
}
else {
emit('cancel', currentModel.value, initialValue.value);
}
}
emit('beforeHide');
}
function onHide () {
emit('hide');
}
function getContent () {
const child = slots.default !== void 0
? [].concat(slots.default(scope.value))
: [];
props.title && child.unshift(
vue.h('div', { class: 'q-dialog__title q-mt-sm q-mb-sm' }, props.title)
);
props.buttons === true && child.push(
vue.h('div', { class: 'q-popup-edit__buttons row justify-center no-wrap' }, [
vue.h(QBtn, {
flat: true,
color: props.color,
label: props.labelCancel || $q.lang.label.cancel,
onClick: cancel
}),
vue.h(QBtn, {
flat: true,
color: props.color,
label: props.labelSet || $q.lang.label.set,
onClick: set
})
])
);
return child
}
// expose public methods
Object.assign(proxy, {
set,
cancel,
show (e) { menuRef.value !== null && menuRef.value.show(e); },
hide (e) { menuRef.value !== null && menuRef.value.hide(e); },
updatePosition
});
return () => {
if (props.disable === true) { return }
return vue.h(QMenu, {
ref: menuRef,
class: 'q-popup-edit',
cover: props.cover,
onBeforeShow,
onShow,
onBeforeHide,
onHide,
onEscapeKey: cancel
}, getContent)
}
}
});
var QPopupProxy = createComponent({
name: 'QPopupProxy',
props: {
...useAnchorProps,
breakpoint: {
type: [ String, Number ],
default: 450
}
},
emits: [ 'show', 'hide' ],
setup (props, { slots, emit, attrs }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const showing = vue.ref(false);
const popupRef = vue.ref(null);
const breakpoint = vue.computed(() => parseInt(props.breakpoint, 10));
const { canShow } = useAnchor({ showing });
function getType () {
return $q.screen.width < breakpoint.value || $q.screen.height < breakpoint.value
? 'dialog'
: 'menu'
}
const type = vue.ref(getType());
const popupProps = vue.computed(() => (
type.value === 'menu' ? { maxHeight: '99vh' } : {})
);
vue.watch(() => getType(), val => {
if (showing.value !== true) {
type.value = val;
}
});
function onShow (evt) {
showing.value = true;
emit('show', evt);
}
function onHide (evt) {
showing.value = false;
type.value = getType();
emit('hide', evt);
}
// expose public methods
Object.assign(proxy, {
show (evt) { canShow(evt) === true && popupRef.value.show(evt); },
hide (evt) { popupRef.value.hide(evt); },
toggle (evt) { popupRef.value.toggle(evt); }
});
injectProp(proxy, 'currentComponent', () => ({
type: type.value,
ref: popupRef.value
}));
return () => {
const data = {
ref: popupRef,
...popupProps.value,
...attrs,
onShow,
onHide
};
let component;
if (type.value === 'dialog') {
component = QDialog;
}
else {
component = QMenu;
Object.assign(data, {
target: props.target,
contextMenu: props.contextMenu,
noParentEvent: true,
separateClosePopup: true
});
}
return vue.h(component, data, slots.default)
}
}
});
const defaultSizes = {
xs: 2,
sm: 4,
md: 6,
lg: 10,
xl: 14
};
function width (val, reverse, $q) {
return {
transform: reverse === true
? `translateX(${ $q.lang.rtl === true ? '-' : '' }100%) scale3d(${ -val },1,1)`
: `scale3d(${ val },1,1)`
}
}
var QLinearProgress = createComponent({
name: 'QLinearProgress',
props: {
...useDarkProps,
...useSizeProps,
value: {
type: Number,
default: 0
},
buffer: Number,
color: String,
trackColor: String,
reverse: Boolean,
stripe: Boolean,
indeterminate: Boolean,
query: Boolean,
rounded: Boolean,
animationSpeed: {
type: [ String, Number ],
default: 2100
},
instantFeedback: Boolean
},
setup (props, { slots }) {
const { proxy } = vue.getCurrentInstance();
const isDark = useDark(props, proxy.$q);
const sizeStyle = useSize(props, defaultSizes);
const motion = vue.computed(() => props.indeterminate === true || props.query === true);
const widthReverse = vue.computed(() => props.reverse !== props.query);
const style = vue.computed(() => ({
...(sizeStyle.value !== null ? sizeStyle.value : {}),
'--q-linear-progress-speed': `${ props.animationSpeed }ms`
}));
const classes = vue.computed(() =>
'q-linear-progress'
+ (props.color !== void 0 ? ` text-${ props.color }` : '')
+ (props.reverse === true || props.query === true ? ' q-linear-progress--reverse' : '')
+ (props.rounded === true ? ' rounded-borders' : '')
);
const trackStyle = vue.computed(() => width(props.buffer !== void 0 ? props.buffer : 1, widthReverse.value, proxy.$q));
const transitionSuffix = vue.computed(() => `with${ props.instantFeedback === true ? 'out' : '' }-transition`);
const trackClass = vue.computed(() =>
'q-linear-progress__track absolute-full'
+ ` q-linear-progress__track--${ transitionSuffix.value }`
+ ` q-linear-progress__track--${ isDark.value === true ? 'dark' : 'light' }`
+ (props.trackColor !== void 0 ? ` bg-${ props.trackColor }` : '')
);
const modelStyle = vue.computed(() => width(motion.value === true ? 1 : props.value, widthReverse.value, proxy.$q));
const modelClass = vue.computed(() =>
'q-linear-progress__model absolute-full'
+ ` q-linear-progress__model--${ transitionSuffix.value }`
+ ` q-linear-progress__model--${ motion.value === true ? 'in' : '' }determinate`
);
const stripeStyle = vue.computed(() => ({ width: `${ props.value * 100 }%` }));
const stripeClass = vue.computed(() =>
`q-linear-progress__stripe absolute-${ props.reverse === true ? 'right' : 'left' }`
+ ` q-linear-progress__stripe--${ transitionSuffix.value }`
);
return () => {
const child = [
vue.h('div', {
class: trackClass.value,
style: trackStyle.value
}),
vue.h('div', {
class: modelClass.value,
style: modelStyle.value
})
];
props.stripe === true && motion.value === false && child.push(
vue.h('div', {
class: stripeClass.value,
style: stripeStyle.value
})
);
return vue.h('div', {
class: classes.value,
style: style.value,
role: 'progressbar',
'aria-valuemin': 0,
'aria-valuemax': 1,
'aria-valuenow': props.indeterminate === true
? void 0
: props.value
}, hMergeSlot(slots.default, child))
}
}
});
const
PULLER_HEIGHT = 40,
OFFSET_TOP = 20;
var QPullToRefresh = createComponent({
name: 'QPullToRefresh',
props: {
color: String,
bgColor: String,
icon: String,
noMouse: Boolean,
disable: Boolean,
scrollTarget: {
default: void 0
}
},
emits: [ 'refresh' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const state = vue.ref('pull');
const pullRatio = vue.ref(0);
const pulling = vue.ref(false);
const pullPosition = vue.ref(-PULLER_HEIGHT);
const animating = vue.ref(false);
const positionCSS = vue.ref({});
const style = vue.computed(() => ({
opacity: pullRatio.value,
transform: `translateY(${ pullPosition.value }px) rotate(${ pullRatio.value * 360 }deg)`
}));
const classes = vue.computed(() =>
'q-pull-to-refresh__puller row flex-center'
+ (animating.value === true ? ' q-pull-to-refresh__puller--animating' : '')
+ (props.bgColor !== void 0 ? ` bg-${ props.bgColor }` : '')
);
function pull (event) {
if (event.isFinal === true) {
if (pulling.value === true) {
pulling.value = false;
if (state.value === 'pulled') {
state.value = 'refreshing';
animateTo({ pos: OFFSET_TOP });
trigger();
}
else if (state.value === 'pull') {
animateTo({ pos: -PULLER_HEIGHT, ratio: 0 });
}
}
return
}
if (animating.value === true || state.value === 'refreshing') {
return false
}
if (event.isFirst === true) {
if (getVerticalScrollPosition(localScrollTarget) !== 0 || event.direction !== 'down') {
if (pulling.value === true) {
pulling.value = false;
state.value = 'pull';
animateTo({ pos: -PULLER_HEIGHT, ratio: 0 });
}
return false
}
pulling.value = true;
const { top, left } = $el.getBoundingClientRect();
positionCSS.value = {
top: top + 'px',
left: left + 'px',
width: window.getComputedStyle($el).getPropertyValue('width')
};
}
prevent(event.evt);
const distance = Math.min(140, Math.max(0, event.distance.y));
pullPosition.value = distance - PULLER_HEIGHT;
pullRatio.value = between(distance / (OFFSET_TOP + PULLER_HEIGHT), 0, 1);
const newState = pullPosition.value > OFFSET_TOP ? 'pulled' : 'pull';
if (state.value !== newState) {
state.value = newState;
}
}
const directives = vue.computed(() => {
// if props.disable === false
const modifiers = { down: true };
if (props.noMouse !== true) {
modifiers.mouse = true;
}
return [ [
TouchPan,
pull,
void 0,
modifiers
] ]
});
const contentClass = vue.computed(() =>
`q-pull-to-refresh__content${ pulling.value === true ? ' no-pointer-events' : '' }`
);
function trigger () {
emit('refresh', () => {
animateTo({ pos: -PULLER_HEIGHT, ratio: 0 }, () => {
state.value = 'pull';
});
});
}
let $el, localScrollTarget, timer = null;
function animateTo ({ pos, ratio }, done) {
animating.value = true;
pullPosition.value = pos;
if (ratio !== void 0) {
pullRatio.value = ratio;
}
timer !== null && clearTimeout(timer);
timer = setTimeout(() => {
timer = null;
animating.value = false;
done && done();
}, 300);
}
function updateScrollTarget () {
localScrollTarget = getScrollTarget($el, props.scrollTarget);
}
vue.watch(() => props.scrollTarget, updateScrollTarget);
vue.onMounted(() => {
$el = proxy.$el;
updateScrollTarget();
});
vue.onBeforeUnmount(() => {
timer !== null && clearTimeout(timer);
});
// expose public methods
Object.assign(proxy, { trigger, updateScrollTarget });
return () => {
const child = [
vue.h('div', { class: contentClass.value }, hSlot(slots.default)),
vue.h('div', {
class: 'q-pull-to-refresh__puller-container fixed row flex-center no-pointer-events z-top',
style: positionCSS.value
}, [
vue.h('div', {
class: classes.value,
style: style.value
}, [
state.value !== 'refreshing'
? vue.h(QIcon, {
name: props.icon || $q.iconSet.pullToRefresh.icon,
color: props.color,
size: '32px'
})
: vue.h(QSpinner, {
size: '24px',
color: props.color
})
])
])
];
return hDir(
'div',
{ class: 'q-pull-to-refresh' },
child,
'main',
props.disable === false,
() => directives.value
)
}
}
});
const dragType = {
MIN: 0,
RANGE: 1,
MAX: 2
};
var QRange = createComponent({
name: 'QRange',
props: {
...useSliderProps,
modelValue: {
type: Object,
default: () => ({ min: null, max: null }),
validator: v => 'min' in v && 'max' in v
},
dragRange: Boolean,
dragOnlyRange: Boolean,
leftLabelColor: String,
leftLabelTextColor: String,
rightLabelColor: String,
rightLabelTextColor: String,
leftLabelValue: [ String, Number ],
rightLabelValue: [ String, Number ],
leftThumbColor: String,
rightThumbColor: String
},
emits: useSliderEmits,
setup (props, { emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const { state, methods } = useSlider({
updateValue, updatePosition, getDragging,
formAttrs: vue.computed(() => ({
type: 'hidden',
name: props.name,
value: `${ props.modelValue.min }|${ props.modelValue.max }`
}))
});
const rootRef = vue.ref(null);
const curMinRatio = vue.ref(0);
const curMaxRatio = vue.ref(0);
const model = vue.ref({ min: 0, max: 0 });
function normalizeModel () {
model.value.min = props.modelValue.min === null
? state.innerMin.value
: between(props.modelValue.min, state.innerMin.value, state.innerMax.value);
model.value.max = props.modelValue.max === null
? state.innerMax.value
: between(props.modelValue.max, state.innerMin.value, state.innerMax.value);
}
vue.watch(
() => `${ props.modelValue.min }|${ props.modelValue.max }|${ state.innerMin.value }|${ state.innerMax.value }`,
normalizeModel
);
normalizeModel();
const modelMinRatio = vue.computed(() => methods.convertModelToRatio(model.value.min));
const modelMaxRatio = vue.computed(() => methods.convertModelToRatio(model.value.max));
const ratioMin = vue.computed(() => (
state.active.value === true ? curMinRatio.value : modelMinRatio.value
));
const ratioMax = vue.computed(() => (
state.active.value === true ? curMaxRatio.value : modelMaxRatio.value
));
const selectionBarStyle = vue.computed(() => {
const acc = {
[ state.positionProp.value ]: `${ 100 * ratioMin.value }%`,
[ state.sizeProp.value ]: `${ 100 * (ratioMax.value - ratioMin.value) }%`
};
if (props.selectionImg !== void 0) {
acc.backgroundImage = `url(${ props.selectionImg }) !important`;
}
return acc
});
const trackContainerEvents = vue.computed(() => {
if (state.editable.value !== true) {
return {}
}
if ($q.platform.is.mobile === true) {
return { onClick: methods.onMobileClick }
}
const evt = { onMousedown: methods.onActivate };
if (props.dragRange === true || props.dragOnlyRange === true) {
Object.assign(evt, {
onFocus: () => { state.focus.value = 'both'; },
onBlur: methods.onBlur,
onKeydown,
onKeyup: methods.onKeyup
});
}
return evt
});
function getEvents (side) {
return $q.platform.is.mobile !== true && state.editable.value === true && props.dragOnlyRange !== true
? {
onFocus: () => { state.focus.value = side; },
onBlur: methods.onBlur,
onKeydown,
onKeyup: methods.onKeyup
}
: {}
}
const thumbTabindex = vue.computed(() => (props.dragOnlyRange !== true ? state.tabindex.value : null));
const trackContainerTabindex = vue.computed(() => (
$q.platform.is.mobile !== true && (props.dragRange || props.dragOnlyRange === true)
? state.tabindex.value
: null
));
const minThumbRef = vue.ref(null);
const minEvents = vue.computed(() => getEvents('min'));
const getMinThumb = methods.getThumbRenderFn({
focusValue: 'min',
getNodeData: () => ({
ref: minThumbRef,
key: 'tmin',
...minEvents.value,
tabindex: thumbTabindex.value
}),
ratio: ratioMin,
label: vue.computed(() => (
props.leftLabelValue !== void 0
? props.leftLabelValue
: model.value.min
)),
thumbColor: vue.computed(() => props.leftThumbColor || props.thumbColor || props.color),
labelColor: vue.computed(() => props.leftLabelColor || props.labelColor),
labelTextColor: vue.computed(() => props.leftLabelTextColor || props.labelTextColor)
});
const maxEvents = vue.computed(() => getEvents('max'));
const getMaxThumb = methods.getThumbRenderFn({
focusValue: 'max',
getNodeData: () => ({
...maxEvents.value,
key: 'tmax',
tabindex: thumbTabindex.value
}),
ratio: ratioMax,
label: vue.computed(() => (
props.rightLabelValue !== void 0
? props.rightLabelValue
: model.value.max
)),
thumbColor: vue.computed(() => props.rightThumbColor || props.thumbColor || props.color),
labelColor: vue.computed(() => props.rightLabelColor || props.labelColor),
labelTextColor: vue.computed(() => props.rightLabelTextColor || props.labelTextColor)
});
function updateValue (change) {
if (model.value.min !== props.modelValue.min || model.value.max !== props.modelValue.max) {
emit('update:modelValue', { ...model.value });
}
change === true && emit('change', { ...model.value });
}
function getDragging (event) {
const
{ left, top, width, height } = rootRef.value.getBoundingClientRect(),
sensitivity = props.dragOnlyRange === true
? 0
: (props.vertical === true
? minThumbRef.value.offsetHeight / (2 * height)
: minThumbRef.value.offsetWidth / (2 * width)
);
const dragging = {
left,
top,
width,
height,
valueMin: model.value.min,
valueMax: model.value.max,
ratioMin: modelMinRatio.value,
ratioMax: modelMaxRatio.value
};
const ratio = methods.getDraggingRatio(event, dragging);
if (props.dragOnlyRange !== true && ratio < dragging.ratioMin + sensitivity) {
dragging.type = dragType.MIN;
}
else if (props.dragOnlyRange === true || ratio < dragging.ratioMax - sensitivity) {
if (props.dragRange === true || props.dragOnlyRange === true) {
dragging.type = dragType.RANGE;
Object.assign(dragging, {
offsetRatio: ratio,
offsetModel: methods.convertRatioToModel(ratio),
rangeValue: dragging.valueMax - dragging.valueMin,
rangeRatio: dragging.ratioMax - dragging.ratioMin
});
}
else {
dragging.type = dragging.ratioMax - ratio < ratio - dragging.ratioMin
? dragType.MAX
: dragType.MIN;
}
}
else {
dragging.type = dragType.MAX;
}
return dragging
}
function updatePosition (event, dragging = state.dragging.value) {
let pos;
const ratio = methods.getDraggingRatio(event, dragging);
const localModel = methods.convertRatioToModel(ratio);
switch (dragging.type) {
case dragType.MIN:
if (ratio <= dragging.ratioMax) {
pos = {
minR: ratio,
maxR: dragging.ratioMax,
min: localModel,
max: dragging.valueMax
};
state.focus.value = 'min';
}
else {
pos = {
minR: dragging.ratioMax,
maxR: ratio,
min: dragging.valueMax,
max: localModel
};
state.focus.value = 'max';
}
break
case dragType.MAX:
if (ratio >= dragging.ratioMin) {
pos = {
minR: dragging.ratioMin,
maxR: ratio,
min: dragging.valueMin,
max: localModel
};
state.focus.value = 'max';
}
else {
pos = {
minR: ratio,
maxR: dragging.ratioMin,
min: localModel,
max: dragging.valueMin
};
state.focus.value = 'min';
}
break
case dragType.RANGE:
const
ratioDelta = ratio - dragging.offsetRatio,
minR = between(dragging.ratioMin + ratioDelta, 0, 1 - dragging.rangeRatio),
modelDelta = localModel - dragging.offsetModel,
min = between(dragging.valueMin + modelDelta, props.min, props.max - dragging.rangeValue);
pos = {
minR,
maxR: minR + dragging.rangeRatio,
min: parseFloat(min.toFixed(state.decimals.value)),
max: parseFloat((min + dragging.rangeValue).toFixed(state.decimals.value))
};
state.focus.value = 'both';
break
}
// If either of the values to be emitted are null, set them to the defaults the user has entered.
model.value = model.value.min === null || model.value.max === null
? { min: pos.min || props.min, max: pos.max || props.max }
: { min: pos.min, max: pos.max };
if (props.snap !== true || props.step === 0) {
curMinRatio.value = pos.minR;
curMaxRatio.value = pos.maxR;
}
else {
curMinRatio.value = methods.convertModelToRatio(model.value.min);
curMaxRatio.value = methods.convertModelToRatio(model.value.max);
}
}
function onKeydown (evt) {
if (!keyCodes$2.includes(evt.keyCode)) {
return
}
stopAndPrevent(evt);
const
stepVal = ([ 34, 33 ].includes(evt.keyCode) ? 10 : 1) * state.step.value,
offset = (
([ 34, 37, 40 ].includes(evt.keyCode) ? -1 : 1)
* (state.isReversed.value === true ? -1 : 1)
* (props.vertical === true ? -1 : 1) * stepVal
);
if (state.focus.value === 'both') {
const interval = model.value.max - model.value.min;
const min = between(
parseFloat((model.value.min + offset).toFixed(state.decimals.value)),
state.innerMin.value,
state.innerMax.value - interval
);
model.value = {
min,
max: parseFloat((min + interval).toFixed(state.decimals.value))
};
}
else if (state.focus.value === false) {
return
}
else {
const which = state.focus.value;
model.value = {
...model.value,
[ which ]: between(
parseFloat((model.value[ which ] + offset).toFixed(state.decimals.value)),
which === 'min' ? state.innerMin.value : model.value.min,
which === 'max' ? state.innerMax.value : model.value.max
)
};
}
updateValue();
}
return () => {
const content = methods.getContent(
selectionBarStyle,
trackContainerTabindex,
trackContainerEvents,
node => {
node.push(
getMinThumb(),
getMaxThumb()
);
}
);
return vue.h('div', {
ref: rootRef,
class: 'q-range ' + state.classes.value + (
props.modelValue.min === null || props.modelValue.max === null
? ' q-slider--no-value'
: ''
),
...state.attributes.value,
'aria-valuenow': props.modelValue.min + '|' + props.modelValue.max
}, content)
}
}
});
var QRating = createComponent({
name: 'QRating',
props: {
...useSizeProps,
...useFormProps,
modelValue: {
type: Number,
required: true
},
max: {
type: [ String, Number ],
default: 5
},
icon: [ String, Array ],
iconHalf: [ String, Array ],
iconSelected: [ String, Array ],
iconAriaLabel: [ String, Array ],
color: [ String, Array ],
colorHalf: [ String, Array ],
colorSelected: [ String, Array ],
noReset: Boolean,
noDimming: Boolean,
readonly: Boolean,
disable: Boolean
},
emits: [ 'update:modelValue' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const sizeStyle = useSize(props);
const formAttrs = useFormAttrs(props);
const injectFormInput = useFormInject(formAttrs);
const mouseModel = vue.ref(0);
let iconRefs = {};
const editable = vue.computed(() =>
props.readonly !== true && props.disable !== true
);
const classes = vue.computed(() =>
'q-rating row inline items-center'
+ ` q-rating--${ editable.value === true ? '' : 'non-' }editable`
+ (props.noDimming === true ? ' q-rating--no-dimming' : '')
+ (props.disable === true ? ' disabled' : '')
+ (
props.color !== void 0 && Array.isArray(props.color) === false
? ` text-${ props.color }`
: ''
)
);
const iconData = vue.computed(() => {
const
iconLen = Array.isArray(props.icon) === true ? props.icon.length : 0,
selIconLen = Array.isArray(props.iconSelected) === true ? props.iconSelected.length : 0,
halfIconLen = Array.isArray(props.iconHalf) === true ? props.iconHalf.length : 0,
colorLen = Array.isArray(props.color) === true ? props.color.length : 0,
selColorLen = Array.isArray(props.colorSelected) === true ? props.colorSelected.length : 0,
halfColorLen = Array.isArray(props.colorHalf) === true ? props.colorHalf.length : 0;
return {
iconLen,
icon: iconLen > 0 ? props.icon[ iconLen - 1 ] : props.icon,
selIconLen,
selIcon: selIconLen > 0 ? props.iconSelected[ selIconLen - 1 ] : props.iconSelected,
halfIconLen,
halfIcon: halfIconLen > 0 ? props.iconHalf[ selIconLen - 1 ] : props.iconHalf,
colorLen,
color: colorLen > 0 ? props.color[ colorLen - 1 ] : props.color,
selColorLen,
selColor: selColorLen > 0 ? props.colorSelected[ selColorLen - 1 ] : props.colorSelected,
halfColorLen,
halfColor: halfColorLen > 0 ? props.colorHalf[ halfColorLen - 1 ] : props.colorHalf
}
});
const iconLabel = vue.computed(() => {
if (typeof props.iconAriaLabel === 'string') {
const label = props.iconAriaLabel.length !== 0 ? `${ props.iconAriaLabel } ` : '';
return i => `${ label }${ i }`
}
if (Array.isArray(props.iconAriaLabel) === true) {
const iMax = props.iconAriaLabel.length;
if (iMax > 0) {
return i => props.iconAriaLabel[ Math.min(i, iMax) - 1 ]
}
}
return (i, label) => `${ label } ${ i }`
});
const stars = vue.computed(() => {
const
acc = [],
icons = iconData.value,
ceil = Math.ceil(props.modelValue),
tabindex = editable.value === true ? 0 : null;
const halfIndex = props.iconHalf === void 0 || ceil === props.modelValue
? -1
: ceil;
for (let i = 1; i <= props.max; i++) {
const
active = (mouseModel.value === 0 && props.modelValue >= i) || (mouseModel.value > 0 && mouseModel.value >= i),
half = halfIndex === i && mouseModel.value < i,
exSelected = mouseModel.value > 0 && (half === true ? ceil : props.modelValue) >= i && mouseModel.value < i,
color = half === true
? (i <= icons.halfColorLen ? props.colorHalf[ i - 1 ] : icons.halfColor)
: (
icons.selColor !== void 0 && active === true
? (i <= icons.selColorLen ? props.colorSelected[ i - 1 ] : icons.selColor)
: (i <= icons.colorLen ? props.color[ i - 1 ] : icons.color)
),
name = (
half === true
? (i <= icons.halfIconLen ? props.iconHalf[ i - 1 ] : icons.halfIcon)
: (
icons.selIcon !== void 0 && (active === true || exSelected === true)
? (i <= icons.selIconLen ? props.iconSelected[ i - 1 ] : icons.selIcon)
: (i <= icons.iconLen ? props.icon[ i - 1 ] : icons.icon)
)
) || $q.iconSet.rating.icon;
acc.push({
name: (
half === true
? (i <= icons.halfIconLen ? props.iconHalf[ i - 1 ] : icons.halfIcon)
: (
icons.selIcon !== void 0 && (active === true || exSelected === true)
? (i <= icons.selIconLen ? props.iconSelected[ i - 1 ] : icons.selIcon)
: (i <= icons.iconLen ? props.icon[ i - 1 ] : icons.icon)
)
) || $q.iconSet.rating.icon,
attrs: {
tabindex,
role: 'radio',
'aria-checked': props.modelValue === i ? 'true' : 'false',
'aria-label': iconLabel.value(i, name)
},
iconClass: 'q-rating__icon'
+ (active === true || half === true ? ' q-rating__icon--active' : '')
+ (exSelected === true ? ' q-rating__icon--exselected' : '')
+ (mouseModel.value === i ? ' q-rating__icon--hovered' : '')
+ (color !== void 0 ? ` text-${ color }` : '')
});
}
return acc
});
const attributes = vue.computed(() => {
const attrs = { role: 'radiogroup' };
if (props.disable === true) {
attrs[ 'aria-disabled' ] = 'true';
}
if (props.readonly === true) {
attrs[ 'aria-readonly' ] = 'true';
}
return attrs
});
function set (value) {
if (editable.value === true) {
const
model = between(parseInt(value, 10), 1, parseInt(props.max, 10)),
newVal = props.noReset !== true && props.modelValue === model ? 0 : model;
newVal !== props.modelValue && emit('update:modelValue', newVal);
mouseModel.value = 0;
}
}
function setHoverValue (value) {
if (editable.value === true) {
mouseModel.value = value;
}
}
function onKeyup (e, i) {
switch (e.keyCode) {
case 13:
case 32:
set(i);
return stopAndPrevent(e)
case 37: // LEFT ARROW
case 40: // DOWN ARROW
if (iconRefs[ `rt${ i - 1 }` ]) {
iconRefs[ `rt${ i - 1 }` ].focus();
}
return stopAndPrevent(e)
case 39: // RIGHT ARROW
case 38: // UP ARROW
if (iconRefs[ `rt${ i + 1 }` ]) {
iconRefs[ `rt${ i + 1 }` ].focus();
}
return stopAndPrevent(e)
}
}
function resetMouseModel () {
mouseModel.value = 0;
}
vue.onBeforeUpdate(() => {
iconRefs = {};
});
return () => {
const child = [];
stars.value.forEach(({ iconClass, name, attrs }, index) => {
const i = index + 1;
child.push(
vue.h('div', {
key: i,
ref: el => { iconRefs[ `rt${ i }` ] = el; },
class: 'q-rating__icon-container flex flex-center',
...attrs,
onClick () { set(i); },
onMouseover () { setHoverValue(i); },
onMouseout: resetMouseModel,
onFocus () { setHoverValue(i); },
onBlur: resetMouseModel,
onKeyup (e) { onKeyup(e, i); }
}, hMergeSlot(
slots[ `tip-${ i }` ],
[ vue.h(QIcon, { class: iconClass, name }) ]
))
);
});
if (props.name !== void 0 && props.disable !== true) {
injectFormInput(child, 'push');
}
return vue.h('div', {
class: classes.value,
style: sizeStyle.value,
...attributes.value
}, child)
}
}
});
var QResponsive = createComponent({
name: 'QResponsive',
props: useRatioProps,
setup (props, { slots }) {
const ratioStyle = useRatio(props);
return () => vue.h('div', {
class: 'q-responsive'
}, [
vue.h('div', {
class: 'q-responsive__filler overflow-hidden'
}, [
vue.h('div', { style: ratioStyle.value })
]),
vue.h('div', {
class: 'q-responsive__content absolute-full fit'
}, hSlot(slots.default))
])
}
});
const axisList = [ 'vertical', 'horizontal' ];
const dirProps = {
vertical: { offset: 'offsetY', scroll: 'scrollTop', dir: 'down', dist: 'y' },
horizontal: { offset: 'offsetX', scroll: 'scrollLeft', dir: 'right', dist: 'x' }
};
const panOpts = {
prevent: true,
mouse: true,
mouseAllDir: true
};
const getMinThumbSize = size => (size >= 250 ? 50 : Math.ceil(size / 5));
var QScrollArea = createComponent({
name: 'QScrollArea',
props: {
...useDarkProps,
thumbStyle: Object,
verticalThumbStyle: Object,
horizontalThumbStyle: Object,
barStyle: [ Array, String, Object ],
verticalBarStyle: [ Array, String, Object ],
horizontalBarStyle: [ Array, String, Object ],
contentStyle: [ Array, String, Object ],
contentActiveStyle: [ Array, String, Object ],
delay: {
type: [ String, Number ],
default: 1000
},
visible: {
type: Boolean,
default: null
},
tabindex: [ String, Number ],
onScroll: Function
},
setup (props, { slots, emit }) {
// state management
const tempShowing = vue.ref(false);
const panning = vue.ref(false);
const hover = vue.ref(false);
// other...
const container = {
vertical: vue.ref(0),
horizontal: vue.ref(0)
};
const scroll = {
vertical: {
ref: vue.ref(null),
position: vue.ref(0),
size: vue.ref(0)
},
horizontal: {
ref: vue.ref(null),
position: vue.ref(0),
size: vue.ref(0)
}
};
const { proxy } = vue.getCurrentInstance();
const isDark = useDark(props, proxy.$q);
let timer = null, panRefPos;
const targetRef = vue.ref(null);
const classes = vue.computed(() =>
'q-scrollarea'
+ (isDark.value === true ? ' q-scrollarea--dark' : '')
);
scroll.vertical.percentage = vue.computed(() => {
const diff = scroll.vertical.size.value - container.vertical.value;
if (diff <= 0) { return 0 }
const p = between(scroll.vertical.position.value / diff, 0, 1);
return Math.round(p * 10000) / 10000
});
scroll.vertical.thumbHidden = vue.computed(() =>
(
(props.visible === null ? hover.value : props.visible) !== true
&& tempShowing.value === false
&& panning.value === false
) || scroll.vertical.size.value <= container.vertical.value + 1
);
scroll.vertical.thumbStart = vue.computed(() =>
scroll.vertical.percentage.value * (container.vertical.value - scroll.vertical.thumbSize.value)
);
scroll.vertical.thumbSize = vue.computed(() =>
Math.round(
between(
container.vertical.value * container.vertical.value / scroll.vertical.size.value,
getMinThumbSize(container.vertical.value),
container.vertical.value
)
)
);
scroll.vertical.style = vue.computed(() => {
return {
...props.thumbStyle,
...props.verticalThumbStyle,
top: `${ scroll.vertical.thumbStart.value }px`,
height: `${ scroll.vertical.thumbSize.value }px`
}
});
scroll.vertical.thumbClass = vue.computed(() =>
'q-scrollarea__thumb q-scrollarea__thumb--v absolute-right'
+ (scroll.vertical.thumbHidden.value === true ? ' q-scrollarea__thumb--invisible' : '')
);
scroll.vertical.barClass = vue.computed(() =>
'q-scrollarea__bar q-scrollarea__bar--v absolute-right'
+ (scroll.vertical.thumbHidden.value === true ? ' q-scrollarea__bar--invisible' : '')
);
scroll.horizontal.percentage = vue.computed(() => {
const diff = scroll.horizontal.size.value - container.horizontal.value;
if (diff <= 0) { return 0 }
const p = between(Math.abs(scroll.horizontal.position.value) / diff, 0, 1);
return Math.round(p * 10000) / 10000
});
scroll.horizontal.thumbHidden = vue.computed(() =>
(
(props.visible === null ? hover.value : props.visible) !== true
&& tempShowing.value === false
&& panning.value === false
) || scroll.horizontal.size.value <= container.horizontal.value + 1
);
scroll.horizontal.thumbStart = vue.computed(() =>
scroll.horizontal.percentage.value * (container.horizontal.value - scroll.horizontal.thumbSize.value)
);
scroll.horizontal.thumbSize = vue.computed(() =>
Math.round(
between(
container.horizontal.value * container.horizontal.value / scroll.horizontal.size.value,
getMinThumbSize(container.horizontal.value),
container.horizontal.value
)
)
);
scroll.horizontal.style = vue.computed(() => {
return {
...props.thumbStyle,
...props.horizontalThumbStyle,
[ proxy.$q.lang.rtl === true ? 'right' : 'left' ]: `${ scroll.horizontal.thumbStart.value }px`,
width: `${ scroll.horizontal.thumbSize.value }px`
}
});
scroll.horizontal.thumbClass = vue.computed(() =>
'q-scrollarea__thumb q-scrollarea__thumb--h absolute-bottom'
+ (scroll.horizontal.thumbHidden.value === true ? ' q-scrollarea__thumb--invisible' : '')
);
scroll.horizontal.barClass = vue.computed(() =>
'q-scrollarea__bar q-scrollarea__bar--h absolute-bottom'
+ (scroll.horizontal.thumbHidden.value === true ? ' q-scrollarea__bar--invisible' : '')
);
const mainStyle = vue.computed(() => (
scroll.vertical.thumbHidden.value === true && scroll.horizontal.thumbHidden.value === true
? props.contentStyle
: props.contentActiveStyle
));
const thumbVertDir = [ [
TouchPan,
e => { onPanThumb(e, 'vertical'); },
void 0,
{ vertical: true, ...panOpts }
] ];
const thumbHorizDir = [ [
TouchPan,
e => { onPanThumb(e, 'horizontal'); },
void 0,
{ horizontal: true, ...panOpts }
] ];
function getScroll () {
const info = {};
axisList.forEach(axis => {
const data = scroll[ axis ];
info[ axis + 'Position' ] = data.position.value;
info[ axis + 'Percentage' ] = data.percentage.value;
info[ axis + 'Size' ] = data.size.value;
info[ axis + 'ContainerSize' ] = container[ axis ].value;
});
return info
}
// we have lots of listeners, so
// ensure we're not emitting same info
// multiple times
const emitScroll = debounce(() => {
const info = getScroll();
info.ref = proxy;
emit('scroll', info);
}, 0);
function localSetScrollPosition (axis, offset, duration) {
if (axisList.includes(axis) === false) {
console.error('[QScrollArea]: wrong first param of setScrollPosition (vertical/horizontal)');
return
}
const fn = axis === 'vertical'
? setVerticalScrollPosition
: setHorizontalScrollPosition;
fn(targetRef.value, offset, duration);
}
function updateContainer ({ height, width }) {
let change = false;
if (container.vertical.value !== height) {
container.vertical.value = height;
change = true;
}
if (container.horizontal.value !== width) {
container.horizontal.value = width;
change = true;
}
change === true && startTimer();
}
function updateScroll ({ position }) {
let change = false;
if (scroll.vertical.position.value !== position.top) {
scroll.vertical.position.value = position.top;
change = true;
}
if (scroll.horizontal.position.value !== position.left) {
scroll.horizontal.position.value = position.left;
change = true;
}
change === true && startTimer();
}
function updateScrollSize ({ height, width }) {
if (scroll.horizontal.size.value !== width) {
scroll.horizontal.size.value = width;
startTimer();
}
if (scroll.vertical.size.value !== height) {
scroll.vertical.size.value = height;
startTimer();
}
}
function onPanThumb (e, axis) {
const data = scroll[ axis ];
if (e.isFirst === true) {
if (data.thumbHidden.value === true) {
return
}
panRefPos = data.position.value;
panning.value = true;
}
else if (panning.value !== true) {
return
}
if (e.isFinal === true) {
panning.value = false;
}
const dProp = dirProps[ axis ];
const containerSize = container[ axis ].value;
const multiplier = (data.size.value - containerSize) / (containerSize - data.thumbSize.value);
const distance = e.distance[ dProp.dist ];
const pos = panRefPos + (e.direction === dProp.dir ? 1 : -1) * distance * multiplier;
setScroll(pos, axis);
}
function onMousedown (evt, axis) {
const data = scroll[ axis ];
if (data.thumbHidden.value !== true) {
const offset = evt[ dirProps[ axis ].offset ];
if (offset < data.thumbStart.value || offset > data.thumbStart.value + data.thumbSize.value) {
const pos = offset - data.thumbSize.value / 2;
setScroll(pos / container[ axis ].value * data.size.value, axis);
}
// activate thumb pan
if (data.ref.value !== null) {
data.ref.value.dispatchEvent(new MouseEvent(evt.type, evt));
}
}
}
function onVerticalMousedown (evt) {
onMousedown(evt, 'vertical');
}
function onHorizontalMousedown (evt) {
onMousedown(evt, 'horizontal');
}
function startTimer () {
tempShowing.value = true;
timer !== null && clearTimeout(timer);
timer = setTimeout(() => {
timer = null;
tempShowing.value = false;
}, props.delay);
props.onScroll !== void 0 && emitScroll();
}
function setScroll (offset, axis) {
targetRef.value[ dirProps[ axis ].scroll ] = offset;
}
function onMouseenter () {
hover.value = true;
}
function onMouseleave () {
hover.value = false;
}
let scrollPosition = null;
vue.watch(() => proxy.$q.lang.rtl, rtl => {
if (targetRef.value !== null) {
setHorizontalScrollPosition(
targetRef.value,
Math.abs(scroll.horizontal.position.value) * (rtl === true ? -1 : 1)
);
}
});
vue.onDeactivated(() => {
scrollPosition = {
top: scroll.vertical.position.value,
left: scroll.horizontal.position.value
};
});
vue.onActivated(() => {
if (scrollPosition === null) { return }
const scrollTarget = targetRef.value;
if (scrollTarget !== null) {
setHorizontalScrollPosition(scrollTarget, scrollPosition.left);
setVerticalScrollPosition(scrollTarget, scrollPosition.top);
}
});
vue.onBeforeUnmount(emitScroll.cancel);
// expose public methods
Object.assign(proxy, {
getScrollTarget: () => targetRef.value,
getScroll,
getScrollPosition: () => ({
top: scroll.vertical.position.value,
left: scroll.horizontal.position.value
}),
getScrollPercentage: () => ({
top: scroll.vertical.percentage.value,
left: scroll.horizontal.percentage.value
}),
setScrollPosition: localSetScrollPosition,
setScrollPercentage (axis, percentage, duration) {
localSetScrollPosition(
axis,
percentage
* (scroll[ axis ].size.value - container[ axis ].value)
* (axis === 'horizontal' && proxy.$q.lang.rtl === true ? -1 : 1),
duration
);
}
});
return () => {
return vue.h('div', {
class: classes.value,
onMouseenter,
onMouseleave
}, [
vue.h('div', {
ref: targetRef,
class: 'q-scrollarea__container scroll relative-position fit hide-scrollbar',
tabindex: props.tabindex !== void 0 ? props.tabindex : void 0
}, [
vue.h('div', {
class: 'q-scrollarea__content absolute',
style: mainStyle.value
}, hMergeSlot(slots.default, [
vue.h(QResizeObserver, {
debounce: 0,
onResize: updateScrollSize
})
])),
vue.h(QScrollObserver, {
axis: 'both',
onScroll: updateScroll
})
]),
vue.h(QResizeObserver, {
debounce: 0,
onResize: updateContainer
}),
vue.h('div', {
class: scroll.vertical.barClass.value,
style: [ props.barStyle, props.verticalBarStyle ],
'aria-hidden': 'true',
onMousedown: onVerticalMousedown
}),
vue.h('div', {
class: scroll.horizontal.barClass.value,
style: [ props.barStyle, props.horizontalBarStyle ],
'aria-hidden': 'true',
onMousedown: onHorizontalMousedown
}),
vue.withDirectives(
vue.h('div', {
ref: scroll.vertical.ref,
class: scroll.vertical.thumbClass.value,
style: scroll.vertical.style.value,
'aria-hidden': 'true'
}),
thumbVertDir
),
vue.withDirectives(
vue.h('div', {
ref: scroll.horizontal.ref,
class: scroll.horizontal.thumbClass.value,
style: scroll.horizontal.style.value,
'aria-hidden': 'true'
}),
thumbHorizDir
)
])
}
}
});
const aggBucketSize = 1000;
const scrollToEdges = [
'start',
'center',
'end',
'start-force',
'center-force',
'end-force'
];
const filterProto = Array.prototype.filter;
const setOverflowAnchor = window.getComputedStyle(document.body).overflowAnchor === void 0
? noop
: function (contentEl, index) {
if (contentEl === null) {
return
}
if (contentEl._qOverflowAnimationFrame !== void 0) {
cancelAnimationFrame(contentEl._qOverflowAnimationFrame);
}
contentEl._qOverflowAnimationFrame = requestAnimationFrame(() => {
if (contentEl === null) {
return
}
contentEl._qOverflowAnimationFrame = void 0;
const children = contentEl.children || [];
filterProto
.call(children, el => el.dataset && el.dataset.qVsAnchor !== void 0)
.forEach(el => {
delete el.dataset.qVsAnchor;
});
const el = children[ index ];
if (el && el.dataset) {
el.dataset.qVsAnchor = '';
}
});
};
function sumFn (acc, h) {
return acc + h
}
function getScrollDetails (
parent,
child,
beforeRef,
afterRef,
horizontal,
rtl,
stickyStart,
stickyEnd
) {
const
parentCalc = parent === window ? document.scrollingElement || document.documentElement : parent,
propElSize = horizontal === true ? 'offsetWidth' : 'offsetHeight',
details = {
scrollStart: 0,
scrollViewSize: -stickyStart - stickyEnd,
scrollMaxSize: 0,
offsetStart: -stickyStart,
offsetEnd: -stickyEnd
};
if (horizontal === true) {
if (parent === window) {
details.scrollStart = window.pageXOffset || window.scrollX || document.body.scrollLeft || 0;
details.scrollViewSize += document.documentElement.clientWidth;
}
else {
details.scrollStart = parentCalc.scrollLeft;
details.scrollViewSize += parentCalc.clientWidth;
}
details.scrollMaxSize = parentCalc.scrollWidth;
if (rtl === true) {
details.scrollStart = (rtlHasScrollBug === true ? details.scrollMaxSize - details.scrollViewSize : 0) - details.scrollStart;
}
}
else {
if (parent === window) {
details.scrollStart = window.pageYOffset || window.scrollY || document.body.scrollTop || 0;
details.scrollViewSize += document.documentElement.clientHeight;
}
else {
details.scrollStart = parentCalc.scrollTop;
details.scrollViewSize += parentCalc.clientHeight;
}
details.scrollMaxSize = parentCalc.scrollHeight;
}
if (beforeRef !== null) {
for (let el = beforeRef.previousElementSibling; el !== null; el = el.previousElementSibling) {
if (el.classList.contains('q-virtual-scroll--skip') === false) {
details.offsetStart += el[ propElSize ];
}
}
}
if (afterRef !== null) {
for (let el = afterRef.nextElementSibling; el !== null; el = el.nextElementSibling) {
if (el.classList.contains('q-virtual-scroll--skip') === false) {
details.offsetEnd += el[ propElSize ];
}
}
}
if (child !== parent) {
const
parentRect = parentCalc.getBoundingClientRect(),
childRect = child.getBoundingClientRect();
if (horizontal === true) {
details.offsetStart += childRect.left - parentRect.left;
details.offsetEnd -= childRect.width;
}
else {
details.offsetStart += childRect.top - parentRect.top;
details.offsetEnd -= childRect.height;
}
if (parent !== window) {
details.offsetStart += details.scrollStart;
}
details.offsetEnd += details.scrollMaxSize - details.offsetStart;
}
return details
}
function setScroll (parent, scroll, horizontal, rtl) {
if (scroll === 'end') {
scroll = (parent === window ? document.body : parent)[
horizontal === true ? 'scrollWidth' : 'scrollHeight'
];
}
if (parent === window) {
if (horizontal === true) {
if (rtl === true) {
scroll = (rtlHasScrollBug === true ? document.body.scrollWidth - document.documentElement.clientWidth : 0) - scroll;
}
window.scrollTo(scroll, window.pageYOffset || window.scrollY || document.body.scrollTop || 0);
}
else {
window.scrollTo(window.pageXOffset || window.scrollX || document.body.scrollLeft || 0, scroll);
}
}
else if (horizontal === true) {
if (rtl === true) {
scroll = (rtlHasScrollBug === true ? parent.scrollWidth - parent.offsetWidth : 0) - scroll;
}
parent.scrollLeft = scroll;
}
else {
parent.scrollTop = scroll;
}
}
function sumSize (sizeAgg, size, from, to) {
if (from >= to) { return 0 }
const
lastTo = size.length,
fromAgg = Math.floor(from / aggBucketSize),
toAgg = Math.floor((to - 1) / aggBucketSize) + 1;
let total = sizeAgg.slice(fromAgg, toAgg).reduce(sumFn, 0);
if (from % aggBucketSize !== 0) {
total -= size.slice(fromAgg * aggBucketSize, from).reduce(sumFn, 0);
}
if (to % aggBucketSize !== 0 && to !== lastTo) {
total -= size.slice(to, toAgg * aggBucketSize).reduce(sumFn, 0);
}
return total
}
const commonVirtScrollProps = {
virtualScrollSliceSize: {
type: [ Number, String ],
default: null
},
virtualScrollSliceRatioBefore: {
type: [ Number, String ],
default: 1
},
virtualScrollSliceRatioAfter: {
type: [ Number, String ],
default: 1
},
virtualScrollItemSize: {
type: [ Number, String ],
default: 24
},
virtualScrollStickySizeStart: {
type: [ Number, String ],
default: 0
},
virtualScrollStickySizeEnd: {
type: [ Number, String ],
default: 0
},
tableColspan: [ Number, String ]
};
const commonVirtPropsList = Object.keys(commonVirtScrollProps);
const useVirtualScrollProps = {
virtualScrollHorizontal: Boolean,
onVirtualScroll: Function,
...commonVirtScrollProps
};
function useVirtualScroll ({
virtualScrollLength, getVirtualScrollTarget, getVirtualScrollEl,
virtualScrollItemSizeComputed // optional
}) {
const vm = vue.getCurrentInstance();
const { props, emit, proxy } = vm;
const { $q } = proxy;
let prevScrollStart, prevToIndex, localScrollViewSize, virtualScrollSizesAgg = [], virtualScrollSizes;
const virtualScrollPaddingBefore = vue.ref(0);
const virtualScrollPaddingAfter = vue.ref(0);
const virtualScrollSliceSizeComputed = vue.ref({});
const beforeRef = vue.ref(null);
const afterRef = vue.ref(null);
const contentRef = vue.ref(null);
const virtualScrollSliceRange = vue.ref({ from: 0, to: 0 });
const colspanAttr = vue.computed(() => (props.tableColspan !== void 0 ? props.tableColspan : 100));
if (virtualScrollItemSizeComputed === void 0) {
virtualScrollItemSizeComputed = vue.computed(() => props.virtualScrollItemSize);
}
const needsReset = vue.computed(() => virtualScrollItemSizeComputed.value + ';' + props.virtualScrollHorizontal);
const needsSliceRecalc = vue.computed(() =>
needsReset.value + ';' + props.virtualScrollSliceRatioBefore + ';' + props.virtualScrollSliceRatioAfter
);
vue.watch(needsSliceRecalc, () => { setVirtualScrollSize(); });
vue.watch(needsReset, reset);
function reset () {
localResetVirtualScroll(prevToIndex, true);
}
function refresh (toIndex) {
localResetVirtualScroll(toIndex === void 0 ? prevToIndex : toIndex);
}
function scrollTo (toIndex, edge) {
const scrollEl = getVirtualScrollTarget();
if (scrollEl === void 0 || scrollEl === null || scrollEl.nodeType === 8) {
return
}
const scrollDetails = getScrollDetails(
scrollEl,
getVirtualScrollEl(),
beforeRef.value,
afterRef.value,
props.virtualScrollHorizontal,
$q.lang.rtl,
props.virtualScrollStickySizeStart,
props.virtualScrollStickySizeEnd
);
localScrollViewSize !== scrollDetails.scrollViewSize && setVirtualScrollSize(scrollDetails.scrollViewSize);
setVirtualScrollSliceRange(
scrollEl,
scrollDetails,
Math.min(virtualScrollLength.value - 1, Math.max(0, parseInt(toIndex, 10) || 0)),
0,
scrollToEdges.indexOf(edge) > -1 ? edge : (prevToIndex > -1 && toIndex > prevToIndex ? 'end' : 'start')
);
}
function localOnVirtualScrollEvt () {
const scrollEl = getVirtualScrollTarget();
if (scrollEl === void 0 || scrollEl === null || scrollEl.nodeType === 8) {
return
}
const
scrollDetails = getScrollDetails(
scrollEl,
getVirtualScrollEl(),
beforeRef.value,
afterRef.value,
props.virtualScrollHorizontal,
$q.lang.rtl,
props.virtualScrollStickySizeStart,
props.virtualScrollStickySizeEnd
),
listLastIndex = virtualScrollLength.value - 1,
listEndOffset = scrollDetails.scrollMaxSize - scrollDetails.offsetStart - scrollDetails.offsetEnd - virtualScrollPaddingAfter.value;
if (prevScrollStart === scrollDetails.scrollStart) {
return
}
if (scrollDetails.scrollMaxSize <= 0) {
setVirtualScrollSliceRange(scrollEl, scrollDetails, 0, 0);
return
}
localScrollViewSize !== scrollDetails.scrollViewSize && setVirtualScrollSize(scrollDetails.scrollViewSize);
updateVirtualScrollSizes(virtualScrollSliceRange.value.from);
const scrollMaxStart = Math.floor(scrollDetails.scrollMaxSize
- Math.max(scrollDetails.scrollViewSize, scrollDetails.offsetEnd)
- Math.min(virtualScrollSizes[ listLastIndex ], scrollDetails.scrollViewSize / 2));
if (scrollMaxStart > 0 && Math.ceil(scrollDetails.scrollStart) >= scrollMaxStart) {
setVirtualScrollSliceRange(
scrollEl,
scrollDetails,
listLastIndex,
scrollDetails.scrollMaxSize - scrollDetails.offsetEnd - virtualScrollSizesAgg.reduce(sumFn, 0)
);
return
}
let
toIndex = 0,
listOffset = scrollDetails.scrollStart - scrollDetails.offsetStart,
offset = listOffset;
if (listOffset <= listEndOffset && listOffset + scrollDetails.scrollViewSize >= virtualScrollPaddingBefore.value) {
listOffset -= virtualScrollPaddingBefore.value;
toIndex = virtualScrollSliceRange.value.from;
offset = listOffset;
}
else {
for (let j = 0; listOffset >= virtualScrollSizesAgg[ j ] && toIndex < listLastIndex; j++) {
listOffset -= virtualScrollSizesAgg[ j ];
toIndex += aggBucketSize;
}
}
while (listOffset > 0 && toIndex < listLastIndex) {
listOffset -= virtualScrollSizes[ toIndex ];
if (listOffset > -scrollDetails.scrollViewSize) {
toIndex++;
offset = listOffset;
}
else {
offset = virtualScrollSizes[ toIndex ] + listOffset;
}
}
setVirtualScrollSliceRange(
scrollEl,
scrollDetails,
toIndex,
offset
);
}
function setVirtualScrollSliceRange (scrollEl, scrollDetails, toIndex, offset, align) {
const alignForce = typeof align === 'string' && align.indexOf('-force') > -1;
const alignEnd = alignForce === true ? align.replace('-force', '') : align;
const alignRange = alignEnd !== void 0 ? alignEnd : 'start';
let
from = Math.max(0, toIndex - virtualScrollSliceSizeComputed.value[ alignRange ]),
to = from + virtualScrollSliceSizeComputed.value.total;
if (to > virtualScrollLength.value) {
to = virtualScrollLength.value;
from = Math.max(0, to - virtualScrollSliceSizeComputed.value.total);
}
prevScrollStart = scrollDetails.scrollStart;
const rangeChanged = from !== virtualScrollSliceRange.value.from || to !== virtualScrollSliceRange.value.to;
if (rangeChanged === false && alignEnd === void 0) {
emitScroll(toIndex);
return
}
const { activeElement } = document;
const contentEl = contentRef.value;
if (
rangeChanged === true
&& contentEl !== null
&& contentEl !== activeElement
&& contentEl.contains(activeElement) === true
) {
contentEl.addEventListener('focusout', onBlurRefocusFn);
setTimeout(() => {
contentEl !== null && contentEl.removeEventListener('focusout', onBlurRefocusFn);
});
}
setOverflowAnchor(contentEl, toIndex - from);
const sizeBefore = alignEnd !== void 0 ? virtualScrollSizes.slice(from, toIndex).reduce(sumFn, 0) : 0;
if (rangeChanged === true) {
// vue key matching algorithm works only if
// the array of VNodes changes on only one of the ends
// so we first change one end and then the other
const tempTo = to >= virtualScrollSliceRange.value.from && from <= virtualScrollSliceRange.value.to
? virtualScrollSliceRange.value.to
: to;
virtualScrollSliceRange.value = { from, to: tempTo };
virtualScrollPaddingBefore.value = sumSize(virtualScrollSizesAgg, virtualScrollSizes, 0, from);
virtualScrollPaddingAfter.value = sumSize(virtualScrollSizesAgg, virtualScrollSizes, to, virtualScrollLength.value);
requestAnimationFrame(() => {
if (virtualScrollSliceRange.value.to !== to && prevScrollStart === scrollDetails.scrollStart) {
virtualScrollSliceRange.value = { from: virtualScrollSliceRange.value.from, to };
virtualScrollPaddingAfter.value = sumSize(virtualScrollSizesAgg, virtualScrollSizes, to, virtualScrollLength.value);
}
});
}
requestAnimationFrame(() => {
// if the scroll was changed give up
// (another call to setVirtualScrollSliceRange before animation frame)
if (prevScrollStart !== scrollDetails.scrollStart) {
return
}
if (rangeChanged === true) {
updateVirtualScrollSizes(from);
}
const
sizeAfter = virtualScrollSizes.slice(from, toIndex).reduce(sumFn, 0),
posStart = sizeAfter + scrollDetails.offsetStart + virtualScrollPaddingBefore.value,
posEnd = posStart + virtualScrollSizes[ toIndex ];
let scrollPosition = posStart + offset;
if (alignEnd !== void 0) {
const sizeDiff = sizeAfter - sizeBefore;
const scrollStart = scrollDetails.scrollStart + sizeDiff;
scrollPosition = alignForce !== true && scrollStart < posStart && posEnd < scrollStart + scrollDetails.scrollViewSize
? scrollStart
: (
alignEnd === 'end'
? posEnd - scrollDetails.scrollViewSize
: posStart - (alignEnd === 'start' ? 0 : Math.round((scrollDetails.scrollViewSize - virtualScrollSizes[ toIndex ]) / 2))
);
}
prevScrollStart = scrollPosition;
setScroll(
scrollEl,
scrollPosition,
props.virtualScrollHorizontal,
$q.lang.rtl
);
emitScroll(toIndex);
});
}
function updateVirtualScrollSizes (from) {
const contentEl = contentRef.value;
if (contentEl) {
const
children = filterProto.call(
contentEl.children,
el => el.classList && el.classList.contains('q-virtual-scroll--skip') === false
),
childrenLength = children.length,
sizeFn = props.virtualScrollHorizontal === true
? el => el.getBoundingClientRect().width
: el => el.offsetHeight;
let
index = from,
size, diff;
for (let i = 0; i < childrenLength;) {
size = sizeFn(children[ i ]);
i++;
while (i < childrenLength && children[ i ].classList.contains('q-virtual-scroll--with-prev') === true) {
size += sizeFn(children[ i ]);
i++;
}
diff = size - virtualScrollSizes[ index ];
if (diff !== 0) {
virtualScrollSizes[ index ] += diff;
virtualScrollSizesAgg[ Math.floor(index / aggBucketSize) ] += diff;
}
index++;
}
}
}
function onBlurRefocusFn () {
contentRef.value !== null && contentRef.value !== void 0 && contentRef.value.focus();
}
function localResetVirtualScroll (toIndex, fullReset) {
const defaultSize = 1 * virtualScrollItemSizeComputed.value;
if (fullReset === true || Array.isArray(virtualScrollSizes) === false) {
virtualScrollSizes = [];
}
const oldVirtualScrollSizesLength = virtualScrollSizes.length;
virtualScrollSizes.length = virtualScrollLength.value;
for (let i = virtualScrollLength.value - 1; i >= oldVirtualScrollSizesLength; i--) {
virtualScrollSizes[ i ] = defaultSize;
}
const jMax = Math.floor((virtualScrollLength.value - 1) / aggBucketSize);
virtualScrollSizesAgg = [];
for (let j = 0; j <= jMax; j++) {
let size = 0;
const iMax = Math.min((j + 1) * aggBucketSize, virtualScrollLength.value);
for (let i = j * aggBucketSize; i < iMax; i++) {
size += virtualScrollSizes[ i ];
}
virtualScrollSizesAgg.push(size);
}
prevToIndex = -1;
prevScrollStart = void 0;
virtualScrollPaddingBefore.value = sumSize(virtualScrollSizesAgg, virtualScrollSizes, 0, virtualScrollSliceRange.value.from);
virtualScrollPaddingAfter.value = sumSize(virtualScrollSizesAgg, virtualScrollSizes, virtualScrollSliceRange.value.to, virtualScrollLength.value);
if (toIndex >= 0) {
updateVirtualScrollSizes(virtualScrollSliceRange.value.from);
vue.nextTick(() => { scrollTo(toIndex); });
}
else {
onVirtualScrollEvt();
}
}
function setVirtualScrollSize (scrollViewSize) {
if (scrollViewSize === void 0 && typeof window !== 'undefined') {
const scrollEl = getVirtualScrollTarget();
if (scrollEl !== void 0 && scrollEl !== null && scrollEl.nodeType !== 8) {
scrollViewSize = getScrollDetails(
scrollEl,
getVirtualScrollEl(),
beforeRef.value,
afterRef.value,
props.virtualScrollHorizontal,
$q.lang.rtl,
props.virtualScrollStickySizeStart,
props.virtualScrollStickySizeEnd
).scrollViewSize;
}
}
localScrollViewSize = scrollViewSize;
const virtualScrollSliceRatioBefore = parseFloat(props.virtualScrollSliceRatioBefore) || 0;
const virtualScrollSliceRatioAfter = parseFloat(props.virtualScrollSliceRatioAfter) || 0;
const multiplier = 1 + virtualScrollSliceRatioBefore + virtualScrollSliceRatioAfter;
const view = scrollViewSize === void 0 || scrollViewSize <= 0
? 1
: Math.ceil(scrollViewSize / virtualScrollItemSizeComputed.value);
const baseSize = Math.max(
1,
view,
Math.ceil((props.virtualScrollSliceSize > 0 ? props.virtualScrollSliceSize : 10) / multiplier)
);
virtualScrollSliceSizeComputed.value = {
total: Math.ceil(baseSize * multiplier),
start: Math.ceil(baseSize * virtualScrollSliceRatioBefore),
center: Math.ceil(baseSize * (0.5 + virtualScrollSliceRatioBefore)),
end: Math.ceil(baseSize * (1 + virtualScrollSliceRatioBefore)),
view
};
}
function padVirtualScroll (tag, content) {
const paddingSize = props.virtualScrollHorizontal === true ? 'width' : 'height';
const style = {
[ '--q-virtual-scroll-item-' + paddingSize ]: virtualScrollItemSizeComputed.value + 'px'
};
return [
tag === 'tbody'
? vue.h(tag, {
class: 'q-virtual-scroll__padding',
key: 'before',
ref: beforeRef
}, [
vue.h('tr', [
vue.h('td', {
style: { [ paddingSize ]: `${ virtualScrollPaddingBefore.value }px`, ...style },
colspan: colspanAttr.value
})
])
])
: vue.h(tag, {
class: 'q-virtual-scroll__padding',
key: 'before',
ref: beforeRef,
style: { [ paddingSize ]: `${ virtualScrollPaddingBefore.value }px`, ...style }
}),
vue.h(tag, {
class: 'q-virtual-scroll__content',
key: 'content',
ref: contentRef,
tabindex: -1
}, content.flat()),
tag === 'tbody'
? vue.h(tag, {
class: 'q-virtual-scroll__padding',
key: 'after',
ref: afterRef
}, [
vue.h('tr', [
vue.h('td', {
style: { [ paddingSize ]: `${ virtualScrollPaddingAfter.value }px`, ...style },
colspan: colspanAttr.value
})
])
])
: vue.h(tag, {
class: 'q-virtual-scroll__padding',
key: 'after',
ref: afterRef,
style: { [ paddingSize ]: `${ virtualScrollPaddingAfter.value }px`, ...style }
})
]
}
function emitScroll (index) {
if (prevToIndex !== index) {
props.onVirtualScroll !== void 0 && emit('virtualScroll', {
index,
from: virtualScrollSliceRange.value.from,
to: virtualScrollSliceRange.value.to - 1,
direction: index < prevToIndex ? 'decrease' : 'increase',
ref: proxy
});
prevToIndex = index;
}
}
setVirtualScrollSize();
const onVirtualScrollEvt = debounce(
localOnVirtualScrollEvt,
$q.platform.is.ios === true ? 120 : 35
);
vue.onBeforeMount(() => {
setVirtualScrollSize();
});
let shouldActivate = false;
vue.onDeactivated(() => {
shouldActivate = true;
});
vue.onActivated(() => {
if (shouldActivate !== true) { return }
const scrollEl = getVirtualScrollTarget();
if (prevScrollStart !== void 0 && scrollEl !== void 0 && scrollEl !== null && scrollEl.nodeType !== 8) {
setScroll(
scrollEl,
prevScrollStart,
props.virtualScrollHorizontal,
$q.lang.rtl
);
}
else {
scrollTo(prevToIndex);
}
});
vue.onBeforeUnmount(() => {
onVirtualScrollEvt.cancel();
});
// expose public methods
Object.assign(proxy, { scrollTo, reset, refresh });
return {
virtualScrollSliceRange,
virtualScrollSliceSizeComputed,
setVirtualScrollSize,
onVirtualScrollEvt,
localResetVirtualScroll,
padVirtualScroll,
scrollTo,
reset,
refresh
}
}
const validateNewValueMode = v => [ 'add', 'add-unique', 'toggle' ].includes(v);
const reEscapeList = '.*+?^${}()|[]\\';
const fieldPropsList = Object.keys(useFieldProps);
var QSelect = createComponent({
name: 'QSelect',
inheritAttrs: false,
props: {
...useVirtualScrollProps,
...useFormProps,
...useFieldProps,
modelValue: {
required: true
},
multiple: Boolean,
displayValue: [ String, Number ],
displayValueHtml: Boolean,
dropdownIcon: String,
options: {
type: Array,
default: () => []
},
optionValue: [ Function, String ],
optionLabel: [ Function, String ],
optionDisable: [ Function, String ],
hideSelected: Boolean,
hideDropdownIcon: Boolean,
fillInput: Boolean,
maxValues: [ Number, String ],
optionsDense: Boolean,
optionsDark: {
type: Boolean,
default: null
},
optionsSelectedClass: String,
optionsHtml: Boolean,
optionsCover: Boolean,
menuShrink: Boolean,
menuAnchor: String,
menuSelf: String,
menuOffset: Array,
popupContentClass: String,
popupContentStyle: [ String, Array, Object ],
useInput: Boolean,
useChips: Boolean,
newValueMode: {
type: String,
validator: validateNewValueMode
},
mapOptions: Boolean,
emitValue: Boolean,
inputDebounce: {
type: [ Number, String ],
default: 500
},
inputClass: [ Array, String, Object ],
inputStyle: [ Array, String, Object ],
tabindex: {
type: [ String, Number ],
default: 0
},
autocomplete: String,
transitionShow: String,
transitionHide: String,
transitionDuration: [ String, Number ],
behavior: {
type: String,
validator: v => [ 'default', 'menu', 'dialog' ].includes(v),
default: 'default'
},
virtualScrollItemSize: {
type: [ Number, String ],
default: void 0
},
onNewValue: Function,
onFilter: Function
},
emits: [
...useFieldEmits,
'add', 'remove', 'inputValue', 'newValue',
'keyup', 'keypress', 'keydown',
'filterAbort'
],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const menu = vue.ref(false);
const dialog = vue.ref(false);
const optionIndex = vue.ref(-1);
const inputValue = vue.ref('');
const dialogFieldFocused = vue.ref(false);
const innerLoadingIndicator = vue.ref(false);
let inputTimer = null, innerValueCache,
hasDialog, userInputValue, filterId = null, defaultInputValue,
transitionShowComputed, searchBuffer, searchBufferExp;
const inputRef = vue.ref(null);
const targetRef = vue.ref(null);
const menuRef = vue.ref(null);
const dialogRef = vue.ref(null);
const menuContentRef = vue.ref(null);
const nameProp = useFormInputNameAttr(props);
const onComposition = useKeyComposition(onInput);
const virtualScrollLength = vue.computed(() => (
Array.isArray(props.options)
? props.options.length
: 0
));
const virtualScrollItemSizeComputed = vue.computed(() => (
props.virtualScrollItemSize === void 0
? (props.optionsDense === true ? 24 : 48)
: props.virtualScrollItemSize
));
const {
virtualScrollSliceRange,
virtualScrollSliceSizeComputed,
localResetVirtualScroll,
padVirtualScroll,
onVirtualScrollEvt,
scrollTo,
setVirtualScrollSize
} = useVirtualScroll({
virtualScrollLength, getVirtualScrollTarget, getVirtualScrollEl,
virtualScrollItemSizeComputed
});
const state = useFieldState();
const innerValue = vue.computed(() => {
const
mapNull = props.mapOptions === true && props.multiple !== true,
val = props.modelValue !== void 0 && (props.modelValue !== null || mapNull === true)
? (props.multiple === true && Array.isArray(props.modelValue) ? props.modelValue : [ props.modelValue ])
: [];
if (props.mapOptions === true && Array.isArray(props.options) === true) {
const cache = props.mapOptions === true && innerValueCache !== void 0
? innerValueCache
: [];
const values = val.map(v => getOption(v, cache));
return props.modelValue === null && mapNull === true
? values.filter(v => v !== null)
: values
}
return val
});
const innerFieldProps = vue.computed(() => {
const acc = {};
fieldPropsList.forEach(key => {
const val = props[ key ];
if (val !== void 0) {
acc[ key ] = val;
}
});
return acc
});
const isOptionsDark = vue.computed(() => (
props.optionsDark === null
? state.isDark.value
: props.optionsDark
));
const hasValue = vue.computed(() => fieldValueIsFilled(innerValue.value));
const computedInputClass = vue.computed(() => {
let cls = 'q-field__input q-placeholder col';
if (props.hideSelected === true || innerValue.value.length === 0) {
return [ cls, props.inputClass ]
}
cls += ' q-field__input--padding';
return props.inputClass === void 0
? cls
: [ cls, props.inputClass ]
});
const menuContentClass = vue.computed(() =>
(props.virtualScrollHorizontal === true ? 'q-virtual-scroll--horizontal' : '')
+ (props.popupContentClass ? ' ' + props.popupContentClass : '')
);
const noOptions = vue.computed(() => virtualScrollLength.value === 0);
const selectedString = vue.computed(() =>
innerValue.value
.map(opt => getOptionLabel.value(opt))
.join(', ')
);
const ariaCurrentValue = vue.computed(() => (props.displayValue !== void 0
? props.displayValue
: selectedString.value
));
const needsHtmlFn = vue.computed(() => (
props.optionsHtml === true
? () => true
: opt => opt !== void 0 && opt !== null && opt.html === true
));
const valueAsHtml = vue.computed(() => (
props.displayValueHtml === true || (
props.displayValue === void 0 && (
props.optionsHtml === true
|| innerValue.value.some(needsHtmlFn.value)
)
)
));
const tabindex = vue.computed(() => (state.focused.value === true ? props.tabindex : -1));
const comboboxAttrs = vue.computed(() => {
const attrs = {
tabindex: props.tabindex,
role: 'combobox',
'aria-label': props.label,
'aria-readonly': props.readonly === true ? 'true' : 'false',
'aria-autocomplete': props.useInput === true ? 'list' : 'none',
'aria-expanded': menu.value === true ? 'true' : 'false',
'aria-controls': `${ state.targetUid.value }_lb`
};
if (optionIndex.value >= 0) {
attrs[ 'aria-activedescendant' ] = `${ state.targetUid.value }_${ optionIndex.value }`;
}
return attrs
});
const listboxAttrs = vue.computed(() => ({
id: `${ state.targetUid.value }_lb`,
role: 'listbox',
'aria-multiselectable': props.multiple === true ? 'true' : 'false'
}));
const selectedScope = vue.computed(() => {
return innerValue.value.map((opt, i) => ({
index: i,
opt,
html: needsHtmlFn.value(opt),
selected: true,
removeAtIndex: removeAtIndexAndFocus,
toggleOption,
tabindex: tabindex.value
}))
});
const optionScope = vue.computed(() => {
if (virtualScrollLength.value === 0) {
return []
}
const { from, to } = virtualScrollSliceRange.value;
return props.options.slice(from, to).map((opt, i) => {
const disable = isOptionDisabled.value(opt) === true;
const index = from + i;
const itemProps = {
clickable: true,
active: false,
activeClass: computedOptionsSelectedClass.value,
manualFocus: true,
focused: false,
disable,
tabindex: -1,
dense: props.optionsDense,
dark: isOptionsDark.value,
role: 'option',
id: `${ state.targetUid.value }_${ index }`,
onClick: () => { toggleOption(opt); }
};
if (disable !== true) {
isOptionSelected(opt) === true && (itemProps.active = true);
optionIndex.value === index && (itemProps.focused = true);
itemProps[ 'aria-selected' ] = itemProps.active === true ? 'true' : 'false';
if ($q.platform.is.desktop === true) {
itemProps.onMousemove = () => { menu.value === true && setOptionIndex(index); };
}
}
return {
index,
opt,
html: needsHtmlFn.value(opt),
label: getOptionLabel.value(opt),
selected: itemProps.active,
focused: itemProps.focused,
toggleOption,
setOptionIndex,
itemProps
}
})
});
const dropdownArrowIcon = vue.computed(() => (
props.dropdownIcon !== void 0
? props.dropdownIcon
: $q.iconSet.arrow.dropdown
));
const squaredMenu = vue.computed(() =>
props.optionsCover === false
&& props.outlined !== true
&& props.standout !== true
&& props.borderless !== true
&& props.rounded !== true
);
const computedOptionsSelectedClass = vue.computed(() => (
props.optionsSelectedClass !== void 0
? props.optionsSelectedClass
: (props.color !== void 0 ? `text-${ props.color }` : '')
));
// returns method to get value of an option;
// takes into account 'option-value' prop
const getOptionValue = vue.computed(() => getPropValueFn(props.optionValue, 'value'));
// returns method to get label of an option;
// takes into account 'option-label' prop
const getOptionLabel = vue.computed(() => getPropValueFn(props.optionLabel, 'label'));
// returns method to tell if an option is disabled;
// takes into account 'option-disable' prop
const isOptionDisabled = vue.computed(() => getPropValueFn(props.optionDisable, 'disable'));
const innerOptionsValue = vue.computed(() => innerValue.value.map(opt => getOptionValue.value(opt)));
const inputControlEvents = vue.computed(() => {
const evt = {
onInput,
// Safari < 10.2 & UIWebView doesn't fire compositionend when
// switching focus before confirming composition choice
// this also fixes the issue where some browsers e.g. iOS Chrome
// fires "change" instead of "input" on autocomplete.
onChange: onComposition,
onKeydown: onTargetKeydown,
onKeyup: onTargetAutocomplete,
onKeypress: onTargetKeypress,
onFocus: selectInputText,
onClick (e) { hasDialog === true && stop(e); }
};
evt.onCompositionstart = evt.onCompositionupdate = evt.onCompositionend = onComposition;
return evt
});
vue.watch(innerValue, val => {
innerValueCache = val;
if (
props.useInput === true
&& props.fillInput === true
&& props.multiple !== true
// Prevent re-entering in filter while filtering
// Also prevent clearing inputValue while filtering
&& state.innerLoading.value !== true
&& ((dialog.value !== true && menu.value !== true) || hasValue.value !== true)
) {
userInputValue !== true && resetInputValue();
if (dialog.value === true || menu.value === true) {
filter('');
}
}
}, { immediate: true });
vue.watch(() => props.fillInput, resetInputValue);
vue.watch(menu, updateMenu);
vue.watch(virtualScrollLength, rerenderMenu);
function getEmittingOptionValue (opt) {
return props.emitValue === true
? getOptionValue.value(opt)
: opt
}
function removeAtIndex (index) {
if (index > -1 && index < innerValue.value.length) {
if (props.multiple === true) {
const model = props.modelValue.slice();
emit('remove', { index, value: model.splice(index, 1)[ 0 ] });
emit('update:modelValue', model);
}
else {
emit('update:modelValue', null);
}
}
}
function removeAtIndexAndFocus (index) {
removeAtIndex(index);
state.focus();
}
function add (opt, unique) {
const val = getEmittingOptionValue(opt);
if (props.multiple !== true) {
props.fillInput === true && updateInputValue(
getOptionLabel.value(opt),
true,
true
);
emit('update:modelValue', val);
return
}
if (innerValue.value.length === 0) {
emit('add', { index: 0, value: val });
emit('update:modelValue', props.multiple === true ? [ val ] : val);
return
}
if (unique === true && isOptionSelected(opt) === true) {
return
}
if (props.maxValues !== void 0 && props.modelValue.length >= props.maxValues) {
return
}
const model = props.modelValue.slice();
emit('add', { index: model.length, value: val });
model.push(val);
emit('update:modelValue', model);
}
function toggleOption (opt, keepOpen) {
if (state.editable.value !== true || opt === void 0 || isOptionDisabled.value(opt) === true) {
return
}
const optValue = getOptionValue.value(opt);
if (props.multiple !== true) {
if (keepOpen !== true) {
updateInputValue(
props.fillInput === true ? getOptionLabel.value(opt) : '',
true,
true
);
hidePopup();
}
targetRef.value !== null && targetRef.value.focus();
if (
innerValue.value.length === 0
|| isDeepEqual(getOptionValue.value(innerValue.value[ 0 ]), optValue) !== true
) {
emit('update:modelValue', props.emitValue === true ? optValue : opt);
}
return
}
(hasDialog !== true || dialogFieldFocused.value === true) && state.focus();
selectInputText();
if (innerValue.value.length === 0) {
const val = props.emitValue === true ? optValue : opt;
emit('add', { index: 0, value: val });
emit('update:modelValue', props.multiple === true ? [ val ] : val);
return
}
const
model = props.modelValue.slice(),
index = innerOptionsValue.value.findIndex(v => isDeepEqual(v, optValue));
if (index > -1) {
emit('remove', { index, value: model.splice(index, 1)[ 0 ] });
}
else {
if (props.maxValues !== void 0 && model.length >= props.maxValues) {
return
}
const val = props.emitValue === true ? optValue : opt;
emit('add', { index: model.length, value: val });
model.push(val);
}
emit('update:modelValue', model);
}
function setOptionIndex (index) {
if ($q.platform.is.desktop !== true) { return }
const val = index > -1 && index < virtualScrollLength.value
? index
: -1;
if (optionIndex.value !== val) {
optionIndex.value = val;
}
}
function moveOptionSelection (offset = 1, skipInputValue) {
if (menu.value === true) {
let index = optionIndex.value;
do {
index = normalizeToInterval(
index + offset,
-1,
virtualScrollLength.value - 1
);
}
while (index !== -1 && index !== optionIndex.value && isOptionDisabled.value(props.options[ index ]) === true)
if (optionIndex.value !== index) {
setOptionIndex(index);
scrollTo(index);
if (skipInputValue !== true && props.useInput === true && props.fillInput === true) {
setInputValue(index >= 0
? getOptionLabel.value(props.options[ index ])
: defaultInputValue
);
}
}
}
}
function getOption (value, valueCache) {
const fn = opt => isDeepEqual(getOptionValue.value(opt), value);
return props.options.find(fn) || valueCache.find(fn) || value
}
function getPropValueFn (propValue, defaultVal) {
const val = propValue !== void 0
? propValue
: defaultVal;
return typeof val === 'function'
? val
: opt => (opt !== null && typeof opt === 'object' && val in opt ? opt[ val ] : opt)
}
function isOptionSelected (opt) {
const val = getOptionValue.value(opt);
return innerOptionsValue.value.find(v => isDeepEqual(v, val)) !== void 0
}
function selectInputText (e) {
if (
props.useInput === true
&& targetRef.value !== null
&& (e === void 0 || (targetRef.value === e.target && e.target.value === selectedString.value))
) {
targetRef.value.select();
}
}
function onTargetKeyup (e) {
// if ESC and we have an opened menu
// then stop propagation (might be caught by a QDialog
// and so it will also close the QDialog, which is wrong)
if (isKeyCode(e, 27) === true && menu.value === true) {
stop(e);
// on ESC we need to close the dialog also
hidePopup();
resetInputValue();
}
emit('keyup', e);
}
function onTargetAutocomplete (e) {
const { value } = e.target;
if (e.keyCode !== void 0) {
onTargetKeyup(e);
return
}
e.target.value = '';
if (inputTimer !== null) {
clearTimeout(inputTimer);
inputTimer = null;
}
resetInputValue();
if (typeof value === 'string' && value.length !== 0) {
const needle = value.toLocaleLowerCase();
const findFn = extractFn => {
const option = props.options.find(opt => extractFn.value(opt).toLocaleLowerCase() === needle);
if (option === void 0) {
return false
}
if (innerValue.value.indexOf(option) === -1) {
toggleOption(option);
}
else {
hidePopup();
}
return true
};
const fillFn = afterFilter => {
if (findFn(getOptionValue) === true) {
return
}
if (findFn(getOptionLabel) === true || afterFilter === true) {
return
}
filter(value, true, () => fillFn(true));
};
fillFn();
}
else {
state.clearValue(e);
}
}
function onTargetKeypress (e) {
emit('keypress', e);
}
function onTargetKeydown (e) {
emit('keydown', e);
if (shouldIgnoreKey(e) === true) {
return
}
const newValueModeValid = inputValue.value.length !== 0
&& (props.newValueMode !== void 0 || props.onNewValue !== void 0);
const tabShouldSelect = e.shiftKey !== true
&& props.multiple !== true
&& (optionIndex.value > -1 || newValueModeValid === true);
// escape
if (e.keyCode === 27) {
prevent(e); // prevent clearing the inputValue
return
}
// tab
if (e.keyCode === 9 && tabShouldSelect === false) {
closeMenu();
return
}
if (
e.target === void 0
|| e.target.id !== state.targetUid.value
|| state.editable.value !== true
) { return }
// down
if (
e.keyCode === 40
&& state.innerLoading.value !== true
&& menu.value === false
) {
stopAndPrevent(e);
showPopup();
return
}
// backspace
if (
e.keyCode === 8
&& props.hideSelected !== true
&& inputValue.value.length === 0
) {
if (props.multiple === true && Array.isArray(props.modelValue) === true) {
removeAtIndex(props.modelValue.length - 1);
}
else if (props.multiple !== true && props.modelValue !== null) {
emit('update:modelValue', null);
}
return
}
// home, end - 36, 35
if (
(e.keyCode === 35 || e.keyCode === 36)
&& (typeof inputValue.value !== 'string' || inputValue.value.length === 0)
) {
stopAndPrevent(e);
optionIndex.value = -1;
moveOptionSelection(e.keyCode === 36 ? 1 : -1, props.multiple);
}
// pg up, pg down - 33, 34
if (
(e.keyCode === 33 || e.keyCode === 34)
&& virtualScrollSliceSizeComputed.value !== void 0
) {
stopAndPrevent(e);
optionIndex.value = Math.max(
-1,
Math.min(
virtualScrollLength.value,
optionIndex.value + (e.keyCode === 33 ? -1 : 1) * virtualScrollSliceSizeComputed.value.view
)
);
moveOptionSelection(e.keyCode === 33 ? 1 : -1, props.multiple);
}
// up, down
if (e.keyCode === 38 || e.keyCode === 40) {
stopAndPrevent(e);
moveOptionSelection(e.keyCode === 38 ? -1 : 1, props.multiple);
}
const optionsLength = virtualScrollLength.value;
// clear search buffer if expired
if (searchBuffer === void 0 || searchBufferExp < Date.now()) {
searchBuffer = '';
}
// keyboard search when not having use-input
if (
optionsLength > 0
&& props.useInput !== true
&& e.key !== void 0
&& e.key.length === 1 // printable char
&& e.altKey === false // not kbd shortcut
&& e.ctrlKey === false // not kbd shortcut
&& e.metaKey === false // not kbd shortcut, especially on macOS with Command key
&& (e.keyCode !== 32 || searchBuffer.length !== 0) // space in middle of search
) {
menu.value !== true && showPopup(e);
const
char = e.key.toLocaleLowerCase(),
keyRepeat = searchBuffer.length === 1 && searchBuffer[ 0 ] === char;
searchBufferExp = Date.now() + 1500;
if (keyRepeat === false) {
stopAndPrevent(e);
searchBuffer += char;
}
const searchRe = new RegExp('^' + searchBuffer.split('').map(l => (reEscapeList.indexOf(l) > -1 ? '\\' + l : l)).join('.*'), 'i');
let index = optionIndex.value;
if (keyRepeat === true || index < 0 || searchRe.test(getOptionLabel.value(props.options[ index ])) !== true) {
do {
index = normalizeToInterval(index + 1, -1, optionsLength - 1);
}
while (index !== optionIndex.value && (
isOptionDisabled.value(props.options[ index ]) === true
|| searchRe.test(getOptionLabel.value(props.options[ index ])) !== true
))
}
if (optionIndex.value !== index) {
vue.nextTick(() => {
setOptionIndex(index);
scrollTo(index);
if (index >= 0 && props.useInput === true && props.fillInput === true) {
setInputValue(getOptionLabel.value(props.options[ index ]));
}
});
}
return
}
// enter, space (when not using use-input and not in search), or tab (when not using multiple and option selected)
// same target is checked above
if (
e.keyCode !== 13
&& (e.keyCode !== 32 || props.useInput === true || searchBuffer !== '')
&& (e.keyCode !== 9 || tabShouldSelect === false)
) { return }
e.keyCode !== 9 && stopAndPrevent(e);
if (optionIndex.value > -1 && optionIndex.value < optionsLength) {
toggleOption(props.options[ optionIndex.value ]);
return
}
if (newValueModeValid === true) {
const done = (val, mode) => {
if (mode) {
if (validateNewValueMode(mode) !== true) {
return
}
}
else {
mode = props.newValueMode;
}
if (val === void 0 || val === null) {
return
}
updateInputValue('', props.multiple !== true, true);
const fn = mode === 'toggle' ? toggleOption : add;
fn(val, mode === 'add-unique');
if (props.multiple !== true) {
targetRef.value !== null && targetRef.value.focus();
hidePopup();
}
};
if (props.onNewValue !== void 0) {
emit('newValue', inputValue.value, done);
}
else {
done(inputValue.value);
}
if (props.multiple !== true) {
return
}
}
if (menu.value === true) {
closeMenu();
}
else if (state.innerLoading.value !== true) {
showPopup();
}
}
function getVirtualScrollEl () {
return hasDialog === true
? menuContentRef.value
: (
menuRef.value !== null && menuRef.value.contentEl !== null
? menuRef.value.contentEl
: void 0
)
}
function getVirtualScrollTarget () {
return getVirtualScrollEl()
}
function getSelection () {
if (props.hideSelected === true) {
return []
}
if (slots[ 'selected-item' ] !== void 0) {
return selectedScope.value.map(scope => slots[ 'selected-item' ](scope)).slice()
}
if (slots.selected !== void 0) {
return [].concat(slots.selected())
}
if (props.useChips === true) {
return selectedScope.value.map((scope, i) => vue.h(QChip, {
key: 'option-' + i,
removable: state.editable.value === true && isOptionDisabled.value(scope.opt) !== true,
dense: true,
textColor: props.color,
tabindex: tabindex.value,
onRemove () { scope.removeAtIndex(i); }
}, () => vue.h('span', {
class: 'ellipsis',
[ scope.html === true ? 'innerHTML' : 'textContent' ]: getOptionLabel.value(scope.opt)
})))
}
return [
vue.h('span', {
[ valueAsHtml.value === true ? 'innerHTML' : 'textContent' ]: ariaCurrentValue.value
})
]
}
function getAllOptions () {
if (noOptions.value === true) {
return slots[ 'no-option' ] !== void 0
? slots[ 'no-option' ]({ inputValue: inputValue.value })
: void 0
}
const fn = slots.option !== void 0
? slots.option
: scope => {
return vue.h(QItem, {
key: scope.index,
...scope.itemProps
}, () => {
return vue.h(
QItemSection,
() => vue.h(
QItemLabel,
() => vue.h('span', {
[ scope.html === true ? 'innerHTML' : 'textContent' ]: scope.label
})
)
)
})
};
let options = padVirtualScroll('div', optionScope.value.map(fn));
if (slots[ 'before-options' ] !== void 0) {
options = slots[ 'before-options' ]().concat(options);
}
return hMergeSlot(slots[ 'after-options' ], options)
}
function getInput (fromDialog, isTarget) {
const attrs = isTarget === true ? { ...comboboxAttrs.value, ...state.splitAttrs.attributes.value } : void 0;
const data = {
ref: isTarget === true ? targetRef : void 0,
key: 'i_t',
class: computedInputClass.value,
style: props.inputStyle,
value: inputValue.value !== void 0 ? inputValue.value : '',
// required for Android in order to show ENTER key when in form
type: 'search',
...attrs,
id: isTarget === true ? state.targetUid.value : void 0,
maxlength: props.maxlength,
autocomplete: props.autocomplete,
'data-autofocus': fromDialog === true || props.autofocus === true || void 0,
disabled: props.disable === true,
readonly: props.readonly === true,
...inputControlEvents.value
};
if (fromDialog !== true && hasDialog === true) {
if (Array.isArray(data.class) === true) {
data.class = [ ...data.class, 'no-pointer-events' ];
}
else {
data.class += ' no-pointer-events';
}
}
return vue.h('input', data)
}
function onInput (e) {
if (inputTimer !== null) {
clearTimeout(inputTimer);
inputTimer = null;
}
if (e && e.target && e.target.qComposing === true) {
return
}
setInputValue(e.target.value || '');
// mark it here as user input so that if updateInputValue is called
// before filter is called the indicator is reset
userInputValue = true;
defaultInputValue = inputValue.value;
if (
state.focused.value !== true
&& (hasDialog !== true || dialogFieldFocused.value === true)
) {
state.focus();
}
if (props.onFilter !== void 0) {
inputTimer = setTimeout(() => {
inputTimer = null;
filter(inputValue.value);
}, props.inputDebounce);
}
}
function setInputValue (val) {
if (inputValue.value !== val) {
inputValue.value = val;
emit('inputValue', val);
}
}
function updateInputValue (val, noFiltering, internal) {
userInputValue = internal !== true;
if (props.useInput === true) {
setInputValue(val);
if (noFiltering === true || internal !== true) {
defaultInputValue = val;
}
noFiltering !== true && filter(val);
}
}
function filter (val, keepClosed, afterUpdateFn) {
if (props.onFilter === void 0 || (keepClosed !== true && state.focused.value !== true)) {
return
}
if (state.innerLoading.value === true) {
emit('filterAbort');
}
else {
state.innerLoading.value = true;
innerLoadingIndicator.value = true;
}
if (
val !== ''
&& props.multiple !== true
&& innerValue.value.length !== 0
&& userInputValue !== true
&& val === getOptionLabel.value(innerValue.value[ 0 ])
) {
val = '';
}
const localFilterId = setTimeout(() => {
menu.value === true && (menu.value = false);
}, 10);
filterId !== null && clearTimeout(filterId);
filterId = localFilterId;
emit(
'filter',
val,
(fn, afterFn) => {
if ((keepClosed === true || state.focused.value === true) && filterId === localFilterId) {
clearTimeout(filterId);
typeof fn === 'function' && fn();
// hide indicator to allow arrow to animate
innerLoadingIndicator.value = false;
vue.nextTick(() => {
state.innerLoading.value = false;
if (state.editable.value === true) {
if (keepClosed === true) {
menu.value === true && hidePopup();
}
else if (menu.value === true) {
updateMenu(true);
}
else {
menu.value = true;
}
}
typeof afterFn === 'function' && vue.nextTick(() => { afterFn(proxy); });
typeof afterUpdateFn === 'function' && vue.nextTick(() => { afterUpdateFn(proxy); });
});
}
},
() => {
if (state.focused.value === true && filterId === localFilterId) {
clearTimeout(filterId);
state.innerLoading.value = false;
innerLoadingIndicator.value = false;
}
menu.value === true && (menu.value = false);
}
);
}
function getMenu () {
return vue.h(QMenu, {
ref: menuRef,
class: menuContentClass.value,
style: props.popupContentStyle,
modelValue: menu.value,
fit: props.menuShrink !== true,
cover: props.optionsCover === true && noOptions.value !== true && props.useInput !== true,
anchor: props.menuAnchor,
self: props.menuSelf,
offset: props.menuOffset,
dark: isOptionsDark.value,
noParentEvent: true,
noRefocus: true,
noFocus: true,
square: squaredMenu.value,
transitionShow: props.transitionShow,
transitionHide: props.transitionHide,
transitionDuration: props.transitionDuration,
separateClosePopup: true,
...listboxAttrs.value,
onScrollPassive: onVirtualScrollEvt,
onBeforeShow: onControlPopupShow,
onBeforeHide: onMenuBeforeHide,
onShow: onMenuShow
}, getAllOptions)
}
function onMenuBeforeHide (e) {
onControlPopupHide(e);
closeMenu();
}
function onMenuShow () {
setVirtualScrollSize();
}
function onDialogFieldFocus (e) {
stop(e);
targetRef.value !== null && targetRef.value.focus();
dialogFieldFocused.value = true;
window.scrollTo(window.pageXOffset || window.scrollX || document.body.scrollLeft || 0, 0);
}
function onDialogFieldBlur (e) {
stop(e);
vue.nextTick(() => {
dialogFieldFocused.value = false;
});
}
function getDialog () {
const content = [
vue.h(QField, {
class: `col-auto ${ state.fieldClass.value }`,
...innerFieldProps.value,
for: state.targetUid.value,
dark: isOptionsDark.value,
square: true,
loading: innerLoadingIndicator.value,
itemAligned: false,
filled: true,
stackLabel: inputValue.value.length !== 0,
...state.splitAttrs.listeners.value,
onFocus: onDialogFieldFocus,
onBlur: onDialogFieldBlur
}, {
...slots,
rawControl: () => state.getControl(true),
before: void 0,
after: void 0
})
];
menu.value === true && content.push(
vue.h('div', {
ref: menuContentRef,
class: menuContentClass.value + ' scroll',
style: props.popupContentStyle,
...listboxAttrs.value,
onClick: prevent,
onScrollPassive: onVirtualScrollEvt
}, getAllOptions())
);
return vue.h(QDialog, {
ref: dialogRef,
modelValue: dialog.value,
position: props.useInput === true ? 'top' : void 0,
transitionShow: transitionShowComputed,
transitionHide: props.transitionHide,
transitionDuration: props.transitionDuration,
onBeforeShow: onControlPopupShow,
onBeforeHide: onDialogBeforeHide,
onHide: onDialogHide,
onShow: onDialogShow
}, () => vue.h('div', {
class: 'q-select__dialog'
+ (isOptionsDark.value === true ? ' q-select__dialog--dark q-dark' : '')
+ (dialogFieldFocused.value === true ? ' q-select__dialog--focused' : '')
}, content))
}
function onDialogBeforeHide (e) {
onControlPopupHide(e);
if (dialogRef.value !== null) {
dialogRef.value.__updateRefocusTarget(
state.rootRef.value.querySelector('.q-field__native > [tabindex]:last-child')
);
}
state.focused.value = false;
}
function onDialogHide (e) {
hidePopup();
state.focused.value === false && emit('blur', e);
resetInputValue();
}
function onDialogShow () {
const el = document.activeElement;
if (
(el === null || el.id !== state.targetUid.value)
&& targetRef.value !== null
&& targetRef.value !== el
) {
targetRef.value.focus();
}
setVirtualScrollSize();
}
function closeMenu () {
if (dialog.value === true) {
return
}
optionIndex.value = -1;
if (menu.value === true) {
menu.value = false;
}
if (state.focused.value === false) {
if (filterId !== null) {
clearTimeout(filterId);
filterId = null;
}
if (state.innerLoading.value === true) {
emit('filterAbort');
state.innerLoading.value = false;
innerLoadingIndicator.value = false;
}
}
}
function showPopup (e) {
if (state.editable.value !== true) {
return
}
if (hasDialog === true) {
state.onControlFocusin(e);
dialog.value = true;
vue.nextTick(() => {
state.focus();
});
}
else {
state.focus();
}
if (props.onFilter !== void 0) {
filter(inputValue.value);
}
else if (noOptions.value !== true || slots[ 'no-option' ] !== void 0) {
menu.value = true;
}
}
function hidePopup () {
dialog.value = false;
closeMenu();
}
function resetInputValue () {
props.useInput === true && updateInputValue(
props.multiple !== true && props.fillInput === true && innerValue.value.length !== 0
? getOptionLabel.value(innerValue.value[ 0 ]) || ''
: '',
true,
true
);
}
function updateMenu (show) {
let optionIndex = -1;
if (show === true) {
if (innerValue.value.length !== 0) {
const val = getOptionValue.value(innerValue.value[ 0 ]);
optionIndex = props.options.findIndex(v => isDeepEqual(getOptionValue.value(v), val));
}
localResetVirtualScroll(optionIndex);
}
setOptionIndex(optionIndex);
}
function rerenderMenu (newLength, oldLength) {
if (menu.value === true && state.innerLoading.value === false) {
localResetVirtualScroll(-1, true);
vue.nextTick(() => {
if (menu.value === true && state.innerLoading.value === false) {
if (newLength > oldLength) {
localResetVirtualScroll();
}
else {
updateMenu(true);
}
}
});
}
}
function updateMenuPosition () {
if (dialog.value === false && menuRef.value !== null) {
menuRef.value.updatePosition();
}
}
function onControlPopupShow (e) {
e !== void 0 && stop(e);
emit('popupShow', e);
state.hasPopupOpen = true;
state.onControlFocusin(e);
}
function onControlPopupHide (e) {
e !== void 0 && stop(e);
emit('popupHide', e);
state.hasPopupOpen = false;
state.onControlFocusout(e);
}
function updatePreState () {
hasDialog = $q.platform.is.mobile !== true && props.behavior !== 'dialog'
? false
: props.behavior !== 'menu' && (
props.useInput === true
? slots[ 'no-option' ] !== void 0 || props.onFilter !== void 0 || noOptions.value === false
: true
);
transitionShowComputed = $q.platform.is.ios === true && hasDialog === true && props.useInput === true
? 'fade'
: props.transitionShow;
}
vue.onBeforeUpdate(updatePreState);
vue.onUpdated(updateMenuPosition);
updatePreState();
vue.onBeforeUnmount(() => {
inputTimer !== null && clearTimeout(inputTimer);
});
// expose public methods
Object.assign(proxy, {
showPopup, hidePopup,
removeAtIndex, add, toggleOption,
getOptionIndex: () => optionIndex.value,
setOptionIndex, moveOptionSelection,
filter, updateMenuPosition, updateInputValue,
isOptionSelected,
getEmittingOptionValue,
isOptionDisabled: (...args) => isOptionDisabled.value.apply(null, args) === true,
getOptionValue: (...args) => getOptionValue.value.apply(null, args),
getOptionLabel: (...args) => getOptionLabel.value.apply(null, args)
});
Object.assign(state, {
innerValue,
fieldClass: vue.computed(() =>
`q-select q-field--auto-height q-select--with${ props.useInput !== true ? 'out' : '' }-input`
+ ` q-select--with${ props.useChips !== true ? 'out' : '' }-chips`
+ ` q-select--${ props.multiple === true ? 'multiple' : 'single' }`
),
inputRef,
targetRef,
hasValue,
showPopup,
floatingLabel: vue.computed(() =>
(props.hideSelected !== true && hasValue.value === true)
|| typeof inputValue.value === 'number'
|| inputValue.value.length !== 0
|| fieldValueIsFilled(props.displayValue)
),
getControlChild: () => {
if (
state.editable.value !== false && (
dialog.value === true // dialog always has menu displayed, so need to render it
|| noOptions.value !== true
|| slots[ 'no-option' ] !== void 0
)
) {
return hasDialog === true ? getDialog() : getMenu()
}
else if (state.hasPopupOpen === true) {
// explicitly set it otherwise TAB will not blur component
state.hasPopupOpen = false;
}
},
controlEvents: {
onFocusin (e) { state.onControlFocusin(e); },
onFocusout (e) {
state.onControlFocusout(e, () => {
resetInputValue();
closeMenu();
});
},
onClick (e) {
// label from QField will propagate click on the input
prevent(e);
if (hasDialog !== true && menu.value === true) {
closeMenu();
targetRef.value !== null && targetRef.value.focus();
return
}
showPopup(e);
}
},
getControl: fromDialog => {
const child = getSelection();
const isTarget = fromDialog === true || dialog.value !== true || hasDialog !== true;
if (props.useInput === true) {
child.push(getInput(fromDialog, isTarget));
}
// there can be only one (when dialog is opened the control in dialog should be target)
else if (state.editable.value === true) {
const attrs = isTarget === true ? comboboxAttrs.value : void 0;
child.push(
vue.h('input', {
ref: isTarget === true ? targetRef : void 0,
key: 'd_t',
class: 'q-select__focus-target',
id: isTarget === true ? state.targetUid.value : void 0,
value: ariaCurrentValue.value,
readonly: true,
'data-autofocus': fromDialog === true || props.autofocus === true || void 0,
...attrs,
onKeydown: onTargetKeydown,
onKeyup: onTargetKeyup,
onKeypress: onTargetKeypress
})
);
if (isTarget === true && typeof props.autocomplete === 'string' && props.autocomplete.length !== 0) {
child.push(
vue.h('input', {
class: 'q-select__autocomplete-input',
autocomplete: props.autocomplete,
tabindex: -1,
onKeyup: onTargetAutocomplete
})
);
}
}
if (nameProp.value !== void 0 && props.disable !== true && innerOptionsValue.value.length !== 0) {
const opts = innerOptionsValue.value.map(value => vue.h('option', { value, selected: true }));
child.push(
vue.h('select', {
class: 'hidden',
name: nameProp.value,
multiple: props.multiple
}, opts)
);
}
const attrs = props.useInput === true || isTarget !== true ? void 0 : state.splitAttrs.attributes.value;
return vue.h('div', {
class: 'q-field__native row items-center',
...attrs,
...state.splitAttrs.listeners.value
}, child)
},
getInnerAppend: () => (
props.loading !== true && innerLoadingIndicator.value !== true && props.hideDropdownIcon !== true
? [
vue.h(QIcon, {
class: 'q-select__dropdown-icon' + (menu.value === true ? ' rotate-180' : ''),
name: dropdownArrowIcon.value
})
]
: null
)
});
return useField(state)
}
});
const skeletonTypes = [
'text', 'rect', 'circle',
'QBtn', 'QBadge', 'QChip', 'QToolbar',
'QCheckbox', 'QRadio', 'QToggle',
'QSlider', 'QRange', 'QInput',
'QAvatar'
];
const skeletonAnimations = [
'wave', 'pulse', 'pulse-x', 'pulse-y', 'fade', 'blink', 'none'
];
var QSkeleton = createComponent({
name: 'QSkeleton',
props: {
...useDarkProps,
tag: {
type: String,
default: 'div'
},
type: {
type: String,
validator: v => skeletonTypes.includes(v),
default: 'rect'
},
animation: {
type: String,
validator: v => skeletonAnimations.includes(v),
default: 'wave'
},
animationSpeed: {
type: [ String, Number ],
default: 1500
},
square: Boolean,
bordered: Boolean,
size: String,
width: String,
height: String
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const style = vue.computed(() => {
const size = props.size !== void 0
? [ props.size, props.size ]
: [ props.width, props.height ];
return {
'--q-skeleton-speed': `${ props.animationSpeed }ms`,
width: size[ 0 ],
height: size[ 1 ]
}
});
const classes = vue.computed(() =>
`q-skeleton q-skeleton--${ isDark.value === true ? 'dark' : 'light' } q-skeleton--type-${ props.type }`
+ (props.animation !== 'none' ? ` q-skeleton--anim q-skeleton--anim-${ props.animation }` : '')
+ (props.square === true ? ' q-skeleton--square' : '')
+ (props.bordered === true ? ' q-skeleton--bordered' : '')
);
return () => vue.h(props.tag, {
class: classes.value,
style: style.value
}, hSlot(slots.default))
}
});
const slotsDef = [
[ 'left', 'center', 'start', 'width' ],
[ 'right', 'center', 'end', 'width' ],
[ 'top', 'start', 'center', 'height' ],
[ 'bottom', 'end', 'center', 'height' ]
];
var QSlideItem = createComponent({
name: 'QSlideItem',
props: {
...useDarkProps,
leftColor: String,
rightColor: String,
topColor: String,
bottomColor: String,
onSlide: Function
},
emits: [ 'action', 'top', 'right', 'bottom', 'left' ],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const { getCacheWithFn } = useCache();
const contentRef = vue.ref(null);
let timer = null, pan = {}, dirRefs = {}, dirContentRefs = {};
const langDir = vue.computed(() => (
$q.lang.rtl === true
? { left: 'right', right: 'left' }
: { left: 'left', right: 'right' }
));
const classes = vue.computed(() =>
'q-slide-item q-item-type overflow-hidden'
+ (isDark.value === true ? ' q-slide-item--dark q-dark' : '')
);
function reset () {
contentRef.value.style.transform = 'translate(0,0)';
}
function emitSlide (side, ratio, isReset) {
props.onSlide !== void 0 && emit('slide', { side, ratio, isReset });
}
function onPan (evt) {
const node = contentRef.value;
if (evt.isFirst) {
pan = {
dir: null,
size: { left: 0, right: 0, top: 0, bottom: 0 },
scale: 0
};
node.classList.add('no-transition');
slotsDef.forEach(slotName => {
if (slots[ slotName[ 0 ] ] !== void 0) {
const node = dirContentRefs[ slotName[ 0 ] ];
node.style.transform = 'scale(1)';
pan.size[ slotName[ 0 ] ] = node.getBoundingClientRect()[ slotName[ 3 ] ];
}
});
pan.axis = (evt.direction === 'up' || evt.direction === 'down')
? 'Y'
: 'X';
}
else if (evt.isFinal) {
node.classList.remove('no-transition');
if (pan.scale === 1) {
node.style.transform = `translate${ pan.axis }(${ pan.dir * 100 }%)`;
timer !== null && clearTimeout(timer);
timer = setTimeout(() => {
timer = null;
emit(pan.showing, { reset });
emit('action', { side: pan.showing, reset });
}, 230);
}
else {
node.style.transform = 'translate(0,0)';
emitSlide(pan.showing, 0, true);
}
return
}
else {
evt.direction = pan.axis === 'X'
? evt.offset.x < 0 ? 'left' : 'right'
: evt.offset.y < 0 ? 'up' : 'down';
}
if (
(slots.left === void 0 && evt.direction === langDir.value.right)
|| (slots.right === void 0 && evt.direction === langDir.value.left)
|| (slots.top === void 0 && evt.direction === 'down')
|| (slots.bottom === void 0 && evt.direction === 'up')
) {
node.style.transform = 'translate(0,0)';
return
}
let showing, dir, dist;
if (pan.axis === 'X') {
dir = evt.direction === 'left' ? -1 : 1;
showing = dir === 1 ? langDir.value.left : langDir.value.right;
dist = evt.distance.x;
}
else {
dir = evt.direction === 'up' ? -2 : 2;
showing = dir === 2 ? 'top' : 'bottom';
dist = evt.distance.y;
}
if (pan.dir !== null && Math.abs(dir) !== Math.abs(pan.dir)) {
return
}
if (pan.dir !== dir) {
[ 'left', 'right', 'top', 'bottom' ].forEach(d => {
if (dirRefs[ d ]) {
dirRefs[ d ].style.visibility = showing === d
? 'visible'
: 'hidden';
}
});
pan.showing = showing;
pan.dir = dir;
}
pan.scale = Math.max(0, Math.min(1, (dist - 40) / pan.size[ showing ]));
node.style.transform = `translate${ pan.axis }(${ dist * dir / Math.abs(dir) }px)`;
dirContentRefs[ showing ].style.transform = `scale(${ pan.scale })`;
emitSlide(showing, pan.scale, false);
}
vue.onBeforeUpdate(() => {
dirRefs = {};
dirContentRefs = {};
});
vue.onBeforeUnmount(() => {
timer !== null && clearTimeout(timer);
});
// expose public methods
Object.assign(proxy, { reset });
return () => {
const
content = [],
slotsList = {
left: slots[ langDir.value.right ] !== void 0,
right: slots[ langDir.value.left ] !== void 0,
up: slots.bottom !== void 0,
down: slots.top !== void 0
},
dirs = Object.keys(slotsList).filter(key => slotsList[ key ] === true);
slotsDef.forEach(slotName => {
const dir = slotName[ 0 ];
if (slots[ dir ] !== void 0) {
content.push(
vue.h('div', {
ref: el => { dirRefs[ dir ] = el; },
class: `q-slide-item__${ dir } absolute-full row no-wrap items-${ slotName[ 1 ] } justify-${ slotName[ 2 ] }`
+ (props[ dir + 'Color' ] !== void 0 ? ` bg-${ props[ dir + 'Color' ] }` : '')
}, [
vue.h('div', { ref: el => { dirContentRefs[ dir ] = el; } }, slots[ dir ]())
])
);
}
});
const node = vue.h('div', {
key: `${ dirs.length === 0 ? 'only-' : '' } content`,
ref: contentRef,
class: 'q-slide-item__content'
}, hSlot(slots.default));
if (dirs.length === 0) {
content.push(node);
}
else {
content.push(
vue.withDirectives(node, getCacheWithFn('dir#' + dirs.join(''), () => {
const modifiers = {
prevent: true,
stop: true,
mouse: true
};
dirs.forEach(dir => {
modifiers[ dir ] = true;
});
return [ [
TouchPan,
onPan,
void 0,
modifiers
] ]
}))
);
}
return vue.h('div', { class: classes.value }, content)
}
}
});
const space = vue.h('div', { class: 'q-space' });
var QSpace = createComponent({
name: 'QSpace',
setup () {
return () => space
}
});
const svg$l = [
vue.h('g', {
transform: 'matrix(1 0 0 -1 0 80)'
}, [
vue.h('rect', {
width: '10',
height: '20',
rx: '3'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0s',
dur: '4.3s',
values: '20;45;57;80;64;32;66;45;64;23;66;13;64;56;34;34;2;23;76;79;20',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '15',
width: '10',
height: '80',
rx: '3'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0s',
dur: '2s',
values: '80;55;33;5;75;23;73;33;12;14;60;80',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '30',
width: '10',
height: '50',
rx: '3'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0s',
dur: '1.4s',
values: '50;34;78;23;56;23;34;76;80;54;21;50',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '45',
width: '10',
height: '30',
rx: '3'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0s',
dur: '2s',
values: '30;45;13;80;56;72;45;76;34;23;67;30',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerAudio = createComponent({
name: 'QSpinnerAudio',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
fill: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 55 80',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$l)
}
});
const svg$k = [
vue.h('g', {
transform: 'translate(1 1)',
'stroke-width': '2',
fill: 'none',
'fill-rule': 'evenodd'
}, [
vue.h('circle', {
cx: '5',
cy: '50',
r: '5'
}, [
vue.h('animate', {
attributeName: 'cy',
begin: '0s',
dur: '2.2s',
values: '50;5;50;50',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'cx',
begin: '0s',
dur: '2.2s',
values: '5;27;49;5',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '27',
cy: '5',
r: '5'
}, [
vue.h('animate', {
attributeName: 'cy',
begin: '0s',
dur: '2.2s',
from: '5',
to: '5',
values: '5;50;50;5',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'cx',
begin: '0s',
dur: '2.2s',
from: '27',
to: '27',
values: '27;49;5;27',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '49',
cy: '50',
r: '5'
}, [
vue.h('animate', {
attributeName: 'cy',
begin: '0s',
dur: '2.2s',
values: '50;50;5;50',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'cx',
from: '49',
to: '49',
begin: '0s',
dur: '2.2s',
values: '49;5;27;49',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerBall = createComponent({
name: 'QSpinnerBall',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
stroke: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 57 57',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$k)
}
});
const svg$j = [
vue.h('rect', {
y: '10',
width: '15',
height: '120',
rx: '6'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0.5s',
dur: '1s',
values: '120;110;100;90;80;70;60;50;40;140;120',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'y',
begin: '0.5s',
dur: '1s',
values: '10;15;20;25;30;35;40;45;50;0;10',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '30',
y: '10',
width: '15',
height: '120',
rx: '6'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0.25s',
dur: '1s',
values: '120;110;100;90;80;70;60;50;40;140;120',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'y',
begin: '0.25s',
dur: '1s',
values: '10;15;20;25;30;35;40;45;50;0;10',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '60',
width: '15',
height: '140',
rx: '6'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0s',
dur: '1s',
values: '120;110;100;90;80;70;60;50;40;140;120',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'y',
begin: '0s',
dur: '1s',
values: '10;15;20;25;30;35;40;45;50;0;10',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '90',
y: '10',
width: '15',
height: '120',
rx: '6'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0.25s',
dur: '1s',
values: '120;110;100;90;80;70;60;50;40;140;120',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'y',
begin: '0.25s',
dur: '1s',
values: '10;15;20;25;30;35;40;45;50;0;10',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('rect', {
x: '120',
y: '10',
width: '15',
height: '120',
rx: '6'
}, [
vue.h('animate', {
attributeName: 'height',
begin: '0.5s',
dur: '1s',
values: '120;110;100;90;80;70;60;50;40;140;120',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'y',
begin: '0.5s',
dur: '1s',
values: '10;15;20;25;30;35;40;45;50;0;10',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
];
var QSpinnerBars = createComponent({
name: 'QSpinnerBars',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
fill: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 135 140',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$j)
}
});
const svg$i = [
vue.h('rect', {
x: '25',
y: '25',
width: '50',
height: '50',
fill: 'none',
'stroke-width': '4',
stroke: 'currentColor'
}, [
vue.h('animateTransform', {
id: 'spinnerBox',
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '180 50 50',
dur: '0.5s',
begin: 'rectBox.end'
})
]),
vue.h('rect', {
x: '27',
y: '27',
width: '46',
height: '50',
fill: 'currentColor'
}, [
vue.h('animate', {
id: 'rectBox',
attributeName: 'height',
begin: '0s;spinnerBox.end',
dur: '1.3s',
from: '50',
to: '0',
fill: 'freeze'
})
])
];
var QSpinnerBox = createComponent({
name: 'QSpinnerBox',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$i)
}
});
const svg$h = [
vue.h('circle', {
cx: '50',
cy: '50',
r: '48',
fill: 'none',
'stroke-width': '4',
'stroke-miterlimit': '10',
stroke: 'currentColor'
}),
vue.h('line', {
'stroke-linecap': 'round',
'stroke-width': '4',
'stroke-miterlimit': '10',
stroke: 'currentColor',
x1: '50',
y1: '50',
x2: '85',
y2: '50.5'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '2s',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
'stroke-linecap': 'round',
'stroke-width': '4',
'stroke-miterlimit': '10',
stroke: 'currentColor',
x1: '50',
y1: '50',
x2: '49.5',
y2: '74'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '15s',
repeatCount: 'indefinite'
})
])
];
var QSpinnerClock = createComponent({
name: 'QSpinnerClock',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$h)
}
});
const svg$g = [
vue.h('rect', {
x: '0',
y: '0',
width: ' 100',
height: '100',
fill: 'none'
}),
vue.h('path', {
d: 'M78,19H22c-6.6,0-12,5.4-12,12v31c0,6.6,5.4,12,12,12h37.2c0.4,3,1.8,5.6,3.7,7.6c2.4,2.5,5.1,4.1,9.1,4 c-1.4-2.1-2-7.2-2-10.3c0-0.4,0-0.8,0-1.3h8c6.6,0,12-5.4,12-12V31C90,24.4,84.6,19,78,19z',
fill: 'currentColor'
}),
vue.h('circle', {
cx: '30',
cy: '47',
r: '5',
fill: '#fff'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
values: '0;1;1',
keyTimes: '0;0.2;1',
dur: '1s',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '50',
cy: '47',
r: '5',
fill: '#fff'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
values: '0;0;1;1',
keyTimes: '0;0.2;0.4;1',
dur: '1s',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '70',
cy: '47',
r: '5',
fill: '#fff'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
values: '0;0;1;1',
keyTimes: '0;0.4;0.6;1',
dur: '1s',
repeatCount: 'indefinite'
})
])
];
var QSpinnerComment = createComponent({
name: 'QSpinnerComment',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
xmlns: 'http://www.w3.org/2000/svg',
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid'
}, svg$g)
}
});
const svg$f = [
vue.h('rect', {
x: '0',
y: '0',
width: ' 100',
height: '100',
fill: 'none'
}),
vue.h('g', {
transform: 'translate(25 25)'
}, [
vue.h('rect', {
x: '-20',
y: '-20',
width: ' 40',
height: '40',
fill: 'currentColor',
opacity: '0.9'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '1.5',
to: '1',
repeatCount: 'indefinite',
begin: '0s',
dur: '1s',
calcMode: 'spline',
keySplines: '0.2 0.8 0.2 0.8',
keyTimes: '0;1'
})
])
]),
vue.h('g', {
transform: 'translate(75 25)'
}, [
vue.h('rect', {
x: '-20',
y: '-20',
width: ' 40',
height: '40',
fill: 'currentColor',
opacity: '0.8'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '1.5',
to: '1',
repeatCount: 'indefinite',
begin: '0.1s',
dur: '1s',
calcMode: 'spline',
keySplines: '0.2 0.8 0.2 0.8',
keyTimes: '0;1'
})
])
]),
vue.h('g', {
transform: 'translate(25 75)'
}, [
vue.h('rect', {
x: '-20',
y: '-20',
width: ' 40',
height: '40',
fill: 'currentColor',
opacity: '0.7'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '1.5',
to: '1',
repeatCount: 'indefinite',
begin: '0.3s',
dur: '1s',
calcMode: 'spline',
keySplines: '0.2 0.8 0.2 0.8',
keyTimes: '0;1'
})
])
]),
vue.h('g', {
transform: 'translate(75 75)'
}, [
vue.h('rect', {
x: '-20',
y: '-20',
width: ' 40',
height: '40',
fill: 'currentColor',
opacity: '0.6'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '1.5',
to: '1',
repeatCount: 'indefinite',
begin: '0.2s',
dur: '1s',
calcMode: 'spline',
keySplines: '0.2 0.8 0.2 0.8',
keyTimes: '0;1'
})
])
])
];
var QSpinnerCube = createComponent({
name: 'QSpinnerCube',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
xmlns: 'http://www.w3.org/2000/svg',
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid'
}, svg$f)
}
});
const svg$e = [
vue.h('circle', {
cx: '15',
cy: '15',
r: '15'
}, [
vue.h('animate', {
attributeName: 'r',
from: '15',
to: '15',
begin: '0s',
dur: '0.8s',
values: '15;9;15',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'fill-opacity',
from: '1',
to: '1',
begin: '0s',
dur: '0.8s',
values: '1;.5;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '60',
cy: '15',
r: '9',
'fill-opacity': '.3'
}, [
vue.h('animate', {
attributeName: 'r',
from: '9',
to: '9',
begin: '0s',
dur: '0.8s',
values: '9;15;9',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'fill-opacity',
from: '.5',
to: '.5',
begin: '0s',
dur: '0.8s',
values: '.5;1;.5',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '105',
cy: '15',
r: '15'
}, [
vue.h('animate', {
attributeName: 'r',
from: '15',
to: '15',
begin: '0s',
dur: '0.8s',
values: '15;9;15',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'fill-opacity',
from: '1',
to: '1',
begin: '0s',
dur: '0.8s',
values: '1;.5;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
];
var QSpinnerDots = createComponent({
name: 'QSpinnerDots',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
fill: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 120 30',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$e)
}
});
const svg$d = [
vue.h('g', {
transform: 'translate(20 50)'
}, [
vue.h('rect', {
x: '-10',
y: '-30',
width: ' 20',
height: '60',
fill: 'currentColor',
opacity: '0.6'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '2',
to: '1',
begin: '0s',
repeatCount: 'indefinite',
dur: '1s',
calcMode: 'spline',
keySplines: '0.1 0.9 0.4 1',
keyTimes: '0;1',
values: '2;1'
})
])
]),
vue.h('g', {
transform: 'translate(50 50)'
}, [
vue.h('rect', {
x: '-10',
y: '-30',
width: ' 20',
height: '60',
fill: 'currentColor',
opacity: '0.8'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '2',
to: '1',
begin: '0.1s',
repeatCount: 'indefinite',
dur: '1s',
calcMode: 'spline',
keySplines: '0.1 0.9 0.4 1',
keyTimes: '0;1',
values: '2;1'
})
])
]),
vue.h('g', {
transform: 'translate(80 50)'
}, [
vue.h('rect', {
x: '-10',
y: '-30',
width: ' 20',
height: '60',
fill: 'currentColor',
opacity: '0.9'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'scale',
from: '2',
to: '1',
begin: '0.2s',
repeatCount: 'indefinite',
dur: '1s',
calcMode: 'spline',
keySplines: '0.1 0.9 0.4 1',
keyTimes: '0;1',
values: '2;1'
})
])
])
];
var QSpinnerFacebook = createComponent({
name: 'QSpinnerFacebook',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
xmlns: 'http://www.w3.org/2000/svg',
preserveAspectRatio: 'xMidYMid'
}, svg$d)
}
});
const svg$c = [
vue.h('g', {
transform: 'translate(-20,-20)'
}, [
vue.h('path', {
d: 'M79.9,52.6C80,51.8,80,50.9,80,50s0-1.8-0.1-2.6l-5.1-0.4c-0.3-2.4-0.9-4.6-1.8-6.7l4.2-2.9c-0.7-1.6-1.6-3.1-2.6-4.5 L70,35c-1.4-1.9-3.1-3.5-4.9-4.9l2.2-4.6c-1.4-1-2.9-1.9-4.5-2.6L59.8,27c-2.1-0.9-4.4-1.5-6.7-1.8l-0.4-5.1C51.8,20,50.9,20,50,20 s-1.8,0-2.6,0.1l-0.4,5.1c-2.4,0.3-4.6,0.9-6.7,1.8l-2.9-4.1c-1.6,0.7-3.1,1.6-4.5,2.6l2.1,4.6c-1.9,1.4-3.5,3.1-5,4.9l-4.5-2.1 c-1,1.4-1.9,2.9-2.6,4.5l4.1,2.9c-0.9,2.1-1.5,4.4-1.8,6.8l-5,0.4C20,48.2,20,49.1,20,50s0,1.8,0.1,2.6l5,0.4 c0.3,2.4,0.9,4.7,1.8,6.8l-4.1,2.9c0.7,1.6,1.6,3.1,2.6,4.5l4.5-2.1c1.4,1.9,3.1,3.5,5,4.9l-2.1,4.6c1.4,1,2.9,1.9,4.5,2.6l2.9-4.1 c2.1,0.9,4.4,1.5,6.7,1.8l0.4,5.1C48.2,80,49.1,80,50,80s1.8,0,2.6-0.1l0.4-5.1c2.3-0.3,4.6-0.9,6.7-1.8l2.9,4.2 c1.6-0.7,3.1-1.6,4.5-2.6L65,69.9c1.9-1.4,3.5-3,4.9-4.9l4.6,2.2c1-1.4,1.9-2.9,2.6-4.5L73,59.8c0.9-2.1,1.5-4.4,1.8-6.7L79.9,52.6 z M50,65c-8.3,0-15-6.7-15-15c0-8.3,6.7-15,15-15s15,6.7,15,15C65,58.3,58.3,65,50,65z',
fill: 'currentColor'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '90 50 50',
to: '0 50 50',
dur: '1s',
repeatCount: 'indefinite'
})
])
]),
vue.h('g', {
transform: 'translate(20,20) rotate(15 50 50)'
}, [
vue.h('path', {
d: 'M79.9,52.6C80,51.8,80,50.9,80,50s0-1.8-0.1-2.6l-5.1-0.4c-0.3-2.4-0.9-4.6-1.8-6.7l4.2-2.9c-0.7-1.6-1.6-3.1-2.6-4.5 L70,35c-1.4-1.9-3.1-3.5-4.9-4.9l2.2-4.6c-1.4-1-2.9-1.9-4.5-2.6L59.8,27c-2.1-0.9-4.4-1.5-6.7-1.8l-0.4-5.1C51.8,20,50.9,20,50,20 s-1.8,0-2.6,0.1l-0.4,5.1c-2.4,0.3-4.6,0.9-6.7,1.8l-2.9-4.1c-1.6,0.7-3.1,1.6-4.5,2.6l2.1,4.6c-1.9,1.4-3.5,3.1-5,4.9l-4.5-2.1 c-1,1.4-1.9,2.9-2.6,4.5l4.1,2.9c-0.9,2.1-1.5,4.4-1.8,6.8l-5,0.4C20,48.2,20,49.1,20,50s0,1.8,0.1,2.6l5,0.4 c0.3,2.4,0.9,4.7,1.8,6.8l-4.1,2.9c0.7,1.6,1.6,3.1,2.6,4.5l4.5-2.1c1.4,1.9,3.1,3.5,5,4.9l-2.1,4.6c1.4,1,2.9,1.9,4.5,2.6l2.9-4.1 c2.1,0.9,4.4,1.5,6.7,1.8l0.4,5.1C48.2,80,49.1,80,50,80s1.8,0,2.6-0.1l0.4-5.1c2.3-0.3,4.6-0.9,6.7-1.8l2.9,4.2 c1.6-0.7,3.1-1.6,4.5-2.6L65,69.9c1.9-1.4,3.5-3,4.9-4.9l4.6,2.2c1-1.4,1.9-2.9,2.6-4.5L73,59.8c0.9-2.1,1.5-4.4,1.8-6.7L79.9,52.6 z M50,65c-8.3,0-15-6.7-15-15c0-8.3,6.7-15,15-15s15,6.7,15,15C65,58.3,58.3,65,50,65z',
fill: 'currentColor'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '90 50 50',
dur: '1s',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerGears = createComponent({
name: 'QSpinnerGears',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$c)
}
});
const svg$b = [
vue.h('circle', {
cx: '12.5',
cy: '12.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '0s',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '12.5',
cy: '52.5',
r: '12.5',
'fill-opacity': '.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '100ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '52.5',
cy: '12.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '300ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '52.5',
cy: '52.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '600ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '92.5',
cy: '12.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '800ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '92.5',
cy: '52.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '400ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '12.5',
cy: '92.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '700ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '52.5',
cy: '92.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '500ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '92.5',
cy: '92.5',
r: '12.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '200ms',
dur: '1s',
values: '1;.2;1',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
];
var QSpinnerGrid = createComponent({
name: 'QSpinnerGrid',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
fill: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 105 105',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$b)
}
});
const svg$a = [
vue.h('path', {
d: 'M30.262 57.02L7.195 40.723c-5.84-3.976-7.56-12.06-3.842-18.063 3.715-6 11.467-7.65 17.306-3.68l4.52 3.76 2.6-5.274c3.716-6.002 11.47-7.65 17.304-3.68 5.84 3.97 7.56 12.054 3.842 18.062L34.49 56.118c-.897 1.512-2.793 1.915-4.228.9z',
'fill-opacity': '.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '0s',
dur: '1.4s',
values: '0.5;1;0.5',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('path', {
d: 'M105.512 56.12l-14.44-24.272c-3.716-6.008-1.996-14.093 3.843-18.062 5.835-3.97 13.588-2.322 17.306 3.68l2.6 5.274 4.52-3.76c5.84-3.97 13.593-2.32 17.308 3.68 3.718 6.003 1.998 14.088-3.842 18.064L109.74 57.02c-1.434 1.014-3.33.61-4.228-.9z',
'fill-opacity': '.5'
}, [
vue.h('animate', {
attributeName: 'fill-opacity',
begin: '0.7s',
dur: '1.4s',
values: '0.5;1;0.5',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('path', {
d: 'M67.408 57.834l-23.01-24.98c-5.864-6.15-5.864-16.108 0-22.248 5.86-6.14 15.37-6.14 21.234 0L70 16.168l4.368-5.562c5.863-6.14 15.375-6.14 21.235 0 5.863 6.14 5.863 16.098 0 22.247l-23.007 24.98c-1.43 1.556-3.757 1.556-5.188 0z'
})
];
var QSpinnerHearts = createComponent({
name: 'QSpinnerHearts',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
fill: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 140 64',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$a)
}
});
const svg$9 = [
vue.h('g', [
vue.h('path', {
fill: 'none',
stroke: 'currentColor',
'stroke-width': '5',
'stroke-miterlimit': '10',
d: 'M58.4,51.7c-0.9-0.9-1.4-2-1.4-2.3s0.5-0.4,1.4-1.4 C70.8,43.8,79.8,30.5,80,15.5H70H30H20c0.2,15,9.2,28.1,21.6,32.3c0.9,0.9,1.4,1.2,1.4,1.5s-0.5,1.6-1.4,2.5 C29.2,56.1,20.2,69.5,20,85.5h10h40h10C79.8,69.5,70.8,55.9,58.4,51.7z'
}),
vue.h('clipPath', {
id: 'uil-hourglass-clip1'
}, [
vue.h('rect', {
x: '15',
y: '20',
width: ' 70',
height: '25'
}, [
vue.h('animate', {
attributeName: 'height',
from: '25',
to: '0',
dur: '1s',
repeatCount: 'indefinite',
values: '25;0;0',
keyTimes: '0;0.5;1'
}),
vue.h('animate', {
attributeName: 'y',
from: '20',
to: '45',
dur: '1s',
repeatCount: 'indefinite',
values: '20;45;45',
keyTimes: '0;0.5;1'
})
])
]),
vue.h('clipPath', {
id: 'uil-hourglass-clip2'
}, [
vue.h('rect', {
x: '15',
y: '55',
width: ' 70',
height: '25'
}, [
vue.h('animate', {
attributeName: 'height',
from: '0',
to: '25',
dur: '1s',
repeatCount: 'indefinite',
values: '0;25;25',
keyTimes: '0;0.5;1'
}),
vue.h('animate', {
attributeName: 'y',
from: '80',
to: '55',
dur: '1s',
repeatCount: 'indefinite',
values: '80;55;55',
keyTimes: '0;0.5;1'
})
])
]),
vue.h('path', {
d: 'M29,23c3.1,11.4,11.3,19.5,21,19.5S67.9,34.4,71,23H29z',
'clip-path': 'url(#uil-hourglass-clip1)',
fill: 'currentColor'
}),
vue.h('path', {
d: 'M71.6,78c-3-11.6-11.5-20-21.5-20s-18.5,8.4-21.5,20H71.6z',
'clip-path': 'url(#uil-hourglass-clip2)',
fill: 'currentColor'
}),
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '180 50 50',
repeatCount: 'indefinite',
dur: '1s',
values: '0 50 50;0 50 50;180 50 50',
keyTimes: '0;0.7;1'
})
])
];
var QSpinnerHourglass = createComponent({
name: 'QSpinnerHourglass',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$9)
}
});
const svg$8 = [
vue.h('path', {
d: 'M24.3,30C11.4,30,5,43.3,5,50s6.4,20,19.3,20c19.3,0,32.1-40,51.4-40C88.6,30,95,43.3,95,50s-6.4,20-19.3,20C56.4,70,43.6,30,24.3,30z',
fill: 'none',
stroke: 'currentColor',
'stroke-width': '8',
'stroke-dasharray': '10.691205342610678 10.691205342610678',
'stroke-dashoffset': '0'
}, [
vue.h('animate', {
attributeName: 'stroke-dashoffset',
from: '0',
to: '21.382410685221355',
begin: '0',
dur: '2s',
repeatCount: 'indefinite',
fill: 'freeze'
})
])
];
var QSpinnerInfinity = createComponent({
name: 'QSpinnerInfinity',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid'
}, svg$8)
}
});
const svg$7 = [
vue.h('g', {
'stroke-width': '4',
'stroke-linecap': 'round'
}, [
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(180)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0;1',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(210)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '0;1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(240)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.1;0;1;.85;.7;.65;.55;.45;.35;.25;.15;.1',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(270)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.15;.1;0;1;.85;.7;.65;.55;.45;.35;.25;.15',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(300)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.25;.15;.1;0;1;.85;.7;.65;.55;.45;.35;.25',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(330)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.35;.25;.15;.1;0;1;.85;.7;.65;.55;.45;.35',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(0)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.45;.35;.25;.15;.1;0;1;.85;.7;.65;.55;.45',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(30)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.55;.45;.35;.25;.15;.1;0;1;.85;.7;.65;.55',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(60)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.65;.55;.45;.35;.25;.15;.1;0;1;.85;.7;.65',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(90)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.7;.65;.55;.45;.35;.25;.15;.1;0;1;.85;.7',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(120)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '.85;.7;.65;.55;.45;.35;.25;.15;.1;0;1;.85',
repeatCount: 'indefinite'
})
]),
vue.h('line', {
y1: '17',
y2: '29',
transform: 'translate(32,32) rotate(150)'
}, [
vue.h('animate', {
attributeName: 'stroke-opacity',
dur: '750ms',
values: '1;.85;.7;.65;.55;.45;.35;.25;.15;.1;0;1',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerIos = createComponent({
name: 'QSpinnerIos',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
stroke: 'currentColor',
fill: 'currentColor',
viewBox: '0 0 64 64'
}, svg$7)
}
});
const svg$6 = [
vue.h('circle', {
cx: '50',
cy: '50',
r: '44',
fill: 'none',
'stroke-width': '4',
'stroke-opacity': '.5',
stroke: 'currentColor'
}),
vue.h('circle', {
cx: '8',
cy: '54',
r: '6',
fill: 'currentColor',
'stroke-width': '3',
stroke: 'currentColor'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 48',
to: '360 50 52',
dur: '2s',
repeatCount: 'indefinite'
})
])
];
var QSpinnerOrbit = createComponent({
name: 'QSpinnerOrbit',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$6)
}
});
const svg$5 = [
vue.h('g', {
transform: 'translate(1 1)',
'stroke-width': '2',
fill: 'none',
'fill-rule': 'evenodd'
}, [
vue.h('circle', {
'stroke-opacity': '.5',
cx: '18',
cy: '18',
r: '18'
}),
vue.h('path', {
d: 'M36 18c0-9.94-8.06-18-18-18'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 18 18',
to: '360 18 18',
dur: '1s',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerOval = createComponent({
name: 'QSpinnerOval',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
stroke: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 38 38',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$5)
}
});
const svg$4 = [
vue.h('path', {
d: 'M0 50A50 50 0 0 1 50 0L50 50L0 50',
fill: 'currentColor',
opacity: '0.5'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '0.8s',
repeatCount: 'indefinite'
})
]),
vue.h('path', {
d: 'M50 0A50 50 0 0 1 100 50L50 50L50 0',
fill: 'currentColor',
opacity: '0.5'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '1.6s',
repeatCount: 'indefinite'
})
]),
vue.h('path', {
d: 'M100 50A50 50 0 0 1 50 100L50 50L100 50',
fill: 'currentColor',
opacity: '0.5'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '2.4s',
repeatCount: 'indefinite'
})
]),
vue.h('path', {
d: 'M50 100A50 50 0 0 1 0 50L50 50L50 100',
fill: 'currentColor',
opacity: '0.5'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 50 50',
to: '360 50 50',
dur: '3.2s',
repeatCount: 'indefinite'
})
])
];
var QSpinnerPie = createComponent({
name: 'QSpinnerPie',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$4)
}
});
const svg$3 = [
vue.h('g', {
fill: 'none',
'fill-rule': 'evenodd',
'stroke-width': '2'
}, [
vue.h('circle', {
cx: '22',
cy: '22',
r: '1'
}, [
vue.h('animate', {
attributeName: 'r',
begin: '0s',
dur: '1.8s',
values: '1; 20',
calcMode: 'spline',
keyTimes: '0; 1',
keySplines: '0.165, 0.84, 0.44, 1',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-opacity',
begin: '0s',
dur: '1.8s',
values: '1; 0',
calcMode: 'spline',
keyTimes: '0; 1',
keySplines: '0.3, 0.61, 0.355, 1',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '22',
cy: '22',
r: '1'
}, [
vue.h('animate', {
attributeName: 'r',
begin: '-0.9s',
dur: '1.8s',
values: '1; 20',
calcMode: 'spline',
keyTimes: '0; 1',
keySplines: '0.165, 0.84, 0.44, 1',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-opacity',
begin: '-0.9s',
dur: '1.8s',
values: '1; 0',
calcMode: 'spline',
keyTimes: '0; 1',
keySplines: '0.3, 0.61, 0.355, 1',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerPuff = createComponent({
name: 'QSpinnerPuff',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
stroke: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 44 44',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$3)
}
});
const svg$2 = [
vue.h('g', {
transform: 'scale(0.55)'
}, [
vue.h('circle', {
cx: '30',
cy: '150',
r: '30',
fill: 'currentColor'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
dur: '1s',
begin: '0',
repeatCount: 'indefinite',
keyTimes: '0;0.5;1',
values: '0;1;1'
})
]),
vue.h('path', {
d: 'M90,150h30c0-49.7-40.3-90-90-90v30C63.1,90,90,116.9,90,150z',
fill: 'currentColor'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
dur: '1s',
begin: '0.1',
repeatCount: 'indefinite',
keyTimes: '0;0.5;1',
values: '0;1;1'
})
]),
vue.h('path', {
d: 'M150,150h30C180,67.2,112.8,0,30,0v30C96.3,30,150,83.7,150,150z',
fill: 'currentColor'
}, [
vue.h('animate', {
attributeName: 'opacity',
from: '0',
to: '1',
dur: '1s',
begin: '0.2',
repeatCount: 'indefinite',
keyTimes: '0;0.5;1',
values: '0;1;1'
})
])
])
];
var QSpinnerRadio = createComponent({
name: 'QSpinnerRadio',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 100 100',
preserveAspectRatio: 'xMidYMid',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$2)
}
});
const svg$1 = [
vue.h('g', {
fill: 'none',
'fill-rule': 'evenodd',
transform: 'translate(1 1)',
'stroke-width': '2'
}, [
vue.h('circle', {
cx: '22',
cy: '22',
r: '6'
}, [
vue.h('animate', {
attributeName: 'r',
begin: '1.5s',
dur: '3s',
values: '6;22',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-opacity',
begin: '1.5s',
dur: '3s',
values: '1;0',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-width',
begin: '1.5s',
dur: '3s',
values: '2;0',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '22',
cy: '22',
r: '6'
}, [
vue.h('animate', {
attributeName: 'r',
begin: '3s',
dur: '3s',
values: '6;22',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-opacity',
begin: '3s',
dur: '3s',
values: '1;0',
calcMode: 'linear',
repeatCount: 'indefinite'
}),
vue.h('animate', {
attributeName: 'stroke-width',
begin: '3s',
dur: '3s',
values: '2;0',
calcMode: 'linear',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
cx: '22',
cy: '22',
r: '8'
}, [
vue.h('animate', {
attributeName: 'r',
begin: '0s',
dur: '1.5s',
values: '6;1;2;3;4;5;6',
calcMode: 'linear',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerRings = createComponent({
name: 'QSpinnerRings',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
stroke: 'currentColor',
width: cSize.value,
height: cSize.value,
viewBox: '0 0 45 45',
xmlns: 'http://www.w3.org/2000/svg'
}, svg$1)
}
});
const svg = [
vue.h('defs', [
vue.h('linearGradient', {
x1: '8.042%',
y1: '0%',
x2: '65.682%',
y2: '23.865%',
id: 'a'
}, [
vue.h('stop', {
'stop-color': 'currentColor',
'stop-opacity': '0',
offset: '0%'
}),
vue.h('stop', {
'stop-color': 'currentColor',
'stop-opacity': '.631',
offset: '63.146%'
}),
vue.h('stop', {
'stop-color': 'currentColor',
offset: '100%'
})
])
]),
vue.h('g', {
transform: 'translate(1 1)',
fill: 'none',
'fill-rule': 'evenodd'
}, [
vue.h('path', {
d: 'M36 18c0-9.94-8.06-18-18-18',
stroke: 'url(#a)',
'stroke-width': '2'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 18 18',
to: '360 18 18',
dur: '0.9s',
repeatCount: 'indefinite'
})
]),
vue.h('circle', {
fill: 'currentColor',
cx: '36',
cy: '18',
r: '1'
}, [
vue.h('animateTransform', {
attributeName: 'transform',
type: 'rotate',
from: '0 18 18',
to: '360 18 18',
dur: '0.9s',
repeatCount: 'indefinite'
})
])
])
];
var QSpinnerTail = createComponent({
name: 'QSpinnerTail',
props: useSpinnerProps,
setup (props) {
const { cSize, classes } = useSpinner(props);
return () => vue.h('svg', {
class: classes.value,
width: cSize.value,
height: cSize.value,
viewBox: '0 0 38 38',
xmlns: 'http://www.w3.org/2000/svg'
}, svg)
}
});
var QSplitter = createComponent({
name: 'QSplitter',
props: {
...useDarkProps,
modelValue: {
type: Number,
required: true
},
reverse: Boolean,
unit: {
type: String,
default: '%',
validator: v => [ '%', 'px' ].includes(v)
},
limits: {
type: Array,
validator: v => {
if (v.length !== 2) return false
if (typeof v[ 0 ] !== 'number' || typeof v[ 1 ] !== 'number') return false
return v[ 0 ] >= 0 && v[ 0 ] <= v[ 1 ]
}
},
emitImmediately: Boolean,
horizontal: Boolean,
disable: Boolean,
beforeClass: [ Array, String, Object ],
afterClass: [ Array, String, Object ],
separatorClass: [ Array, String, Object ],
separatorStyle: [ Array, String, Object ]
},
emits: [ 'update:modelValue' ],
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const isDark = useDark(props, $q);
const rootRef = vue.ref(null);
const sideRefs = {
before: vue.ref(null),
after: vue.ref(null)
};
const classes = vue.computed(() =>
'q-splitter no-wrap '
+ `${ props.horizontal === true ? 'q-splitter--horizontal column' : 'q-splitter--vertical row' }`
+ ` q-splitter--${ props.disable === true ? 'disabled' : 'workable' }`
+ (isDark.value === true ? ' q-splitter--dark' : '')
);
const propName = vue.computed(() => (props.horizontal === true ? 'height' : 'width'));
const side = vue.computed(() => (props.reverse !== true ? 'before' : 'after'));
const computedLimits = vue.computed(() => (
props.limits !== void 0
? props.limits
: (props.unit === '%' ? [ 10, 90 ] : [ 50, Infinity ])
));
function getCSSValue (value) {
return (props.unit === '%' ? value : Math.round(value)) + props.unit
}
const styles = vue.computed(() => ({
[ side.value ]: {
[ propName.value ]: getCSSValue(props.modelValue)
}
}));
let __dir, __maxValue, __value, __multiplier, __normalized;
function pan (evt) {
if (evt.isFirst === true) {
const size = rootRef.value.getBoundingClientRect()[ propName.value ];
__dir = props.horizontal === true ? 'up' : 'left';
__maxValue = props.unit === '%' ? 100 : size;
__value = Math.min(__maxValue, computedLimits.value[ 1 ], Math.max(computedLimits.value[ 0 ], props.modelValue));
__multiplier = (props.reverse !== true ? 1 : -1)
* (props.horizontal === true ? 1 : ($q.lang.rtl === true ? -1 : 1))
* (props.unit === '%' ? (size === 0 ? 0 : 100 / size) : 1);
rootRef.value.classList.add('q-splitter--active');
return
}
if (evt.isFinal === true) {
if (__normalized !== props.modelValue) {
emit('update:modelValue', __normalized);
}
rootRef.value.classList.remove('q-splitter--active');
return
}
const val = __value
+ __multiplier
* (evt.direction === __dir ? -1 : 1)
* evt.distance[ props.horizontal === true ? 'y' : 'x' ];
__normalized = Math.min(__maxValue, computedLimits.value[ 1 ], Math.max(computedLimits.value[ 0 ], val));
sideRefs[ side.value ].value.style[ propName.value ] = getCSSValue(__normalized);
if (props.emitImmediately === true && props.modelValue !== __normalized) {
emit('update:modelValue', __normalized);
}
}
const sepDirective = vue.computed(() => {
// if props.disable !== true
return [ [
TouchPan,
pan,
void 0,
{
[ props.horizontal === true ? 'vertical' : 'horizontal' ]: true,
prevent: true,
stop: true,
mouse: true,
mouseAllDir: true
}
] ]
});
function normalize (val, limits) {
if (val < limits[ 0 ]) {
emit('update:modelValue', limits[ 0 ]);
}
else if (val > limits[ 1 ]) {
emit('update:modelValue', limits[ 1 ]);
}
}
vue.watch(() => props.modelValue, v => {
normalize(v, computedLimits.value);
});
vue.watch(() => props.limits, () => {
vue.nextTick(() => {
normalize(props.modelValue, computedLimits.value);
});
});
return () => {
const child = [
vue.h('div', {
ref: sideRefs.before,
class: [
'q-splitter__panel q-splitter__before' + (props.reverse === true ? ' col' : ''),
props.beforeClass
],
style: styles.value.before
}, hSlot(slots.before)),
vue.h('div', {
class: [
'q-splitter__separator',
props.separatorClass
],
style: props.separatorStyle,
'aria-disabled': props.disable === true ? 'true' : void 0
}, [
hDir(
'div',
{ class: 'q-splitter__separator-area absolute-full' },
hSlot(slots.separator),
'sep',
props.disable !== true,
() => sepDirective.value
)
]),
vue.h('div', {
ref: sideRefs.after,
class: [
'q-splitter__panel q-splitter__after' + (props.reverse === true ? '' : ' col'),
props.afterClass
],
style: styles.value.after
}, hSlot(slots.after))
];
return vue.h('div', {
class: classes.value,
ref: rootRef
}, hMergeSlot(slots.default, child))
}
}
});
var StepHeader = createComponent({
name: 'StepHeader',
props: {
stepper: {},
step: {},
goToPanel: Function
},
setup (props, { attrs }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const blurRef = vue.ref(null);
const isActive = vue.computed(() => props.stepper.modelValue === props.step.name);
const isDisable = vue.computed(() => {
const opt = props.step.disable;
return opt === true || opt === ''
});
const isError = vue.computed(() => {
const opt = props.step.error;
return opt === true || opt === ''
});
const isDone = vue.computed(() => {
const opt = props.step.done;
return isDisable.value === false && (opt === true || opt === '')
});
const headerNav = vue.computed(() => {
const
opt = props.step.headerNav,
nav = opt === true || opt === '' || opt === void 0;
return isDisable.value === false
&& props.stepper.headerNav
&& nav
});
const hasPrefix = vue.computed(() => {
return props.step.prefix
&& (isActive.value === false || props.stepper.activeIcon === 'none')
&& (isError.value === false || props.stepper.errorIcon === 'none')
&& (isDone.value === false || props.stepper.doneIcon === 'none')
});
const icon = vue.computed(() => {
const defaultIcon = props.step.icon || props.stepper.inactiveIcon;
if (isActive.value === true) {
const icon = props.step.activeIcon || props.stepper.activeIcon;
return icon === 'none'
? defaultIcon
: icon || $q.iconSet.stepper.active
}
if (isError.value === true) {
const icon = props.step.errorIcon || props.stepper.errorIcon;
return icon === 'none'
? defaultIcon
: icon || $q.iconSet.stepper.error
}
if (isDisable.value === false && isDone.value === true) {
const icon = props.step.doneIcon || props.stepper.doneIcon;
return icon === 'none'
? defaultIcon
: icon || $q.iconSet.stepper.done
}
return defaultIcon
});
const color = vue.computed(() => {
const errorColor = isError.value === true
? props.step.errorColor || props.stepper.errorColor
: void 0;
if (isActive.value === true) {
const color = props.step.activeColor || props.stepper.activeColor || props.step.color;
return color !== void 0
? color
: errorColor
}
if (errorColor !== void 0) {
return errorColor
}
if (isDisable.value === false && isDone.value === true) {
return props.step.doneColor || props.stepper.doneColor || props.step.color || props.stepper.inactiveColor
}
return props.step.color || props.stepper.inactiveColor
});
const classes = vue.computed(() => {
return 'q-stepper__tab col-grow flex items-center no-wrap relative-position'
+ (color.value !== void 0 ? ` text-${ color.value }` : '')
+ (isError.value === true
? ' q-stepper__tab--error q-stepper__tab--error-with-' + (hasPrefix.value === true ? 'prefix' : 'icon')
: '')
+ (isActive.value === true ? ' q-stepper__tab--active' : '')
+ (isDone.value === true ? ' q-stepper__tab--done' : '')
+ (headerNav.value === true ? ' q-stepper__tab--navigation q-focusable q-hoverable' : '')
+ (isDisable.value === true ? ' q-stepper__tab--disabled' : '')
});
const ripple = vue.computed(() => (
props.stepper.headerNav !== true
? false
: headerNav.value
));
function onActivate () {
blurRef.value !== null && blurRef.value.focus();
isActive.value === false && props.goToPanel(props.step.name);
}
function onKeyup (e) {
if (e.keyCode === 13 && isActive.value === false) {
props.goToPanel(props.step.name);
}
}
return () => {
const data = { class: classes.value };
if (headerNav.value === true) {
data.onClick = onActivate;
data.onKeyup = onKeyup;
Object.assign(data,
isDisable.value === true
? { tabindex: -1, 'aria-disabled': 'true' }
: { tabindex: attrs.tabindex || 0 }
);
}
const child = [
vue.h('div', { class: 'q-focus-helper', tabindex: -1, ref: blurRef }),
vue.h('div', { class: 'q-stepper__dot row flex-center q-stepper__line relative-position' }, [
vue.h('span', { class: 'row flex-center' }, [
hasPrefix.value === true
? props.step.prefix
: vue.h(QIcon, { name: icon.value })
])
])
];
if (props.step.title !== void 0 && props.step.title !== null) {
const content = [
vue.h('div', { class: 'q-stepper__title' }, props.step.title)
];
if (props.step.caption !== void 0 && props.step.caption !== null) {
content.push(
vue.h('div', { class: 'q-stepper__caption' }, props.step.caption)
);
}
child.push(
vue.h('div', {
class: 'q-stepper__label q-stepper__line relative-position'
}, content)
);
}
return vue.withDirectives(
vue.h('div', data, child),
[ [ Ripple, ripple.value ] ]
)
}
}
});
function getStepWrapper (slots) {
return vue.h('div', {
class: 'q-stepper__step-content'
}, [
vue.h('div', {
class: 'q-stepper__step-inner'
}, hSlot(slots.default))
])
}
const PanelWrapper = {
setup (_, { slots }) {
return () => getStepWrapper(slots)
}
};
var QStep = createComponent({
name: 'QStep',
props: {
...usePanelChildProps,
icon: String,
color: String,
title: {
type: String,
required: true
},
caption: String,
prefix: [ String, Number ],
doneIcon: String,
doneColor: String,
activeIcon: String,
activeColor: String,
errorIcon: String,
errorColor: String,
headerNav: {
type: Boolean,
default: true
},
done: Boolean,
error: Boolean,
onScroll: [ Function, Array ]
},
setup (props, { slots, emit }) {
const { proxy: { $q } } = vue.getCurrentInstance();
const $stepper = vue.inject(stepperKey, emptyRenderFn);
if ($stepper === emptyRenderFn) {
console.error('QStep needs to be a child of QStepper');
return emptyRenderFn
}
const { getCacheWithFn } = useCache();
const rootRef = vue.ref(null);
const isActive = vue.computed(() => $stepper.value.modelValue === props.name);
const scrollEvent = vue.computed(() => (
($q.platform.is.ios !== true && $q.platform.is.chrome === true)
|| isActive.value !== true
|| $stepper.value.vertical !== true
? {}
: {
onScroll (e) {
const { target } = e;
if (target.scrollTop > 0) {
target.scrollTop = 0;
}
props.onScroll !== void 0 && emit('scroll', e);
}
}
));
const contentKey = vue.computed(() => (
typeof props.name === 'string' || typeof props.name === 'number'
? props.name
: String(props.name)
));
function getStepContent () {
const vertical = $stepper.value.vertical;
if (vertical === true && $stepper.value.keepAlive === true) {
return vue.h(
vue.KeepAlive,
$stepper.value.keepAliveProps.value,
isActive.value === true
? [
vue.h(
$stepper.value.needsUniqueKeepAliveWrapper.value === true
? getCacheWithFn(contentKey.value, () => ({ ...PanelWrapper, name: contentKey.value }))
: PanelWrapper,
{ key: contentKey.value },
slots.default
)
]
: void 0
)
}
return vertical !== true || isActive.value === true
? getStepWrapper(slots)
: void 0
}
return () => vue.h(
'div',
{ ref: rootRef, class: 'q-stepper__step', role: 'tabpanel', ...scrollEvent.value },
$stepper.value.vertical === true
? [
vue.h(StepHeader, {
stepper: $stepper.value,
step: props,
goToPanel: $stepper.value.goToPanel
}),
$stepper.value.animated === true
? vue.h(QSlideTransition, getStepContent)
: getStepContent()
]
: [ getStepContent() ]
)
}
});
const camelRE = /(-\w)/g;
function camelizeProps (props) {
const acc = {};
for (const key in props) {
const newKey = key.replace(camelRE, m => m[ 1 ].toUpperCase());
acc[ newKey ] = props[ key ];
}
return acc
}
var QStepper = createComponent({
name: 'QStepper',
props: {
...useDarkProps,
...usePanelProps,
flat: Boolean,
bordered: Boolean,
alternativeLabels: Boolean,
headerNav: Boolean,
contracted: Boolean,
headerClass: String,
inactiveColor: String,
inactiveIcon: String,
doneIcon: String,
doneColor: String,
activeIcon: String,
activeColor: String,
errorIcon: String,
errorColor: String
},
emits: usePanelEmits,
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
const {
updatePanelsList, isValidPanelName,
updatePanelIndex, getPanelContent,
getPanels, panelDirectives, goToPanel,
keepAliveProps, needsUniqueKeepAliveWrapper
} = usePanel();
vue.provide(stepperKey, vue.computed(() => ({
goToPanel,
keepAliveProps,
needsUniqueKeepAliveWrapper,
...props
})));
const classes = vue.computed(() =>
`q-stepper q-stepper--${ props.vertical === true ? 'vertical' : 'horizontal' }`
+ (props.flat === true ? ' q-stepper--flat' : '')
+ (props.bordered === true ? ' q-stepper--bordered' : '')
+ (isDark.value === true ? ' q-stepper--dark q-dark' : '')
);
const headerClasses = vue.computed(() =>
'q-stepper__header row items-stretch justify-between'
+ ` q-stepper__header--${ props.alternativeLabels === true ? 'alternative' : 'standard' }-labels`
+ (props.flat === false || props.bordered === true ? ' q-stepper__header--border' : '')
+ (props.contracted === true ? ' q-stepper__header--contracted' : '')
+ (props.headerClass !== void 0 ? ` ${ props.headerClass }` : '')
);
function getContent () {
const top = hSlot(slots.message, []);
if (props.vertical === true) {
isValidPanelName(props.modelValue) && updatePanelIndex();
const content = vue.h('div', {
class: 'q-stepper__content'
}, hSlot(slots.default));
return top === void 0
? [ content ]
: top.concat(content)
}
return [
vue.h(
'div',
{ class: headerClasses.value },
getPanels().map(panel => {
const step = camelizeProps(panel.props);
return vue.h(StepHeader, {
key: step.name,
stepper: props,
step,
goToPanel
})
})
),
top,
hDir(
'div',
{ class: 'q-stepper__content q-panel-parent' },
getPanelContent(),
'cont',
props.swipeable,
() => panelDirectives.value
)
]
}
return () => {
updatePanelsList(slots);
return vue.h('div', {
class: classes.value
}, hMergeSlot(slots.navigation, getContent()))
}
}
});
var QStepperNavigation = createComponent({
name: 'QStepperNavigation',
setup (_, { slots }) {
return () => vue.h('div', { class: 'q-stepper__nav' }, hSlot(slots.default))
}
});
var QTh = createComponent({
name: 'QTh',
props: {
props: Object,
autoWidth: Boolean
},
emits: [ 'click' ],
setup (props, { slots, emit }) {
const vm = vue.getCurrentInstance();
const { proxy: { $q } } = vm;
const onClick = evt => { emit('click', evt); };
return () => {
if (props.props === void 0) {
return vue.h('th', {
class: props.autoWidth === true ? 'q-table--col-auto-width' : '',
onClick
}, hSlot(slots.default))
}
let col, child;
const name = vm.vnode.key;
if (name) {
col = props.props.colsMap[ name ];
if (col === void 0) { return }
}
else {
col = props.props.col;
}
if (col.sortable === true) {
const action = col.align === 'right'
? 'unshift'
: 'push';
child = hUniqueSlot(slots.default, []);
child[ action ](
vue.h(QIcon, {
class: col.__iconClass,
name: $q.iconSet.table.arrowUp
})
);
}
else {
child = hSlot(slots.default);
}
const data = {
class: col.__thClass
+ (props.autoWidth === true ? ' q-table--col-auto-width' : ''),
style: col.headerStyle,
onClick: evt => {
col.sortable === true && props.props.sort(col);
onClick(evt);
}
};
return vue.h('th', data, child)
}
}
});
function getTableMiddle (props, content) {
return vue.h('div', props, [
vue.h('table', { class: 'q-table' }, content)
])
}
const comps = {
list: QList,
table: QMarkupTable
};
const typeOptions = [ 'list', 'table', '__qtable' ];
var QVirtualScroll = createComponent({
name: 'QVirtualScroll',
props: {
...useVirtualScrollProps,
type: {
type: String,
default: 'list',
validator: v => typeOptions.includes(v)
},
items: {
type: Array,
default: () => []
},
itemsFn: Function,
itemsSize: Number,
scrollTarget: {
default: void 0
}
},
setup (props, { slots, attrs }) {
let localScrollTarget;
const rootRef = vue.ref(null);
const virtualScrollLength = vue.computed(() => (
props.itemsSize >= 0 && props.itemsFn !== void 0
? parseInt(props.itemsSize, 10)
: (Array.isArray(props.items) ? props.items.length : 0)
));
const {
virtualScrollSliceRange,
localResetVirtualScroll,
padVirtualScroll,
onVirtualScrollEvt
} = useVirtualScroll({
virtualScrollLength, getVirtualScrollTarget, getVirtualScrollEl
});
const virtualScrollScope = vue.computed(() => {
if (virtualScrollLength.value === 0) {
return []
}
const mapFn = (item, i) => ({
index: virtualScrollSliceRange.value.from + i,
item
});
return props.itemsFn === void 0
? props.items.slice(virtualScrollSliceRange.value.from, virtualScrollSliceRange.value.to).map(mapFn)
: props.itemsFn(virtualScrollSliceRange.value.from, virtualScrollSliceRange.value.to - virtualScrollSliceRange.value.from).map(mapFn)
});
const classes = vue.computed(() =>
'q-virtual-scroll q-virtual-scroll' + (props.virtualScrollHorizontal === true ? '--horizontal' : '--vertical')
+ (props.scrollTarget !== void 0 ? '' : ' scroll')
);
const attributes = vue.computed(() => (
props.scrollTarget !== void 0 ? {} : { tabindex: 0 }
));
vue.watch(virtualScrollLength, () => {
localResetVirtualScroll();
});
vue.watch(() => props.scrollTarget, () => {
unconfigureScrollTarget();
configureScrollTarget();
});
function getVirtualScrollEl () {
return rootRef.value.$el || rootRef.value
}
function getVirtualScrollTarget () {
return localScrollTarget
}
function configureScrollTarget () {
localScrollTarget = getScrollTarget(getVirtualScrollEl(), props.scrollTarget);
localScrollTarget.addEventListener('scroll', onVirtualScrollEvt, listenOpts.passive);
}
function unconfigureScrollTarget () {
if (localScrollTarget !== void 0) {
localScrollTarget.removeEventListener('scroll', onVirtualScrollEvt, listenOpts.passive);
localScrollTarget = void 0;
}
}
function __getVirtualChildren () {
let child = padVirtualScroll(
props.type === 'list' ? 'div' : 'tbody',
virtualScrollScope.value.map(slots.default)
);
if (slots.before !== void 0) {
child = slots.before().concat(child);
}
return hMergeSlot(slots.after, child)
}
vue.onBeforeMount(() => {
localResetVirtualScroll();
});
vue.onMounted(() => {
configureScrollTarget();
});
vue.onActivated(() => {
configureScrollTarget();
});
vue.onDeactivated(() => {
unconfigureScrollTarget();
});
vue.onBeforeUnmount(() => {
unconfigureScrollTarget();
});
return () => {
if (slots.default === void 0) {
console.error('QVirtualScroll: default scoped slot is required for rendering');
return
}
return props.type === '__qtable'
? getTableMiddle(
{ ref: rootRef, class: 'q-table__middle ' + classes.value },
__getVirtualChildren()
)
: vue.h(comps[ props.type ], {
...attrs,
ref: rootRef,
class: [ attrs.class, classes.value ],
...attributes.value
}, __getVirtualChildren)
}
}
});
function sortDate (a, b) {
return (new Date(a)) - (new Date(b))
}
const useTableSortProps = {
sortMethod: Function,
binaryStateSort: Boolean,
columnSortOrder: {
type: String,
validator: v => v === 'ad' || v === 'da',
default: 'ad'
}
};
function useTableSort (props, computedPagination, colList, setPagination) {
const columnToSort = vue.computed(() => {
const { sortBy } = computedPagination.value;
return sortBy
? colList.value.find(def => def.name === sortBy) || null
: null
});
const computedSortMethod = vue.computed(() => (
props.sortMethod !== void 0
? props.sortMethod
: (data, sortBy, descending) => {
const col = colList.value.find(def => def.name === sortBy);
if (col === void 0 || col.field === void 0) {
return data
}
const
dir = descending === true ? -1 : 1,
val = typeof col.field === 'function'
? v => col.field(v)
: v => v[ col.field ];
return data.sort((a, b) => {
let
A = val(a),
B = val(b);
if (A === null || A === void 0) {
return -1 * dir
}
if (B === null || B === void 0) {
return 1 * dir
}
if (col.sort !== void 0) {
return col.sort(A, B, a, b) * dir
}
if (isNumber(A) === true && isNumber(B) === true) {
return (A - B) * dir
}
if (isDate(A) === true && isDate(B) === true) {
return sortDate(A, B) * dir
}
if (typeof A === 'boolean' && typeof B === 'boolean') {
return (A - B) * dir
}
[ A, B ] = [ A, B ].map(s => (s + '').toLocaleString().toLowerCase());
return A < B
? -1 * dir
: (A === B ? 0 : dir)
})
}
));
function sort (col /* String(col name) or Object(col definition) */) {
let sortOrder = props.columnSortOrder;
if (isObject(col) === true) {
if (col.sortOrder) {
sortOrder = col.sortOrder;
}
col = col.name;
}
else {
const def = colList.value.find(def => def.name === col);
if (def !== void 0 && def.sortOrder) {
sortOrder = def.sortOrder;
}
}
let { sortBy, descending } = computedPagination.value;
if (sortBy !== col) {
sortBy = col;
descending = sortOrder === 'da';
}
else if (props.binaryStateSort === true) {
descending = !descending;
}
else if (descending === true) {
if (sortOrder === 'ad') {
sortBy = null;
}
else {
descending = false;
}
}
else { // ascending
if (sortOrder === 'ad') {
descending = true;
}
else {
sortBy = null;
}
}
setPagination({ sortBy, descending, page: 1 });
}
return {
columnToSort,
computedSortMethod,
sort
}
}
const useTableFilterProps = {
filter: [ String, Object ],
filterMethod: Function
};
function useTableFilter (props, setPagination) {
const computedFilterMethod = vue.computed(() => (
props.filterMethod !== void 0
? props.filterMethod
: (rows, terms, cols, cellValue) => {
const lowerTerms = terms ? terms.toLowerCase() : '';
return rows.filter(
row => cols.some(col => {
const val = cellValue(col, row) + '';
const haystack = (val === 'undefined' || val === 'null') ? '' : val.toLowerCase();
return haystack.indexOf(lowerTerms) !== -1
})
)
}
));
vue.watch(
() => props.filter,
() => {
vue.nextTick(() => {
setPagination({ page: 1 }, true);
});
},
{ deep: true }
);
return { computedFilterMethod }
}
function samePagination (oldPag, newPag) {
for (const prop in newPag) {
if (newPag[ prop ] !== oldPag[ prop ]) {
return false
}
}
return true
}
function fixPagination (p) {
if (p.page < 1) {
p.page = 1;
}
if (p.rowsPerPage !== void 0 && p.rowsPerPage < 1) {
p.rowsPerPage = 0;
}
return p
}
const useTablePaginationProps = {
pagination: Object,
rowsPerPageOptions: {
type: Array,
default: () => [ 5, 7, 10, 15, 20, 25, 50, 0 ]
},
'onUpdate:pagination': [ Function, Array ]
};
function useTablePaginationState (vm, getCellValue) {
const { props, emit } = vm;
const innerPagination = vue.ref(
Object.assign({
sortBy: null,
descending: false,
page: 1,
rowsPerPage: props.rowsPerPageOptions.length !== 0
? props.rowsPerPageOptions[ 0 ]
: 5
}, props.pagination)
);
const computedPagination = vue.computed(() => {
const pag = props[ 'onUpdate:pagination' ] !== void 0
? { ...innerPagination.value, ...props.pagination }
: innerPagination.value;
return fixPagination(pag)
});
const isServerSide = vue.computed(() => computedPagination.value.rowsNumber !== void 0);
function sendServerRequest (pagination) {
requestServerInteraction({
pagination,
filter: props.filter
});
}
function requestServerInteraction (prop = {}) {
vue.nextTick(() => {
emit('request', {
pagination: prop.pagination || computedPagination.value,
filter: prop.filter || props.filter,
getCellValue
});
});
}
function setPagination (val, forceServerRequest) {
const newPagination = fixPagination({
...computedPagination.value,
...val
});
if (samePagination(computedPagination.value, newPagination) === true) {
if (isServerSide.value === true && forceServerRequest === true) {
sendServerRequest(newPagination);
}
return
}
if (isServerSide.value === true) {
sendServerRequest(newPagination);
return
}
if (
props.pagination !== void 0
&& props[ 'onUpdate:pagination' ] !== void 0
) {
emit('update:pagination', newPagination);
}
else {
innerPagination.value = newPagination;
}
}
return {
innerPagination,
computedPagination,
isServerSide,
requestServerInteraction,
setPagination
}
}
function useTablePagination (vm, innerPagination, computedPagination, isServerSide, setPagination, filteredSortedRowsNumber) {
const { props, emit, proxy: { $q } } = vm;
const computedRowsNumber = vue.computed(() => (
isServerSide.value === true
? computedPagination.value.rowsNumber || 0
: filteredSortedRowsNumber.value
));
const firstRowIndex = vue.computed(() => {
const { page, rowsPerPage } = computedPagination.value;
return (page - 1) * rowsPerPage
});
const lastRowIndex = vue.computed(() => {
const { page, rowsPerPage } = computedPagination.value;
return page * rowsPerPage
});
const isFirstPage = vue.computed(() => computedPagination.value.page === 1);
const pagesNumber = vue.computed(() => (
computedPagination.value.rowsPerPage === 0
? 1
: Math.max(
1,
Math.ceil(computedRowsNumber.value / computedPagination.value.rowsPerPage)
)
));
const isLastPage = vue.computed(() => (
lastRowIndex.value === 0
? true
: computedPagination.value.page >= pagesNumber.value
));
const computedRowsPerPageOptions = vue.computed(() => {
const opts = props.rowsPerPageOptions.includes(innerPagination.value.rowsPerPage)
? props.rowsPerPageOptions
: [ innerPagination.value.rowsPerPage ].concat(props.rowsPerPageOptions);
return opts.map(count => ({
label: count === 0 ? $q.lang.table.allRows : '' + count,
value: count
}))
});
vue.watch(pagesNumber, (lastPage, oldLastPage) => {
if (lastPage === oldLastPage) {
return
}
const currentPage = computedPagination.value.page;
if (lastPage && !currentPage) {
setPagination({ page: 1 });
}
else if (lastPage < currentPage) {
setPagination({ page: lastPage });
}
});
function firstPage () {
setPagination({ page: 1 });
}
function prevPage () {
const { page } = computedPagination.value;
if (page > 1) {
setPagination({ page: page - 1 });
}
}
function nextPage () {
const { page, rowsPerPage } = computedPagination.value;
if (lastRowIndex.value > 0 && page * rowsPerPage < computedRowsNumber.value) {
setPagination({ page: page + 1 });
}
}
function lastPage () {
setPagination({ page: pagesNumber.value });
}
if (props[ 'onUpdate:pagination' ] !== void 0) {
emit('update:pagination', { ...computedPagination.value });
}
return {
firstRowIndex,
lastRowIndex,
isFirstPage,
isLastPage,
pagesNumber,
computedRowsPerPageOptions,
computedRowsNumber,
firstPage,
prevPage,
nextPage,
lastPage
}
}
const useTableRowSelectionProps = {
selection: {
type: String,
default: 'none',
validator: v => [ 'single', 'multiple', 'none' ].includes(v)
},
selected: {
type: Array,
default: () => []
}
};
const useTableRowSelectionEmits = [ 'update:selected', 'selection' ];
function useTableRowSelection (props, emit, computedRows, getRowKey) {
const selectedKeys = vue.computed(() => {
const keys = {};
props.selected.map(getRowKey.value).forEach(key => {
keys[ key ] = true;
});
return keys
});
const hasSelectionMode = vue.computed(() => {
return props.selection !== 'none'
});
const singleSelection = vue.computed(() => {
return props.selection === 'single'
});
const multipleSelection = vue.computed(() => {
return props.selection === 'multiple'
});
const allRowsSelected = vue.computed(() =>
computedRows.value.length !== 0 && computedRows.value.every(
row => selectedKeys.value[ getRowKey.value(row) ] === true
)
);
const someRowsSelected = vue.computed(() =>
allRowsSelected.value !== true
&& computedRows.value.some(row => selectedKeys.value[ getRowKey.value(row) ] === true)
);
const rowsSelectedNumber = vue.computed(() => props.selected.length);
function isRowSelected (key) {
return selectedKeys.value[ key ] === true
}
function clearSelection () {
emit('update:selected', []);
}
function updateSelection (keys, rows, added, evt) {
emit('selection', { rows, added, keys, evt });
const payload = singleSelection.value === true
? (added === true ? rows : [])
: (
added === true
? props.selected.concat(rows)
: props.selected.filter(
row => keys.includes(getRowKey.value(row)) === false
)
);
emit('update:selected', payload);
}
return {
hasSelectionMode,
singleSelection,
multipleSelection,
allRowsSelected,
someRowsSelected,
rowsSelectedNumber,
isRowSelected,
clearSelection,
updateSelection
}
}
function getVal (val) {
return Array.isArray(val)
? val.slice()
: []
}
const useTableRowExpandProps = {
expanded: Array // v-model:expanded
};
const useTableRowExpandEmits = [ 'update:expanded' ];
function useTableRowExpand (props, emit) {
const innerExpanded = vue.ref(getVal(props.expanded));
vue.watch(() => props.expanded, val => {
innerExpanded.value = getVal(val);
});
function isRowExpanded (key) {
return innerExpanded.value.includes(key)
}
function setExpanded (val) {
if (props.expanded !== void 0) {
emit('update:expanded', val);
}
else {
innerExpanded.value = val;
}
}
function updateExpanded (key, add) {
const target = innerExpanded.value.slice();
const index = target.indexOf(key);
if (add === true) {
if (index === -1) {
target.push(key);
setExpanded(target);
}
}
else if (index !== -1) {
target.splice(index, 1);
setExpanded(target);
}
}
return {
isRowExpanded,
setExpanded,
updateExpanded
}
}
const useTableColumnSelectionProps = {
visibleColumns: Array
};
function useTableColumnSelection (props, computedPagination, hasSelectionMode) {
const colList = vue.computed(() => {
if (props.columns !== void 0) {
return props.columns
}
// we infer columns from first row
const row = props.rows[ 0 ];
return row !== void 0
? Object.keys(row).map(name => ({
name,
label: name.toUpperCase(),
field: name,
align: isNumber(row[ name ]) ? 'right' : 'left',
sortable: true
}))
: []
});
const computedCols = vue.computed(() => {
const { sortBy, descending } = computedPagination.value;
const cols = props.visibleColumns !== void 0
? colList.value.filter(col => col.required === true || props.visibleColumns.includes(col.name) === true)
: colList.value;
return cols.map(col => {
const align = col.align || 'right';
const alignClass = `text-${ align }`;
return {
...col,
align,
__iconClass: `q-table__sort-icon q-table__sort-icon--${ align }`,
__thClass: alignClass
+ (col.headerClasses !== void 0 ? ' ' + col.headerClasses : '')
+ (col.sortable === true ? ' sortable' : '')
+ (col.name === sortBy ? ` sorted ${ descending === true ? 'sort-desc' : '' }` : ''),
__tdStyle: col.style !== void 0
? (
typeof col.style !== 'function'
? () => col.style
: col.style
)
: () => null,
__tdClass: col.classes !== void 0
? (
typeof col.classes !== 'function'
? () => alignClass + ' ' + col.classes
: row => alignClass + ' ' + col.classes(row)
)
: () => alignClass
}
})
});
const computedColsMap = vue.computed(() => {
const names = {};
computedCols.value.forEach(col => {
names[ col.name ] = col;
});
return names
});
const computedColspan = vue.computed(() => {
return props.tableColspan !== void 0
? props.tableColspan
: computedCols.value.length + (hasSelectionMode.value === true ? 1 : 0)
});
return {
colList,
computedCols,
computedColsMap,
computedColspan
}
}
const bottomClass = 'q-table__bottom row items-center';
const commonVirtPropsObj = {};
commonVirtPropsList.forEach(p => { commonVirtPropsObj[ p ] = {}; });
var QTable = createComponent({
name: 'QTable',
props: {
rows: {
type: Array,
default: () => []
},
rowKey: {
type: [ String, Function ],
default: 'id'
},
columns: Array,
loading: Boolean,
iconFirstPage: String,
iconPrevPage: String,
iconNextPage: String,
iconLastPage: String,
title: String,
hideHeader: Boolean,
grid: Boolean,
gridHeader: Boolean,
dense: Boolean,
flat: Boolean,
bordered: Boolean,
square: Boolean,
separator: {
type: String,
default: 'horizontal',
validator: v => [ 'horizontal', 'vertical', 'cell', 'none' ].includes(v)
},
wrapCells: Boolean,
virtualScroll: Boolean,
virtualScrollTarget: {
default: void 0
},
...commonVirtPropsObj,
noDataLabel: String,
noResultsLabel: String,
loadingLabel: String,
selectedRowsLabel: Function,
rowsPerPageLabel: String,
paginationLabel: Function,
color: {
type: String,
default: 'grey-8'
},
titleClass: [ String, Array, Object ],
tableStyle: [ String, Array, Object ],
tableClass: [ String, Array, Object ],
tableHeaderStyle: [ String, Array, Object ],
tableHeaderClass: [ String, Array, Object ],
cardContainerClass: [ String, Array, Object ],
cardContainerStyle: [ String, Array, Object ],
cardStyle: [ String, Array, Object ],
cardClass: [ String, Array, Object ],
hideBottom: Boolean,
hideSelectedBanner: Boolean,
hideNoData: Boolean,
hidePagination: Boolean,
onRowClick: Function,
onRowDblclick: Function,
onRowContextmenu: Function,
...useDarkProps,
...useFullscreenProps,
...useTableColumnSelectionProps,
...useTableFilterProps,
...useTablePaginationProps,
...useTableRowExpandProps,
...useTableRowSelectionProps,
...useTableSortProps
},
emits: [
'request', 'virtualScroll',
...useFullscreenEmits,
...useTableRowExpandEmits,
...useTableRowSelectionEmits
],
setup (props, { slots, emit }) {
const vm = vue.getCurrentInstance();
const { proxy: { $q } } = vm;
const isDark = useDark(props, $q);
const { inFullscreen, toggleFullscreen } = useFullscreen();
const getRowKey = vue.computed(() => (
typeof props.rowKey === 'function'
? props.rowKey
: row => row[ props.rowKey ]
));
const rootRef = vue.ref(null);
const virtScrollRef = vue.ref(null);
const hasVirtScroll = vue.computed(() => props.grid !== true && props.virtualScroll === true);
const cardDefaultClass = vue.computed(() =>
' q-table__card'
+ (isDark.value === true ? ' q-table__card--dark q-dark' : '')
+ (props.square === true ? ' q-table--square' : '')
+ (props.flat === true ? ' q-table--flat' : '')
+ (props.bordered === true ? ' q-table--bordered' : '')
);
const __containerClass = vue.computed(() =>
`q-table__container q-table--${ props.separator }-separator column no-wrap`
+ (props.grid === true ? ' q-table--grid' : cardDefaultClass.value)
+ (isDark.value === true ? ' q-table--dark' : '')
+ (props.dense === true ? ' q-table--dense' : '')
+ (props.wrapCells === false ? ' q-table--no-wrap' : '')
+ (inFullscreen.value === true ? ' fullscreen scroll' : '')
);
const containerClass = vue.computed(() =>
__containerClass.value + (props.loading === true ? ' q-table--loading' : '')
);
vue.watch(
() => props.tableStyle + props.tableClass + props.tableHeaderStyle + props.tableHeaderClass + __containerClass.value,
() => { hasVirtScroll.value === true && virtScrollRef.value !== null && virtScrollRef.value.reset(); }
);
const {
innerPagination,
computedPagination,
isServerSide,
requestServerInteraction,
setPagination
} = useTablePaginationState(vm, getCellValue);
const { computedFilterMethod } = useTableFilter(props, setPagination);
const { isRowExpanded, setExpanded, updateExpanded } = useTableRowExpand(props, emit);
const filteredSortedRows = vue.computed(() => {
let rows = props.rows;
if (isServerSide.value === true || rows.length === 0) {
return rows
}
const { sortBy, descending } = computedPagination.value;
if (props.filter) {
rows = computedFilterMethod.value(rows, props.filter, computedCols.value, getCellValue);
}
if (columnToSort.value !== null) {
rows = computedSortMethod.value(
props.rows === rows ? rows.slice() : rows,
sortBy,
descending
);
}
return rows
});
const filteredSortedRowsNumber = vue.computed(() => filteredSortedRows.value.length);
const computedRows = vue.computed(() => {
let rows = filteredSortedRows.value;
if (isServerSide.value === true) {
return rows
}
const { rowsPerPage } = computedPagination.value;
if (rowsPerPage !== 0) {
if (firstRowIndex.value === 0 && props.rows !== rows) {
if (rows.length > lastRowIndex.value) {
rows = rows.slice(0, lastRowIndex.value);
}
}
else {
rows = rows.slice(firstRowIndex.value, lastRowIndex.value);
}
}
return rows
});
const {
hasSelectionMode,
singleSelection,
multipleSelection,
allRowsSelected,
someRowsSelected,
rowsSelectedNumber,
isRowSelected,
clearSelection,
updateSelection
} = useTableRowSelection(props, emit, computedRows, getRowKey);
const { colList, computedCols, computedColsMap, computedColspan } = useTableColumnSelection(props, computedPagination, hasSelectionMode);
const { columnToSort, computedSortMethod, sort } = useTableSort(props, computedPagination, colList, setPagination);
const {
firstRowIndex,
lastRowIndex,
isFirstPage,
isLastPage,
pagesNumber,
computedRowsPerPageOptions,
computedRowsNumber,
firstPage,
prevPage,
nextPage,
lastPage
} = useTablePagination(vm, innerPagination, computedPagination, isServerSide, setPagination, filteredSortedRowsNumber);
const nothingToDisplay = vue.computed(() => computedRows.value.length === 0);
const virtProps = vue.computed(() => {
const acc = {};
commonVirtPropsList
.forEach(p => { acc[ p ] = props[ p ]; });
if (acc.virtualScrollItemSize === void 0) {
acc.virtualScrollItemSize = props.dense === true ? 28 : 48;
}
return acc
});
function resetVirtualScroll () {
hasVirtScroll.value === true && virtScrollRef.value.reset();
}
function getBody () {
if (props.grid === true) {
return getGridBody()
}
const header = props.hideHeader !== true ? getTHead : null;
if (hasVirtScroll.value === true) {
const topRow = slots[ 'top-row' ];
const bottomRow = slots[ 'bottom-row' ];
const virtSlots = {
default: props => getTBodyTR(props.item, slots.body, props.index)
};
if (topRow !== void 0) {
const topContent = vue.h('tbody', topRow({ cols: computedCols.value }));
virtSlots.before = header === null
? () => topContent
: () => [ header() ].concat(topContent);
}
else if (header !== null) {
virtSlots.before = header;
}
if (bottomRow !== void 0) {
virtSlots.after = () => vue.h('tbody', bottomRow({ cols: computedCols.value }));
}
return vue.h(QVirtualScroll, {
ref: virtScrollRef,
class: props.tableClass,
style: props.tableStyle,
...virtProps.value,
scrollTarget: props.virtualScrollTarget,
items: computedRows.value,
type: '__qtable',
tableColspan: computedColspan.value,
onVirtualScroll: onVScroll
}, virtSlots)
}
const child = [
getTBody()
];
if (header !== null) {
child.unshift(header());
}
return getTableMiddle({
class: [ 'q-table__middle scroll', props.tableClass ],
style: props.tableStyle
}, child)
}
function scrollTo (toIndex, edge) {
if (virtScrollRef.value !== null) {
virtScrollRef.value.scrollTo(toIndex, edge);
return
}
toIndex = parseInt(toIndex, 10);
const rowEl = rootRef.value.querySelector(`tbody tr:nth-of-type(${ toIndex + 1 })`);
if (rowEl !== null) {
const scrollTarget = rootRef.value.querySelector('.q-table__middle.scroll');
const offsetTop = rowEl.offsetTop - props.virtualScrollStickySizeStart;
const direction = offsetTop < scrollTarget.scrollTop ? 'decrease' : 'increase';
scrollTarget.scrollTop = offsetTop;
emit('virtualScroll', {
index: toIndex,
from: 0,
to: innerPagination.value.rowsPerPage - 1,
direction
});
}
}
function onVScroll (info) {
emit('virtualScroll', info);
}
function getProgress () {
return [
vue.h(QLinearProgress, {
class: 'q-table__linear-progress',
color: props.color,
dark: isDark.value,
indeterminate: true,
trackColor: 'transparent'
})
]
}
function getTBodyTR (row, bodySlot, pageIndex) {
const
key = getRowKey.value(row),
selected = isRowSelected(key);
if (bodySlot !== void 0) {
return bodySlot(
getBodyScope({
key,
row,
pageIndex,
__trClass: selected ? 'selected' : ''
})
)
}
const
bodyCell = slots[ 'body-cell' ],
child = computedCols.value.map(col => {
const
bodyCellCol = slots[ `body-cell-${ col.name }` ],
slot = bodyCellCol !== void 0 ? bodyCellCol : bodyCell;
return slot !== void 0
? slot(getBodyCellScope({ key, row, pageIndex, col }))
: vue.h('td', {
class: col.__tdClass(row),
style: col.__tdStyle(row)
}, getCellValue(col, row))
});
if (hasSelectionMode.value === true) {
const slot = slots[ 'body-selection' ];
const content = slot !== void 0
? slot(getBodySelectionScope({ key, row, pageIndex }))
: [
vue.h(QCheckbox, {
modelValue: selected,
color: props.color,
dark: isDark.value,
dense: props.dense,
'onUpdate:modelValue': (adding, evt) => {
updateSelection([ key ], [ row ], adding, evt);
}
})
];
child.unshift(
vue.h('td', { class: 'q-table--col-auto-width' }, content)
);
}
const data = { key, class: { selected } };
if (props.onRowClick !== void 0) {
data.class[ 'cursor-pointer' ] = true;
data.onClick = evt => {
emit('RowClick', evt, row, pageIndex);
};
}
if (props.onRowDblclick !== void 0) {
data.class[ 'cursor-pointer' ] = true;
data.onDblclick = evt => {
emit('RowDblclick', evt, row, pageIndex);
};
}
if (props.onRowContextmenu !== void 0) {
data.class[ 'cursor-pointer' ] = true;
data.onContextmenu = evt => {
emit('RowContextmenu', evt, row, pageIndex);
};
}
return vue.h('tr', data, child)
}
function getTBody () {
const
body = slots.body,
topRow = slots[ 'top-row' ],
bottomRow = slots[ 'bottom-row' ];
let child = computedRows.value.map(
(row, pageIndex) => getTBodyTR(row, body, pageIndex)
);
if (topRow !== void 0) {
child = topRow({ cols: computedCols.value }).concat(child);
}
if (bottomRow !== void 0) {
child = child.concat(bottomRow({ cols: computedCols.value }));
}
return vue.h('tbody', child)
}
function getBodyScope (data) {
injectBodyCommonScope(data);
data.cols = data.cols.map(
col => injectProp({ ...col }, 'value', () => getCellValue(col, data.row))
);
return data
}
function getBodyCellScope (data) {
injectBodyCommonScope(data);
injectProp(data, 'value', () => getCellValue(data.col, data.row));
return data
}
function getBodySelectionScope (data) {
injectBodyCommonScope(data);
return data
}
function injectBodyCommonScope (data) {
Object.assign(data, {
cols: computedCols.value,
colsMap: computedColsMap.value,
sort,
rowIndex: firstRowIndex.value + data.pageIndex,
color: props.color,
dark: isDark.value,
dense: props.dense
});
hasSelectionMode.value === true && injectProp(
data,
'selected',
() => isRowSelected(data.key),
(adding, evt) => {
updateSelection([ data.key ], [ data.row ], adding, evt);
}
);
injectProp(
data,
'expand',
() => isRowExpanded(data.key),
adding => { updateExpanded(data.key, adding); }
);
}
function getCellValue (col, row) {
const val = typeof col.field === 'function' ? col.field(row) : row[ col.field ];
return col.format !== void 0 ? col.format(val, row) : val
}
const marginalsScope = vue.computed(() => ({
pagination: computedPagination.value,
pagesNumber: pagesNumber.value,
isFirstPage: isFirstPage.value,
isLastPage: isLastPage.value,
firstPage,
prevPage,
nextPage,
lastPage,
inFullscreen: inFullscreen.value,
toggleFullscreen
}));
function getTopDiv () {
const
top = slots.top,
topLeft = slots[ 'top-left' ],
topRight = slots[ 'top-right' ],
topSelection = slots[ 'top-selection' ],
hasSelection = hasSelectionMode.value === true
&& topSelection !== void 0
&& rowsSelectedNumber.value > 0,
topClass = 'q-table__top relative-position row items-center';
if (top !== void 0) {
return vue.h('div', { class: topClass }, [ top(marginalsScope.value) ])
}
let child;
if (hasSelection === true) {
child = topSelection(marginalsScope.value).slice();
}
else {
child = [];
if (topLeft !== void 0) {
child.push(
vue.h('div', { class: 'q-table__control' }, [
topLeft(marginalsScope.value)
])
);
}
else if (props.title) {
child.push(
vue.h('div', { class: 'q-table__control' }, [
vue.h('div', {
class: [ 'q-table__title', props.titleClass ]
}, props.title)
])
);
}
}
if (topRight !== void 0) {
child.push(
vue.h('div', { class: 'q-table__separator col' })
);
child.push(
vue.h('div', { class: 'q-table__control' }, [
topRight(marginalsScope.value)
])
);
}
if (child.length === 0) {
return
}
return vue.h('div', { class: topClass }, child)
}
const headerSelectedValue = vue.computed(() => (
someRowsSelected.value === true
? null
: allRowsSelected.value
));
function getTHead () {
const child = getTHeadTR();
if (props.loading === true && slots.loading === void 0) {
child.push(
vue.h('tr', { class: 'q-table__progress' }, [
vue.h('th', {
class: 'relative-position',
colspan: computedColspan.value
}, getProgress())
])
);
}
return vue.h('thead', child)
}
function getTHeadTR () {
const
header = slots.header,
headerCell = slots[ 'header-cell' ];
if (header !== void 0) {
return header(
getHeaderScope({ header: true })
).slice()
}
const child = computedCols.value.map(col => {
const
headerCellCol = slots[ `header-cell-${ col.name }` ],
slot = headerCellCol !== void 0 ? headerCellCol : headerCell,
props = getHeaderScope({ col });
return slot !== void 0
? slot(props)
: vue.h(QTh, {
key: col.name,
props
}, () => col.label)
});
if (singleSelection.value === true && props.grid !== true) {
child.unshift(
vue.h('th', { class: 'q-table--col-auto-width' }, ' ')
);
}
else if (multipleSelection.value === true) {
const slot = slots[ 'header-selection' ];
const content = slot !== void 0
? slot(getHeaderScope({}))
: [
vue.h(QCheckbox, {
color: props.color,
modelValue: headerSelectedValue.value,
dark: isDark.value,
dense: props.dense,
'onUpdate:modelValue': onMultipleSelectionSet
})
];
child.unshift(
vue.h('th', { class: 'q-table--col-auto-width' }, content)
);
}
return [
vue.h('tr', {
class: props.tableHeaderClass,
style: props.tableHeaderStyle
}, child)
]
}
function getHeaderScope (data) {
Object.assign(data, {
cols: computedCols.value,
sort,
colsMap: computedColsMap.value,
color: props.color,
dark: isDark.value,
dense: props.dense
});
if (multipleSelection.value === true) {
injectProp(
data,
'selected',
() => headerSelectedValue.value,
onMultipleSelectionSet
);
}
return data
}
function onMultipleSelectionSet (val) {
if (someRowsSelected.value === true) {
val = false;
}
updateSelection(
computedRows.value.map(getRowKey.value),
computedRows.value,
val
);
}
const navIcon = vue.computed(() => {
const ico = [
props.iconFirstPage || $q.iconSet.table.firstPage,
props.iconPrevPage || $q.iconSet.table.prevPage,
props.iconNextPage || $q.iconSet.table.nextPage,
props.iconLastPage || $q.iconSet.table.lastPage
];
return $q.lang.rtl === true ? ico.reverse() : ico
});
function getBottomDiv () {
if (props.hideBottom === true) {
return
}
if (nothingToDisplay.value === true) {
if (props.hideNoData === true) {
return
}
const message = props.loading === true
? props.loadingLabel || $q.lang.table.loading
: (props.filter ? props.noResultsLabel || $q.lang.table.noResults : props.noDataLabel || $q.lang.table.noData);
const noData = slots[ 'no-data' ];
const children = noData !== void 0
? [ noData({ message, icon: $q.iconSet.table.warning, filter: props.filter }) ]
: [
vue.h(QIcon, {
class: 'q-table__bottom-nodata-icon',
name: $q.iconSet.table.warning
}),
message
];
return vue.h('div', { class: bottomClass + ' q-table__bottom--nodata' }, children)
}
const bottom = slots.bottom;
if (bottom !== void 0) {
return vue.h('div', { class: bottomClass }, [ bottom(marginalsScope.value) ])
}
const child = props.hideSelectedBanner !== true && hasSelectionMode.value === true && rowsSelectedNumber.value > 0
? [
vue.h('div', { class: 'q-table__control' }, [
vue.h('div', [
(props.selectedRowsLabel || $q.lang.table.selectedRecords)(rowsSelectedNumber.value)
])
])
]
: [];
if (props.hidePagination !== true) {
return vue.h('div', {
class: bottomClass + ' justify-end'
}, getPaginationDiv(child))
}
if (child.length !== 0) {
return vue.h('div', { class: bottomClass }, child)
}
}
function onPagSelection (pag) {
setPagination({
page: 1,
rowsPerPage: pag.value
});
}
function getPaginationDiv (child) {
let control;
const
{ rowsPerPage } = computedPagination.value,
paginationLabel = props.paginationLabel || $q.lang.table.pagination,
paginationSlot = slots.pagination,
hasOpts = props.rowsPerPageOptions.length > 1;
child.push(
vue.h('div', { class: 'q-table__separator col' })
);
if (hasOpts === true) {
child.push(
vue.h('div', { class: 'q-table__control' }, [
vue.h('span', { class: 'q-table__bottom-item' }, [
props.rowsPerPageLabel || $q.lang.table.recordsPerPage
]),
vue.h(QSelect, {
class: 'q-table__select inline q-table__bottom-item',
color: props.color,
modelValue: rowsPerPage,
options: computedRowsPerPageOptions.value,
displayValue: rowsPerPage === 0
? $q.lang.table.allRows
: rowsPerPage,
dark: isDark.value,
borderless: true,
dense: true,
optionsDense: true,
optionsCover: true,
'onUpdate:modelValue': onPagSelection
})
])
);
}
if (paginationSlot !== void 0) {
control = paginationSlot(marginalsScope.value);
}
else {
control = [
vue.h('span', rowsPerPage !== 0 ? { class: 'q-table__bottom-item' } : {}, [
rowsPerPage
? paginationLabel(firstRowIndex.value + 1, Math.min(lastRowIndex.value, computedRowsNumber.value), computedRowsNumber.value)
: paginationLabel(1, filteredSortedRowsNumber.value, computedRowsNumber.value)
])
];
if (rowsPerPage !== 0 && pagesNumber.value > 1) {
const btnProps = {
color: props.color,
round: true,
dense: true,
flat: true
};
if (props.dense === true) {
btnProps.size = 'sm';
}
pagesNumber.value > 2 && control.push(
vue.h(QBtn, {
key: 'pgFirst',
...btnProps,
icon: navIcon.value[ 0 ],
disable: isFirstPage.value,
onClick: firstPage
})
);
control.push(
vue.h(QBtn, {
key: 'pgPrev',
...btnProps,
icon: navIcon.value[ 1 ],
disable: isFirstPage.value,
onClick: prevPage
}),
vue.h(QBtn, {
key: 'pgNext',
...btnProps,
icon: navIcon.value[ 2 ],
disable: isLastPage.value,
onClick: nextPage
})
);
pagesNumber.value > 2 && control.push(
vue.h(QBtn, {
key: 'pgLast',
...btnProps,
icon: navIcon.value[ 3 ],
disable: isLastPage.value,
onClick: lastPage
})
);
}
}
child.push(
vue.h('div', { class: 'q-table__control' }, control)
);
return child
}
function getGridHeader () {
const child = props.gridHeader === true
? [
vue.h('table', { class: 'q-table' }, [
getTHead()
])
]
: (
props.loading === true && slots.loading === void 0
? getProgress()
: void 0
);
return vue.h('div', { class: 'q-table__middle' }, child)
}
function getGridBody () {
const item = slots.item !== void 0
? slots.item
: scope => {
const child = scope.cols.map(
col => vue.h('div', { class: 'q-table__grid-item-row' }, [
vue.h('div', { class: 'q-table__grid-item-title' }, [ col.label ]),
vue.h('div', { class: 'q-table__grid-item-value' }, [ col.value ])
])
);
if (hasSelectionMode.value === true) {
const slot = slots[ 'body-selection' ];
const content = slot !== void 0
? slot(scope)
: [
vue.h(QCheckbox, {
modelValue: scope.selected,
color: props.color,
dark: isDark.value,
dense: props.dense,
'onUpdate:modelValue': (adding, evt) => {
updateSelection([ scope.key ], [ scope.row ], adding, evt);
}
})
];
child.unshift(
vue.h('div', { class: 'q-table__grid-item-row' }, content),
vue.h(QSeparator, { dark: isDark.value })
);
}
const data = {
class: [
'q-table__grid-item-card' + cardDefaultClass.value,
props.cardClass
],
style: props.cardStyle
};
if (
props.onRowClick !== void 0
|| props.onRowDblclick !== void 0
) {
data.class[ 0 ] += ' cursor-pointer';
if (props.onRowClick !== void 0) {
data.onClick = evt => {
emit('RowClick', evt, scope.row, scope.pageIndex);
};
}
if (props.onRowDblclick !== void 0) {
data.onDblclick = evt => {
emit('RowDblclick', evt, scope.row, scope.pageIndex);
};
}
}
return vue.h('div', {
class: 'q-table__grid-item col-xs-12 col-sm-6 col-md-4 col-lg-3'
+ (scope.selected === true ? ' q-table__grid-item--selected' : '')
}, [
vue.h('div', data, child)
])
};
return vue.h('div', {
class: [
'q-table__grid-content row',
props.cardContainerClass
],
style: props.cardContainerStyle
}, computedRows.value.map((row, pageIndex) => {
return item(getBodyScope({
key: getRowKey.value(row),
row,
pageIndex
}))
}))
}
// expose public methods and needed computed props
Object.assign(vm.proxy, {
requestServerInteraction,
setPagination,
firstPage,
prevPage,
nextPage,
lastPage,
isRowSelected,
clearSelection,
isRowExpanded,
setExpanded,
sort,
resetVirtualScroll,
scrollTo,
getCellValue
});
injectMultipleProps(vm.proxy, {
filteredSortedRows: () => filteredSortedRows.value,
computedRows: () => computedRows.value,
computedRowsNumber: () => computedRowsNumber.value
});
return () => {
const child = [ getTopDiv() ];
const data = { ref: rootRef, class: containerClass.value };
if (props.grid === true) {
child.push(getGridHeader());
}
else {
Object.assign(data, {
class: [ data.class, props.cardClass ],
style: props.cardStyle
});
}
child.push(
getBody(),
getBottomDiv()
);
if (props.loading === true && slots.loading !== void 0) {
child.push(
slots.loading()
);
}
return vue.h('div', data, child)
}
}
});
var QTr = createComponent({
name: 'QTr',
props: {
props: Object,
noHover: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() =>
'q-tr'
+ (props.props === void 0 || props.props.header === true ? '' : ' ' + props.props.__trClass)
+ (props.noHover === true ? ' q-tr--no-hover' : '')
);
return () => vue.h('tr', { class: classes.value }, hSlot(slots.default))
}
});
var QTd = createComponent({
name: 'QTd',
props: {
props: Object,
autoWidth: Boolean,
noHover: Boolean
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const classes = vue.computed(() =>
'q-td' + (props.autoWidth === true ? ' q-table--col-auto-width' : '')
+ (props.noHover === true ? ' q-td--no-hover' : '')
+ ' '
);
return () => {
if (props.props === void 0) {
return vue.h('td', { class: classes.value }, hSlot(slots.default))
}
const name = vm.vnode.key;
const col = (
(props.props.colsMap !== void 0 ? props.props.colsMap[ name ] : null)
|| props.props.col
);
if (col === void 0) { return }
const { row } = props.props;
return vue.h('td', {
class: classes.value + col.__tdClass(row),
style: col.__tdStyle(row)
}, hSlot(slots.default))
}
}
});
var QRouteTab = createComponent({
name: 'QRouteTab',
props: {
...useRouterLinkProps,
...useTabProps
},
emits: useTabEmits,
setup (props, { slots, emit }) {
const routeData = useRouterLink({
useDisableForRouterLinkProps: false
});
const { renderTab, $tabs } = useTab(
props,
slots,
emit,
{
exact: vue.computed(() => props.exact),
...routeData
}
);
vue.watch(() => `${ props.name } | ${ props.exact } | ${ (routeData.resolvedLink.value || {}).href }`, () => {
$tabs.verifyRouteModel();
});
return () => renderTab(routeData.linkTag.value, routeData.linkAttrs.value)
}
});
function getViewByModel (model, withSeconds) {
if (model.hour !== null) {
if (model.minute === null) {
return 'minute'
}
else if (withSeconds === true && model.second === null) {
return 'second'
}
}
return 'hour'
}
function getCurrentTime () {
const d = new Date();
return {
hour: d.getHours(),
minute: d.getMinutes(),
second: d.getSeconds(),
millisecond: d.getMilliseconds()
}
}
var QTime = createComponent({
name: 'QTime',
props: {
...useDarkProps,
...useFormProps,
...useDatetimeProps,
mask: {
default: null
},
format24h: {
type: Boolean,
default: null
},
defaultDate: {
type: String,
validator: v => /^-?[\d]+\/[0-1]\d\/[0-3]\d$/.test(v)
},
options: Function,
hourOptions: Array,
minuteOptions: Array,
secondOptions: Array,
withSeconds: Boolean,
nowBtn: Boolean
},
emits: useDatetimeEmits,
setup (props, { slots, emit }) {
const vm = vue.getCurrentInstance();
const { $q } = vm.proxy;
const isDark = useDark(props, $q);
const { tabindex, headerClass, getLocale, getCurrentDate } = useDatetime(props, $q);
const formAttrs = useFormAttrs(props);
const injectFormInput = useFormInject(formAttrs);
let draggingClockRect, dragCache;
const clockRef = vue.ref(null);
const mask = vue.computed(() => getMask());
const locale = vue.computed(() => getLocale());
const defaultDateModel = vue.computed(() => getDefaultDateModel());
const model = __splitDate(
props.modelValue,
mask.value, // initial mask
locale.value, // initial locale
props.calendar,
defaultDateModel.value
);
const view = vue.ref(getViewByModel(model));
const innerModel = vue.ref(model);
const isAM = vue.ref(model.hour === null || model.hour < 12);
const classes = vue.computed(() =>
`q-time q-time--${ props.landscape === true ? 'landscape' : 'portrait' }`
+ (isDark.value === true ? ' q-time--dark q-dark' : '')
+ (props.disable === true ? ' disabled' : (props.readonly === true ? ' q-time--readonly' : ''))
+ (props.bordered === true ? ' q-time--bordered' : '')
+ (props.square === true ? ' q-time--square no-border-radius' : '')
+ (props.flat === true ? ' q-time--flat no-shadow' : '')
);
const stringModel = vue.computed(() => {
const time = innerModel.value;
return {
hour: time.hour === null
? '--'
: (
computedFormat24h.value === true
? pad(time.hour)
: String(
isAM.value === true
? (time.hour === 0 ? 12 : time.hour)
: (time.hour > 12 ? time.hour - 12 : time.hour)
)
),
minute: time.minute === null
? '--'
: pad(time.minute),
second: time.second === null
? '--'
: pad(time.second)
}
});
const computedFormat24h = vue.computed(() => (
props.format24h !== null
? props.format24h
: $q.lang.date.format24h
));
const pointerStyle = vue.computed(() => {
const
forHour = view.value === 'hour',
divider = forHour === true ? 12 : 60,
amount = innerModel.value[ view.value ],
degrees = Math.round(amount * (360 / divider)) - 180;
let transform = `rotate(${ degrees }deg) translateX(-50%)`;
if (
forHour === true
&& computedFormat24h.value === true
&& innerModel.value.hour >= 12
) {
transform += ' scale(.7)';
}
return { transform }
});
const minLink = vue.computed(() => innerModel.value.hour !== null);
const secLink = vue.computed(() => minLink.value === true && innerModel.value.minute !== null);
const hourInSelection = vue.computed(() => (
props.hourOptions !== void 0
? val => props.hourOptions.includes(val)
: (
props.options !== void 0
? val => props.options(val, null, null)
: null
)
));
const minuteInSelection = vue.computed(() => (
props.minuteOptions !== void 0
? val => props.minuteOptions.includes(val)
: (
props.options !== void 0
? val => props.options(innerModel.value.hour, val, null)
: null
)
));
const secondInSelection = vue.computed(() => (
props.secondOptions !== void 0
? val => props.secondOptions.includes(val)
: (
props.options !== void 0
? val => props.options(innerModel.value.hour, innerModel.value.minute, val)
: null
)
));
const validHours = vue.computed(() => {
if (hourInSelection.value === null) {
return null
}
const am = getValidValues(0, 11, hourInSelection.value);
const pm = getValidValues(12, 11, hourInSelection.value);
return { am, pm, values: am.values.concat(pm.values) }
});
const validMinutes = vue.computed(() => (
minuteInSelection.value !== null
? getValidValues(0, 59, minuteInSelection.value)
: null
));
const validSeconds = vue.computed(() => (
secondInSelection.value !== null
? getValidValues(0, 59, secondInSelection.value)
: null
));
const viewValidOptions = vue.computed(() => {
switch (view.value) {
case 'hour':
return validHours.value
case 'minute':
return validMinutes.value
case 'second':
return validSeconds.value
}
});
const positions = vue.computed(() => {
let start, end, offset = 0, step = 1;
const values = viewValidOptions.value !== null
? viewValidOptions.value.values
: void 0;
if (view.value === 'hour') {
if (computedFormat24h.value === true) {
start = 0;
end = 23;
}
else {
start = 0;
end = 11;
if (isAM.value === false) {
offset = 12;
}
}
}
else {
start = 0;
end = 55;
step = 5;
}
const pos = [];
for (let val = start, index = start; val <= end; val += step, index++) {
const
actualVal = val + offset,
disable = values !== void 0 && values.includes(actualVal) === false,
label = view.value === 'hour' && val === 0
? (computedFormat24h.value === true ? '00' : '12')
: val;
pos.push({ val: actualVal, index, disable, label });
}
return pos
});
const clockDirectives = vue.computed(() => {
return [ [
TouchPan,
onPan,
void 0,
{
stop: true,
prevent: true,
mouse: true
}
] ]
});
vue.watch(() => props.modelValue, v => {
const model = __splitDate(
v,
mask.value,
locale.value,
props.calendar,
defaultDateModel.value
);
if (
model.dateHash !== innerModel.value.dateHash
|| model.timeHash !== innerModel.value.timeHash
) {
innerModel.value = model;
if (model.hour === null) {
view.value = 'hour';
}
else {
isAM.value = model.hour < 12;
}
}
});
vue.watch([ mask, locale ], () => {
vue.nextTick(() => {
updateValue();
});
});
function setNow () {
const date = {
...getCurrentDate(),
...getCurrentTime()
};
updateValue(date);
Object.assign(innerModel.value, date); // reset any pending changes to innerModel
view.value = 'hour';
}
function getValidValues (start, count, testFn) {
const values = Array.apply(null, { length: count + 1 })
.map((_, index) => {
const i = index + start;
return {
index: i,
val: testFn(i) === true // force boolean
}
})
.filter(v => v.val === true)
.map(v => v.index);
return {
min: values[ 0 ],
max: values[ values.length - 1 ],
values,
threshold: count + 1
}
}
function getWheelDist (a, b, threshold) {
const diff = Math.abs(a - b);
return Math.min(diff, threshold - diff)
}
function getNormalizedClockValue (val, { min, max, values, threshold }) {
if (val === min) {
return min
}
if (val < min || val > max) {
return getWheelDist(val, min, threshold) <= getWheelDist(val, max, threshold)
? min
: max
}
const
index = values.findIndex(v => val <= v),
before = values[ index - 1 ],
after = values[ index ];
return val - before <= after - val
? before
: after
}
function getMask () {
return props.calendar !== 'persian' && props.mask !== null
? props.mask
: `HH:mm${ props.withSeconds === true ? ':ss' : '' }`
}
function getDefaultDateModel () {
if (typeof props.defaultDate !== 'string') {
const date = getCurrentDate(true);
date.dateHash = getDayHash(date);
return date
}
return __splitDate(props.defaultDate, 'YYYY/MM/DD', void 0, props.calendar)
}
function shouldAbortInteraction () {
return vmIsDestroyed(vm) === true
// if we have limited options, can we actually set any?
|| (
viewValidOptions.value !== null
&& (
viewValidOptions.value.values.length === 0
|| (
view.value === 'hour' && computedFormat24h.value !== true
&& validHours.value[ isAM.value === true ? 'am' : 'pm' ].values.length === 0
)
)
)
}
function getClockRect () {
const
clock = clockRef.value,
{ top, left, width } = clock.getBoundingClientRect(),
dist = width / 2;
return {
top: top + dist,
left: left + dist,
dist: dist * 0.7
}
}
function onPan (event) {
if (shouldAbortInteraction() === true) {
return
}
if (event.isFirst === true) {
draggingClockRect = getClockRect();
dragCache = updateClock(event.evt, draggingClockRect);
return
}
dragCache = updateClock(event.evt, draggingClockRect, dragCache);
if (event.isFinal === true) {
draggingClockRect = false;
dragCache = null;
goToNextView();
}
}
function goToNextView () {
if (view.value === 'hour') {
view.value = 'minute';
}
else if (props.withSeconds && view.value === 'minute') {
view.value = 'second';
}
}
function updateClock (evt, clockRect, cacheVal) {
const
pos = position(evt),
height = Math.abs(pos.top - clockRect.top),
distance = Math.sqrt(
Math.pow(Math.abs(pos.top - clockRect.top), 2)
+ Math.pow(Math.abs(pos.left - clockRect.left), 2)
);
let
val,
angle = Math.asin(height / distance) * (180 / Math.PI);
if (pos.top < clockRect.top) {
angle = clockRect.left < pos.left ? 90 - angle : 270 + angle;
}
else {
angle = clockRect.left < pos.left ? angle + 90 : 270 - angle;
}
if (view.value === 'hour') {
val = angle / 30;
if (validHours.value !== null) {
const am = computedFormat24h.value !== true
? isAM.value === true
: (
validHours.value.am.values.length !== 0 && validHours.value.pm.values.length !== 0
? distance >= clockRect.dist
: validHours.value.am.values.length !== 0
);
val = getNormalizedClockValue(
val + (am === true ? 0 : 12),
validHours.value[ am === true ? 'am' : 'pm' ]
);
}
else {
val = Math.round(val);
if (computedFormat24h.value === true) {
if (distance < clockRect.dist) {
if (val < 12) {
val += 12;
}
}
else if (val === 12) {
val = 0;
}
}
else if (isAM.value === true && val === 12) {
val = 0;
}
else if (isAM.value === false && val !== 12) {
val += 12;
}
}
if (computedFormat24h.value === true) {
isAM.value = val < 12;
}
}
else {
val = Math.round(angle / 6) % 60;
if (view.value === 'minute' && validMinutes.value !== null) {
val = getNormalizedClockValue(val, validMinutes.value);
}
else if (view.value === 'second' && validSeconds.value !== null) {
val = getNormalizedClockValue(val, validSeconds.value);
}
}
if (cacheVal !== val) {
setModel[ view.value ](val);
}
return val
}
const setView = {
hour () { view.value = 'hour'; },
minute () { view.value = 'minute'; },
second () { view.value = 'second'; }
};
function setAmOnKey (e) {
e.keyCode === 13 && setAm();
}
function setPmOnKey (e) {
e.keyCode === 13 && setPm();
}
function onClick (evt) {
if (shouldAbortInteraction() !== true) {
// onMousedown() has already updated the offset
// (on desktop only, through mousedown event)
if ($q.platform.is.desktop !== true) {
updateClock(evt, getClockRect());
}
goToNextView();
}
}
function onMousedown (evt) {
if (shouldAbortInteraction() !== true) {
updateClock(evt, getClockRect());
}
}
function onKeyupHour (e) {
if (e.keyCode === 13) { // ENTER
view.value = 'hour';
}
else if ([ 37, 39 ].includes(e.keyCode)) {
const payload = e.keyCode === 37 ? -1 : 1;
if (validHours.value !== null) {
const values = computedFormat24h.value === true
? validHours.value.values
: validHours.value[ isAM.value === true ? 'am' : 'pm' ].values;
if (values.length === 0) { return }
if (innerModel.value.hour === null) {
setHour(values[ 0 ]);
}
else {
const index = (
values.length
+ values.indexOf(innerModel.value.hour)
+ payload
) % values.length;
setHour(values[ index ]);
}
}
else {
const
wrap = computedFormat24h.value === true ? 24 : 12,
offset = computedFormat24h.value !== true && isAM.value === false ? 12 : 0,
val = innerModel.value.hour === null ? -payload : innerModel.value.hour;
setHour(offset + (24 + val + payload) % wrap);
}
}
}
function onKeyupMinute (e) {
if (e.keyCode === 13) { // ENTER
view.value = 'minute';
}
else if ([ 37, 39 ].includes(e.keyCode)) {
const payload = e.keyCode === 37 ? -1 : 1;
if (validMinutes.value !== null) {
const values = validMinutes.value.values;
if (values.length === 0) { return }
if (innerModel.value.minute === null) {
setMinute(values[ 0 ]);
}
else {
const index = (
values.length
+ values.indexOf(innerModel.value.minute)
+ payload
) % values.length;
setMinute(values[ index ]);
}
}
else {
const val = innerModel.value.minute === null ? -payload : innerModel.value.minute;
setMinute((60 + val + payload) % 60);
}
}
}
function onKeyupSecond (e) {
if (e.keyCode === 13) { // ENTER
view.value = 'second';
}
else if ([ 37, 39 ].includes(e.keyCode)) {
const payload = e.keyCode === 37 ? -1 : 1;
if (validSeconds.value !== null) {
const values = validSeconds.value.values;
if (values.length === 0) { return }
if (innerModel.value.seconds === null) {
setSecond(values[ 0 ]);
}
else {
const index = (
values.length
+ values.indexOf(innerModel.value.second)
+ payload
) % values.length;
setSecond(values[ index ]);
}
}
else {
const val = innerModel.value.second === null ? -payload : innerModel.value.second;
setSecond((60 + val + payload) % 60);
}
}
}
function setHour (hour) {
if (innerModel.value.hour !== hour) {
innerModel.value.hour = hour;
verifyAndUpdate();
}
}
function setMinute (minute) {
if (innerModel.value.minute !== minute) {
innerModel.value.minute = minute;
verifyAndUpdate();
}
}
function setSecond (second) {
if (innerModel.value.second !== second) {
innerModel.value.second = second;
verifyAndUpdate();
}
}
const setModel = {
hour: setHour,
minute: setMinute,
second: setSecond
};
function setAm () {
if (isAM.value === false) {
isAM.value = true;
if (innerModel.value.hour !== null) {
innerModel.value.hour -= 12;
verifyAndUpdate();
}
}
}
function setPm () {
if (isAM.value === true) {
isAM.value = false;
if (innerModel.value.hour !== null) {
innerModel.value.hour += 12;
verifyAndUpdate();
}
}
}
function verifyAndUpdate () {
if (hourInSelection.value !== null && hourInSelection.value(innerModel.value.hour) !== true) {
innerModel.value = __splitDate();
view.value = 'hour';
return
}
if (minuteInSelection.value !== null && minuteInSelection.value(innerModel.value.minute) !== true) {
innerModel.value.minute = null;
innerModel.value.second = null;
view.value = 'minute';
return
}
if (props.withSeconds === true && secondInSelection.value !== null && secondInSelection.value(innerModel.value.second) !== true) {
innerModel.value.second = null;
view.value = 'second';
return
}
if (innerModel.value.hour === null || innerModel.value.minute === null || (props.withSeconds === true && innerModel.value.second === null)) {
return
}
updateValue();
}
function updateValue (obj) {
const date = Object.assign({ ...innerModel.value }, obj);
const val = props.calendar === 'persian'
? pad(date.hour) + ':'
+ pad(date.minute)
+ (props.withSeconds === true ? ':' + pad(date.second) : '')
: formatDate(
new Date(
date.year,
date.month === null ? null : date.month - 1,
date.day,
date.hour,
date.minute,
date.second,
date.millisecond
),
mask.value,
locale.value,
date.year,
date.timezoneOffset
);
date.changed = val !== props.modelValue;
emit('update:modelValue', val, date);
}
function getHeader () {
const label = [
vue.h('div', {
class: 'q-time__link '
+ (view.value === 'hour' ? 'q-time__link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
onClick: setView.hour,
onKeyup: onKeyupHour
}, stringModel.value.hour),
vue.h('div', ':'),
vue.h(
'div',
minLink.value === true
? {
class: 'q-time__link '
+ (view.value === 'minute' ? 'q-time__link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
onKeyup: onKeyupMinute,
onClick: setView.minute
}
: { class: 'q-time__link' },
stringModel.value.minute
)
];
if (props.withSeconds === true) {
label.push(
vue.h('div', ':'),
vue.h(
'div',
secLink.value === true
? {
class: 'q-time__link '
+ (view.value === 'second' ? 'q-time__link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
onKeyup: onKeyupSecond,
onClick: setView.second
}
: { class: 'q-time__link' },
stringModel.value.second
)
);
}
const child = [
vue.h('div', {
class: 'q-time__header-label row items-center no-wrap',
dir: 'ltr'
}, label)
];
computedFormat24h.value === false && child.push(
vue.h('div', {
class: 'q-time__header-ampm column items-between no-wrap'
}, [
vue.h('div', {
class: 'q-time__link '
+ (isAM.value === true ? 'q-time__link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
onClick: setAm,
onKeyup: setAmOnKey
}, 'AM'),
vue.h('div', {
class: 'q-time__link '
+ (isAM.value !== true ? 'q-time__link--active' : 'cursor-pointer'),
tabindex: tabindex.value,
onClick: setPm,
onKeyup: setPmOnKey
}, 'PM')
])
);
return vue.h('div', {
class: 'q-time__header flex flex-center no-wrap ' + headerClass.value
}, child)
}
function getClock () {
const current = innerModel.value[ view.value ];
return vue.h('div', {
class: 'q-time__content col relative-position'
}, [
vue.h(vue.Transition, {
name: 'q-transition--scale'
}, () => vue.h('div', {
key: 'clock' + view.value,
class: 'q-time__container-parent absolute-full'
}, [
vue.h('div', {
ref: clockRef,
class: 'q-time__container-child fit overflow-hidden'
}, [
vue.withDirectives(
vue.h('div', {
class: 'q-time__clock cursor-pointer non-selectable',
onClick,
onMousedown
}, [
vue.h('div', { class: 'q-time__clock-circle fit' }, [
vue.h('div', {
class: 'q-time__clock-pointer'
+ (innerModel.value[ view.value ] === null ? ' hidden' : (props.color !== void 0 ? ` text-${ props.color }` : '')),
style: pointerStyle.value
}),
positions.value.map(pos => vue.h('div', {
class: `q-time__clock-position row flex-center q-time__clock-pos-${ pos.index }`
+ (pos.val === current
? ' q-time__clock-position--active ' + headerClass.value
: (pos.disable === true ? ' q-time__clock-position--disable' : ''))
}, [ vue.h('span', pos.label) ]))
])
]),
clockDirectives.value
)
])
])),
props.nowBtn === true ? vue.h(QBtn, {
class: 'q-time__now-button absolute',
icon: $q.iconSet.datetime.now,
unelevated: true,
size: 'sm',
round: true,
color: props.color,
textColor: props.textColor,
tabindex: tabindex.value,
onClick: setNow
}) : null
])
}
// expose public method
vm.proxy.setNow = setNow;
return () => {
const child = [ getClock() ];
const def = hSlot(slots.default);
def !== void 0 && child.push(
vue.h('div', { class: 'q-time__actions' }, def)
);
if (props.name !== void 0 && props.disable !== true) {
injectFormInput(child, 'push');
}
return vue.h('div', {
class: classes.value,
tabindex: -1
}, [
getHeader(),
vue.h('div', { class: 'q-time__main col overflow-auto' }, child)
])
}
}
});
var QTimeline = createComponent({
name: 'QTimeline',
props: {
...useDarkProps,
color: {
type: String,
default: 'primary'
},
side: {
type: String,
default: 'right',
validator: v => [ 'left', 'right' ].includes(v)
},
layout: {
type: String,
default: 'dense',
validator: v => [ 'dense', 'comfortable', 'loose' ].includes(v)
}
},
setup (props, { slots }) {
const vm = vue.getCurrentInstance();
const isDark = useDark(props, vm.proxy.$q);
vue.provide(timelineKey, props);
const classes = vue.computed(() =>
`q-timeline q-timeline--${ props.layout } q-timeline--${ props.layout }--${ props.side }`
+ (isDark.value === true ? ' q-timeline--dark' : '')
);
return () => vue.h('ul', { class: classes.value }, hSlot(slots.default))
}
});
var QTimelineEntry = createComponent({
name: 'QTimelineEntry',
props: {
heading: Boolean,
tag: {
type: String,
default: 'h3'
},
side: {
type: String,
default: 'right',
validator: v => [ 'left', 'right' ].includes(v)
},
icon: String,
avatar: String,
color: String,
title: String,
subtitle: String,
body: String
},
setup (props, { slots }) {
const $timeline = vue.inject(timelineKey, emptyRenderFn);
if ($timeline === emptyRenderFn) {
console.error('QTimelineEntry needs to be child of QTimeline');
return emptyRenderFn
}
const classes = vue.computed(() =>
`q-timeline__entry q-timeline__entry--${ props.side }`
+ (props.icon !== void 0 || props.avatar !== void 0 ? ' q-timeline__entry--icon' : '')
);
const dotClass = vue.computed(() =>
`q-timeline__dot text-${ props.color || $timeline.color }`
);
const reverse = vue.computed(() =>
$timeline.layout === 'comfortable' && $timeline.side === 'left'
);
return () => {
const child = hUniqueSlot(slots.default, []);
if (props.body !== void 0) {
child.unshift(props.body);
}
if (props.heading === true) {
const content = [
vue.h('div'),
vue.h('div'),
vue.h(
props.tag,
{ class: 'q-timeline__heading-title' },
child
)
];
return vue.h('div', {
class: 'q-timeline__heading'
}, reverse.value === true ? content.reverse() : content)
}
let dot;
if (props.icon !== void 0) {
dot = [
vue.h(QIcon, {
class: 'row items-center justify-center',
name: props.icon
})
];
}
else if (props.avatar !== void 0) {
dot = [
vue.h('img', {
class: 'q-timeline__dot-img',
src: props.avatar
})
];
}
const content = [
vue.h('div', { class: 'q-timeline__subtitle' }, [
vue.h('span', {}, hSlot(slots.subtitle, [ props.subtitle ]))
]),
vue.h('div', { class: dotClass.value }, dot),
vue.h('div', { class: 'q-timeline__content' }, [
vue.h('h6', { class: 'q-timeline__title' }, hSlot(slots.title, [ props.title ]))
].concat(child))
];
return vue.h('li', {
class: classes.value
}, reverse.value === true ? content.reverse() : content)
}
}
});
var QToolbar = createComponent({
name: 'QToolbar',
props: {
inset: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() =>
'q-toolbar row no-wrap items-center'
+ (props.inset === true ? ' q-toolbar--inset' : '')
);
return () => vue.h('div', { class: classes.value, role: 'toolbar' }, hSlot(slots.default))
}
});
var QToolbarTitle = createComponent({
name: 'QToolbarTitle',
props: {
shrink: Boolean
},
setup (props, { slots }) {
const classes = vue.computed(() =>
'q-toolbar__title ellipsis'
+ (props.shrink === true ? ' col-shrink' : '')
);
return () => vue.h('div', { class: classes.value }, hSlot(slots.default))
}
});
const tickStrategyOptions = [ 'none', 'strict', 'leaf', 'leaf-filtered' ];
var QTree = createComponent({
name: 'QTree',
props: {
...useDarkProps,
nodes: {
type: Array,
required: true
},
nodeKey: {
type: String,
required: true
},
labelKey: {
type: String,
default: 'label'
},
childrenKey: {
type: String,
default: 'children'
},
dense: Boolean,
color: String,
controlColor: String,
textColor: String,
selectedColor: String,
icon: String,
tickStrategy: {
type: String,
default: 'none',
validator: v => tickStrategyOptions.includes(v)
},
ticked: Array, // v-model:ticked
expanded: Array, // v-model:expanded
selected: {}, // v-model:selected
noSelectionUnset: Boolean,
defaultExpandAll: Boolean,
accordion: Boolean,
filter: String,
filterMethod: Function,
duration: Number,
noConnectors: Boolean,
noTransition: Boolean,
noNodesLabel: String,
noResultsLabel: String
},
emits: [
'update:expanded',
'update:ticked',
'update:selected',
'lazyLoad',
'afterShow',
'afterHide'
],
setup (props, { slots, emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const lazy = vue.ref({});
const innerTicked = vue.ref(props.ticked || []);
const innerExpanded = vue.ref(props.expanded || []);
let blurTargets = {};
vue.onBeforeUpdate(() => {
blurTargets = {};
});
const classes = vue.computed(() =>
`q-tree q-tree--${ props.dense === true ? 'dense' : 'standard' }`
+ (props.noConnectors === true ? ' q-tree--no-connectors' : '')
+ (isDark.value === true ? ' q-tree--dark' : '')
+ (props.color !== void 0 ? ` text-${ props.color }` : '')
);
const hasSelection = vue.computed(() => props.selected !== void 0);
const computedIcon = vue.computed(() => props.icon || $q.iconSet.tree.icon);
const computedControlColor = vue.computed(() => props.controlColor || props.color);
const textColorClass = vue.computed(() => (
props.textColor !== void 0
? ` text-${ props.textColor }`
: ''
));
const selectedColorClass = vue.computed(() => {
const color = props.selectedColor || props.color;
return color ? ` text-${ color }` : ''
});
const computedFilterMethod = vue.computed(() => (
props.filterMethod !== void 0
? props.filterMethod
: (node, filter) => {
const filt = filter.toLowerCase();
return node[ props.labelKey ]
&& node[ props.labelKey ].toLowerCase().indexOf(filt) > -1
}
));
const meta = vue.computed(() => {
const meta = {};
const travel = (node, parent) => {
const tickStrategy = node.tickStrategy || (parent ? parent.tickStrategy : props.tickStrategy);
const
key = node[ props.nodeKey ],
isParent = node[ props.childrenKey ] && Array.isArray(node[ props.childrenKey ]) && node[ props.childrenKey ].length !== 0,
selectable = node.disabled !== true && hasSelection.value === true && node.selectable !== false,
expandable = node.disabled !== true && node.expandable !== false,
hasTicking = tickStrategy !== 'none',
strictTicking = tickStrategy === 'strict',
leafFilteredTicking = tickStrategy === 'leaf-filtered',
leafTicking = tickStrategy === 'leaf' || tickStrategy === 'leaf-filtered';
let tickable = node.disabled !== true && node.tickable !== false;
if (leafTicking === true && tickable === true && parent && parent.tickable !== true) {
tickable = false;
}
let localLazy = node.lazy;
if (
localLazy === true
&& lazy.value[ key ] !== void 0
&& Array.isArray(node[ props.childrenKey ]) === true
) {
localLazy = lazy.value[ key ];
}
const m = {
key,
parent,
isParent,
lazy: localLazy,
disabled: node.disabled,
link: node.disabled !== true && (selectable === true || (expandable === true && (isParent === true || localLazy === true))),
children: [],
matchesFilter: props.filter ? computedFilterMethod.value(node, props.filter) : true,
selected: key === props.selected && selectable === true,
selectable,
expanded: isParent === true ? innerExpanded.value.includes(key) : false,
expandable,
noTick: node.noTick === true || (strictTicking !== true && localLazy && localLazy !== 'loaded'),
tickable,
tickStrategy,
hasTicking,
strictTicking,
leafFilteredTicking,
leafTicking,
ticked: strictTicking === true
? innerTicked.value.includes(key)
: (isParent === true ? false : innerTicked.value.includes(key))
};
meta[ key ] = m;
if (isParent === true) {
m.children = node[ props.childrenKey ].map(n => travel(n, m));
if (props.filter) {
if (m.matchesFilter !== true) {
m.matchesFilter = m.children.some(n => n.matchesFilter);
}
else if (
m.noTick !== true
&& m.disabled !== true
&& m.tickable === true
&& leafFilteredTicking === true
&& m.children.every(n => n.matchesFilter !== true || n.noTick === true || n.tickable !== true) === true
) {
m.tickable = false;
}
}
if (m.matchesFilter === true) {
if (m.noTick !== true && strictTicking !== true && m.children.every(n => n.noTick) === true) {
m.noTick = true;
}
if (leafTicking) {
m.ticked = false;
m.indeterminate = m.children.some(node => node.indeterminate === true);
m.tickable = m.tickable === true && m.children.some(node => node.tickable);
if (m.indeterminate !== true) {
const sel = m.children
.reduce((acc, meta) => (meta.ticked === true ? acc + 1 : acc), 0);
if (sel === m.children.length) {
m.ticked = true;
}
else if (sel > 0) {
m.indeterminate = true;
}
}
if (m.indeterminate === true) {
m.indeterminateNextState = m.children
.every(meta => meta.tickable !== true || meta.ticked !== true);
}
}
}
}
return m
};
props.nodes.forEach(node => travel(node, null));
return meta
});
vue.watch(() => props.ticked, val => {
innerTicked.value = val;
});
vue.watch(() => props.expanded, val => {
innerExpanded.value = val;
});
function getNodeByKey (key) {
const reduce = [].reduce;
const find = (result, node) => {
if (result || !node) {
return result
}
if (Array.isArray(node) === true) {
return reduce.call(Object(node), find, result)
}
if (node[ props.nodeKey ] === key) {
return node
}
if (node[ props.childrenKey ]) {
return find(null, node[ props.childrenKey ])
}
};
return find(null, props.nodes)
}
function getTickedNodes () {
return innerTicked.value.map(key => getNodeByKey(key))
}
function getExpandedNodes () {
return innerExpanded.value.map(key => getNodeByKey(key))
}
function isExpanded (key) {
return key && meta.value[ key ]
? meta.value[ key ].expanded
: false
}
function collapseAll () {
if (props.expanded !== void 0) {
emit('update:expanded', []);
}
else {
innerExpanded.value = [];
}
}
function expandAll () {
const expanded = [];
const travel = node => {
if (node[ props.childrenKey ] && node[ props.childrenKey ].length !== 0) {
if (node.expandable !== false && node.disabled !== true) {
expanded.push(node[ props.nodeKey ]);
node[ props.childrenKey ].forEach(travel);
}
}
};
props.nodes.forEach(travel);
if (props.expanded !== void 0) {
emit('update:expanded', expanded);
}
else {
innerExpanded.value = expanded;
}
}
function setExpanded (key, state, node = getNodeByKey(key), m = meta.value[ key ]) {
if (m.lazy && m.lazy !== 'loaded') {
if (m.lazy === 'loading') {
return
}
lazy.value[ key ] = 'loading';
if (Array.isArray(node[ props.childrenKey ]) !== true) {
node[ props.childrenKey ] = [];
}
emit('lazyLoad', {
node,
key,
done: children => {
lazy.value[ key ] = 'loaded';
node[ props.childrenKey ] = Array.isArray(children) === true ? children : [];
vue.nextTick(() => {
const localMeta = meta.value[ key ];
if (localMeta && localMeta.isParent === true) {
localSetExpanded(key, true);
}
});
},
fail: () => {
delete lazy.value[ key ];
if (node[ props.childrenKey ].length === 0) {
delete node[ props.childrenKey ];
}
}
});
}
else if (m.isParent === true && m.expandable === true) {
localSetExpanded(key, state);
}
}
function localSetExpanded (key, state) {
let target = innerExpanded.value;
const shouldEmit = props.expanded !== void 0;
if (shouldEmit === true) {
target = target.slice();
}
if (state) {
if (props.accordion) {
if (meta.value[ key ]) {
const collapse = [];
if (meta.value[ key ].parent) {
meta.value[ key ].parent.children.forEach(m => {
if (m.key !== key && m.expandable === true) {
collapse.push(m.key);
}
});
}
else {
props.nodes.forEach(node => {
const k = node[ props.nodeKey ];
if (k !== key) {
collapse.push(k);
}
});
}
if (collapse.length !== 0) {
target = target.filter(k => collapse.includes(k) === false);
}
}
}
target = target.concat([ key ])
.filter((key, index, self) => self.indexOf(key) === index);
}
else {
target = target.filter(k => k !== key);
}
if (shouldEmit === true) {
emit('update:expanded', target);
}
else {
innerExpanded.value = target;
}
}
function isTicked (key) {
return key && meta.value[ key ]
? meta.value[ key ].ticked
: false
}
function setTicked (keys, state) {
let target = innerTicked.value;
const shouldEmit = props.ticked !== void 0;
if (shouldEmit === true) {
target = target.slice();
}
if (state) {
target = target.concat(keys)
.filter((key, index, self) => self.indexOf(key) === index);
}
else {
target = target.filter(k => keys.includes(k) === false);
}
if (shouldEmit === true) {
emit('update:ticked', target);
}
}
function getSlotScope (node, meta, key) {
const scope = { tree: proxy, node, key, color: props.color, dark: isDark.value };
injectProp(
scope,
'expanded',
() => { return meta.expanded },
val => { val !== meta.expanded && setExpanded(key, val); }
);
injectProp(
scope,
'ticked',
() => { return meta.ticked },
val => { val !== meta.ticked && setTicked([ key ], val); }
);
return scope
}
function getChildren (nodes) {
return (
props.filter
? nodes.filter(n => meta.value[ n[ props.nodeKey ] ].matchesFilter)
: nodes
).map(child => getNode(child))
}
function getNodeMedia (node) {
if (node.icon !== void 0) {
return vue.h(QIcon, {
class: 'q-tree__icon q-mr-sm',
name: node.icon,
color: node.iconColor
})
}
const src = node.img || node.avatar;
if (src) {
return vue.h('img', {
class: `q-tree__${ node.img ? 'img' : 'avatar' } q-mr-sm`,
src
})
}
}
function onShow () {
emit('afterShow');
}
function onHide () {
emit('afterHide');
}
function getNode (node) {
const
key = node[ props.nodeKey ],
m = meta.value[ key ],
header = node.header
? slots[ `header-${ node.header }` ] || slots[ 'default-header' ]
: slots[ 'default-header' ];
const children = m.isParent === true
? getChildren(node[ props.childrenKey ])
: [];
const isParent = children.length !== 0 || (m.lazy && m.lazy !== 'loaded');
let body = node.body
? slots[ `body-${ node.body }` ] || slots[ 'default-body' ]
: slots[ 'default-body' ];
const slotScope = header !== void 0 || body !== void 0
? getSlotScope(node, m, key)
: null;
if (body !== void 0) {
body = vue.h('div', { class: 'q-tree__node-body relative-position' }, [
vue.h('div', { class: textColorClass.value }, [
body(slotScope)
])
]);
}
return vue.h('div', {
key,
class: 'q-tree__node relative-position'
+ ` q-tree__node--${ isParent === true ? 'parent' : 'child' }`
}, [
vue.h('div', {
class: 'q-tree__node-header relative-position row no-wrap items-center'
+ (m.link === true ? ' q-tree__node--link q-hoverable q-focusable' : '')
+ (m.selected === true ? ' q-tree__node--selected' : '')
+ (m.disabled === true ? ' q-tree__node--disabled' : ''),
tabindex: m.link === true ? 0 : -1,
onClick: (e) => {
onClick(node, m, e);
},
onKeypress (e) {
if (shouldIgnoreKey(e) !== true) {
if (e.keyCode === 13) { onClick(node, m, e, true); }
else if (e.keyCode === 32) { onExpandClick(node, m, e, true); }
}
}
}, [
vue.h('div', {
class: 'q-focus-helper',
tabindex: -1,
ref: el => { blurTargets[ m.key ] = el; }
}),
m.lazy === 'loading'
? vue.h(QSpinner, {
class: 'q-tree__spinner',
color: computedControlColor.value
})
: (
isParent === true
? vue.h(QIcon, {
class: 'q-tree__arrow'
+ (m.expanded === true ? ' q-tree__arrow--rotate' : ''),
name: computedIcon.value,
onClick (e) { onExpandClick(node, m, e); }
})
: null
),
m.hasTicking === true && m.noTick !== true
? vue.h(QCheckbox, {
class: 'q-tree__tickbox',
modelValue: m.indeterminate === true ? null : m.ticked,
color: computedControlColor.value,
dark: isDark.value,
dense: true,
keepColor: true,
disable: m.tickable !== true,
onKeydown: stopAndPrevent,
'onUpdate:modelValue': v => {
onTickedClick(m, v);
}
})
: null,
vue.h('div', {
class: 'q-tree__node-header-content col row no-wrap items-center'
+ (m.selected === true ? selectedColorClass.value : textColorClass.value)
}, [
header
? header(slotScope)
: [
getNodeMedia(node),
vue.h('div', node[ props.labelKey ])
]
])
]),
isParent === true
? (
props.noTransition === true
? vue.h('div', {
class: 'q-tree__node-collapsible' + textColorClass.value,
key: `${ key }__q`
}, [
body,
vue.h('div', {
class: 'q-tree__children'
+ (m.disabled === true ? ' q-tree__node--disabled' : '')
}, m.expanded ? children : null)
])
: vue.h(QSlideTransition, {
duration: props.duration,
onShow,
onHide
}, () => vue.withDirectives(
vue.h('div', {
class: 'q-tree__node-collapsible' + textColorClass.value,
key: `${ key }__q`
}, [
body,
vue.h('div', {
class: 'q-tree__children'
+ (m.disabled === true ? ' q-tree__node--disabled' : '')
}, children)
]),
[ [ vue.vShow, m.expanded ] ]
))
)
: body
])
}
function blur (key) {
const blurTarget = blurTargets[ key ];
blurTarget && blurTarget.focus();
}
function onClick (node, meta, e, keyboard) {
keyboard !== true && meta.selectable !== false && blur(meta.key);
if (hasSelection.value && meta.selectable) {
if (props.noSelectionUnset === false) {
emit('update:selected', meta.key !== props.selected ? meta.key : null);
}
else if (meta.key !== props.selected) {
emit('update:selected', meta.key === void 0 ? null : meta.key);
}
}
else {
onExpandClick(node, meta, e, keyboard);
}
if (typeof node.handler === 'function') {
node.handler(node);
}
}
function onExpandClick (node, meta, e, keyboard) {
if (e !== void 0) {
stopAndPrevent(e);
}
keyboard !== true && meta.selectable !== false && blur(meta.key);
setExpanded(meta.key, !meta.expanded, node, meta);
}
function onTickedClick (meta, state) {
if (meta.indeterminate === true) {
state = meta.indeterminateNextState;
}
if (meta.strictTicking) {
setTicked([ meta.key ], state);
}
else if (meta.leafTicking) {
const keys = [];
const travel = meta => {
if (meta.isParent) {
if (state !== true && meta.noTick !== true && meta.tickable === true) {
keys.push(meta.key);
}
if (meta.leafTicking === true) {
meta.children.forEach(travel);
}
}
else if (
meta.noTick !== true
&& meta.tickable === true
&& (meta.leafFilteredTicking !== true || meta.matchesFilter === true)
) {
keys.push(meta.key);
}
};
travel(meta);
setTicked(keys, state);
}
}
props.defaultExpandAll === true && expandAll();
// expose public methods
Object.assign(proxy, {
getNodeByKey,
getTickedNodes,
getExpandedNodes,
isExpanded,
collapseAll,
expandAll,
setExpanded,
isTicked,
setTicked
});
return () => {
const children = getChildren(props.nodes);
return vue.h(
'div', {
class: classes.value
},
children.length === 0
? (
props.filter
? props.noResultsLabel || $q.lang.tree.noResults
: props.noNodesLabel || $q.lang.tree.noNodes
)
: children
)
}
}
});
function getProgressLabel (p) {
return (p * 100).toFixed(2) + '%'
}
const coreProps = {
...useDarkProps,
...useFileProps,
label: String,
color: String,
textColor: String,
square: Boolean,
flat: Boolean,
bordered: Boolean,
noThumbnails: Boolean,
autoUpload: Boolean,
hideUploadBtn: Boolean,
disable: Boolean,
readonly: Boolean
};
const coreEmits = [
...useFileEmits,
'start', 'finish', 'added', 'removed'
];
function getRenderer (getPlugin, expose) {
const vm = vue.getCurrentInstance();
const { props, slots, emit, proxy } = vm;
const { $q } = proxy;
const isDark = useDark(props, $q);
function updateFileStatus (file, status, uploadedSize) {
file.__status = status;
if (status === 'idle') {
file.__uploaded = 0;
file.__progress = 0;
file.__sizeLabel = humanStorageSize(file.size);
file.__progressLabel = '0.00%';
return
}
if (status === 'failed') {
proxy.$forceUpdate();
return
}
file.__uploaded = status === 'uploaded'
? file.size
: uploadedSize;
file.__progress = status === 'uploaded'
? 1
: Math.min(0.9999, file.__uploaded / file.size);
file.__progressLabel = getProgressLabel(file.__progress);
proxy.$forceUpdate();
}
const editable = vue.computed(() => props.disable !== true && props.readonly !== true);
const dnd = vue.ref(false);
const rootRef = vue.ref(null);
const inputRef = vue.ref(null);
const state = {
files: vue.ref([]),
queuedFiles: vue.ref([]),
uploadedFiles: vue.ref([]),
uploadedSize: vue.ref(0),
updateFileStatus,
isAlive: () => vmIsDestroyed(vm) === false
};
const {
pickFiles,
addFiles,
onDragover,
onDragleave,
processFiles,
getDndNode,
maxFilesNumber,
maxTotalSizeNumber
} = useFile({ editable, dnd, getFileInput, addFilesToQueue });
Object.assign(state, getPlugin({
props,
slots,
emit,
helpers: state,
exposeApi: obj => { Object.assign(state, obj); }
}));
if (state.isBusy === void 0) {
state.isBusy = vue.ref(false);
}
const uploadSize = vue.ref(0);
const uploadProgress = vue.computed(() => (
uploadSize.value === 0
? 0
: state.uploadedSize.value / uploadSize.value
));
const uploadProgressLabel = vue.computed(() => getProgressLabel(uploadProgress.value));
const uploadSizeLabel = vue.computed(() => humanStorageSize(uploadSize.value));
const canAddFiles = vue.computed(() =>
editable.value === true
&& state.isUploading.value !== true
// if single selection and no files are queued:
&& (props.multiple === true || state.queuedFiles.value.length === 0)
// if max-files is set and current number of files does not exceeds it:
&& (props.maxFiles === void 0 || state.files.value.length < maxFilesNumber.value)
// if max-total-size is set and current upload size does not exceeds it:
&& (props.maxTotalSize === void 0 || uploadSize.value < maxTotalSizeNumber.value)
);
const canUpload = vue.computed(() =>
editable.value === true
&& state.isBusy.value !== true
&& state.isUploading.value !== true
&& state.queuedFiles.value.length !== 0
);
vue.provide(uploaderKey, renderInput);
const classes = vue.computed(() =>
'q-uploader column no-wrap'
+ (isDark.value === true ? ' q-uploader--dark q-dark' : '')
+ (props.bordered === true ? ' q-uploader--bordered' : '')
+ (props.square === true ? ' q-uploader--square no-border-radius' : '')
+ (props.flat === true ? ' q-uploader--flat no-shadow' : '')
+ (props.disable === true ? ' disabled q-uploader--disable' : '')
+ (dnd.value === true ? ' q-uploader--dnd' : '')
);
const colorClass = vue.computed(() =>
'q-uploader__header'
+ (props.color !== void 0 ? ` bg-${ props.color }` : '')
+ (props.textColor !== void 0 ? ` text-${ props.textColor }` : '')
);
vue.watch(state.isUploading, (newVal, oldVal) => {
if (oldVal === false && newVal === true) {
emit('start');
}
else if (oldVal === true && newVal === false) {
emit('finish');
}
});
function reset () {
if (props.disable === false) {
state.abort();
state.uploadedSize.value = 0;
uploadSize.value = 0;
revokeImgURLs();
state.files.value = [];
state.queuedFiles.value = [];
state.uploadedFiles.value = [];
}
}
function removeUploadedFiles () {
if (props.disable === false) {
batchRemoveFiles([ 'uploaded' ], () => {
state.uploadedFiles.value = [];
});
}
}
function removeQueuedFiles () {
batchRemoveFiles([ 'idle', 'failed' ], ({ size }) => {
uploadSize.value -= size;
state.queuedFiles.value = [];
});
}
function batchRemoveFiles (statusList, cb) {
if (props.disable === true) {
return
}
const removed = {
files: [],
size: 0
};
const localFiles = state.files.value.filter(f => {
if (statusList.indexOf(f.__status) === -1) {
return true
}
removed.size += f.size;
removed.files.push(f);
f.__img !== void 0 && window.URL.revokeObjectURL(f.__img.src);
return false
});
if (removed.files.length !== 0) {
state.files.value = localFiles;
cb(removed);
emit('removed', removed.files);
}
}
function removeFile (file) {
if (props.disable) { return }
if (file.__status === 'uploaded') {
state.uploadedFiles.value = state.uploadedFiles.value.filter(f => f.__key !== file.__key);
}
else if (file.__status === 'uploading') {
file.__abort();
}
else {
uploadSize.value -= file.size;
}
state.files.value = state.files.value.filter(f => {
if (f.__key !== file.__key) {
return true
}
f.__img !== void 0 && window.URL.revokeObjectURL(f.__img.src);
return false
});
state.queuedFiles.value = state.queuedFiles.value.filter(f => f.__key !== file.__key);
emit('removed', [ file ]);
}
function revokeImgURLs () {
state.files.value.forEach(f => {
f.__img !== void 0 && window.URL.revokeObjectURL(f.__img.src);
});
}
function getFileInput () {
return inputRef.value
|| rootRef.value.getElementsByClassName('q-uploader__input')[ 0 ]
}
function addFilesToQueue (e, fileList) {
const localFiles = processFiles(e, fileList, state.files.value, true);
const fileInput = getFileInput();
if (fileInput !== void 0 && fileInput !== null) {
fileInput.value = '';
}
if (localFiles === void 0) { return }
localFiles.forEach(file => {
state.updateFileStatus(file, 'idle');
uploadSize.value += file.size;
if (props.noThumbnails !== true && file.type.toUpperCase().startsWith('IMAGE')) {
const img = new Image();
img.src = window.URL.createObjectURL(file);
file.__img = img;
}
});
state.files.value = state.files.value.concat(localFiles);
state.queuedFiles.value = state.queuedFiles.value.concat(localFiles);
emit('added', localFiles);
props.autoUpload === true && state.upload();
}
function upload () {
canUpload.value === true && state.upload();
}
function getBtn (show, icon, fn) {
if (show === true) {
const data = {
type: 'a',
key: icon,
icon: $q.iconSet.uploader[ icon ],
flat: true,
dense: true
};
let child = void 0;
if (icon === 'add') {
data.onClick = pickFiles;
child = renderInput;
}
else {
data.onClick = fn;
}
return vue.h(QBtn, data, child)
}
}
function renderInput () {
return vue.h('input', {
ref: inputRef,
class: 'q-uploader__input overflow-hidden absolute-full',
tabindex: -1,
type: 'file',
title: '', // try to remove default tooltip
accept: props.accept,
multiple: props.multiple === true ? 'multiple' : void 0,
capture: props.capture,
onMousedown: stop, // need to stop refocus from QBtn
onClick: pickFiles,
onChange: addFilesToQueue
})
}
function getHeader () {
if (slots.header !== void 0) {
return slots.header(publicApi)
}
return [
vue.h('div', {
class: 'q-uploader__header-content column'
}, [
vue.h('div', {
class: 'flex flex-center no-wrap q-gutter-xs'
}, [
getBtn(state.queuedFiles.value.length !== 0, 'removeQueue', removeQueuedFiles),
getBtn(state.uploadedFiles.value.length !== 0, 'removeUploaded', removeUploadedFiles),
state.isUploading.value === true
? vue.h(QSpinner, { class: 'q-uploader__spinner' })
: null,
vue.h('div', { class: 'col column justify-center' }, [
props.label !== void 0
? vue.h('div', { class: 'q-uploader__title' }, [ props.label ])
: null,
vue.h('div', { class: 'q-uploader__subtitle' }, [
uploadSizeLabel.value + ' / ' + uploadProgressLabel.value
])
]),
getBtn(canAddFiles.value, 'add'),
getBtn(props.hideUploadBtn === false && canUpload.value === true, 'upload', state.upload),
getBtn(state.isUploading.value, 'clear', state.abort)
])
])
]
}
function getList () {
if (slots.list !== void 0) {
return slots.list(publicApi)
}
return state.files.value.map(file => vue.h('div', {
key: file.__key,
class: 'q-uploader__file relative-position'
+ (props.noThumbnails !== true && file.__img !== void 0 ? ' q-uploader__file--img' : '')
+ (
file.__status === 'failed'
? ' q-uploader__file--failed'
: (file.__status === 'uploaded' ? ' q-uploader__file--uploaded' : '')
),
style: props.noThumbnails !== true && file.__img !== void 0
? { backgroundImage: 'url("' + file.__img.src + '")' }
: null
}, [
vue.h('div', {
class: 'q-uploader__file-header row flex-center no-wrap'
}, [
file.__status === 'failed'
? vue.h(QIcon, {
class: 'q-uploader__file-status',
name: $q.iconSet.type.negative,
color: 'negative'
})
: null,
vue.h('div', { class: 'q-uploader__file-header-content col' }, [
vue.h('div', { class: 'q-uploader__title' }, [ file.name ]),
vue.h('div', {
class: 'q-uploader__subtitle row items-center no-wrap'
}, [
file.__sizeLabel + ' / ' + file.__progressLabel
])
]),
file.__status === 'uploading'
? vue.h(QCircularProgress, {
value: file.__progress,
min: 0,
max: 1,
indeterminate: file.__progress === 0
})
: vue.h(QBtn, {
round: true,
dense: true,
flat: true,
icon: $q.iconSet.uploader[ file.__status === 'uploaded' ? 'done' : 'clear' ],
onClick: () => { removeFile(file); }
})
])
]))
}
vue.onBeforeUnmount(() => {
state.isUploading.value === true && state.abort();
state.files.value.length !== 0 && revokeImgURLs();
});
const publicApi = {};
for (const key in state) {
if (vue.isRef(state[ key ]) === true) {
injectProp(publicApi, key, () => state[ key ].value);
}
else { // method or non-computed prop
publicApi[ key ] = state[ key ];
}
}
Object.assign(publicApi, {
upload,
reset,
removeUploadedFiles,
removeQueuedFiles,
removeFile,
pickFiles,
addFiles
});
injectMultipleProps(publicApi, {
canAddFiles: () => canAddFiles.value,
canUpload: () => canUpload.value,
uploadSizeLabel: () => uploadSizeLabel.value,
uploadProgressLabel: () => uploadProgressLabel.value
});
// expose public api (methods & computed props)
expose({
...state,
upload,
reset,
removeUploadedFiles,
removeQueuedFiles,
removeFile,
pickFiles,
addFiles,
canAddFiles,
canUpload,
uploadSizeLabel,
uploadProgressLabel
});
return () => {
const children = [
vue.h('div', { class: colorClass.value }, getHeader()),
vue.h('div', { class: 'q-uploader__list scroll' }, getList()),
getDndNode('uploader')
];
state.isBusy.value === true && children.push(
vue.h('div', {
class: 'q-uploader__overlay absolute-full flex flex-center'
}, [ vue.h(QSpinner) ])
);
const data = { ref: rootRef, class: classes.value };
if (canAddFiles.value === true) {
Object.assign(data, { onDragover, onDragleave });
}
return vue.h('div', data, children)
}
}
const trueFn = () => true;
function getEmitsObject (emitsArray) {
const emitsObject = {};
emitsArray.forEach(val => {
emitsObject[ val ] = trueFn;
});
return emitsObject
}
const coreEmitsObject = getEmitsObject(coreEmits);
var createUploaderComponent = ({ name, props, emits, injectPlugin }) => createComponent({
name,
props: {
...coreProps,
...props
},
emits: isObject(emits) === true
? { ...coreEmitsObject, ...emits }
: [ ...coreEmits, ...emits ],
setup (_, { expose }) {
return getRenderer(injectPlugin, expose)
}
});
function getFn (prop) {
return typeof prop === 'function'
? prop
: () => prop
}
const props$2 = {
url: [ Function, String ],
method: {
type: [ Function, String ],
default: 'POST'
},
fieldName: {
type: [ Function, String ],
default: () => {
return file => file.name
}
},
headers: [ Function, Array ],
formFields: [ Function, Array ],
withCredentials: [ Function, Boolean ],
sendRaw: [ Function, Boolean ],
batch: [ Function, Boolean ],
factory: Function
};
const emits$1 = [ 'factoryFailed', 'uploaded', 'failed', 'uploading' ];
function injectPlugin ({ props, emit, helpers }) {
const xhrs = vue.ref([]);
const promises = vue.ref([]);
const workingThreads = vue.ref(0);
const xhrProps = vue.computed(() => ({
url: getFn(props.url),
method: getFn(props.method),
headers: getFn(props.headers),
formFields: getFn(props.formFields),
fieldName: getFn(props.fieldName),
withCredentials: getFn(props.withCredentials),
sendRaw: getFn(props.sendRaw),
batch: getFn(props.batch)
}));
const isUploading = vue.computed(() => workingThreads.value > 0);
const isBusy = vue.computed(() => promises.value.length !== 0);
let abortPromises;
function abort () {
xhrs.value.forEach(x => { x.abort(); });
if (promises.value.length !== 0) {
abortPromises = true;
}
}
function upload () {
const queue = helpers.queuedFiles.value.slice(0);
helpers.queuedFiles.value = [];
if (xhrProps.value.batch(queue)) {
runFactory(queue);
}
else {
queue.forEach(file => {
runFactory([ file ]);
});
}
}
function runFactory (files) {
workingThreads.value++;
if (typeof props.factory !== 'function') {
performUpload(files, {});
return
}
const res = props.factory(files);
if (!res) {
emit(
'factoryFailed',
new Error('QUploader: factory() does not return properly'),
files
);
workingThreads.value--;
}
else if (typeof res.catch === 'function' && typeof res.then === 'function') {
promises.value.push(res);
const failed = err => {
if (helpers.isAlive() === true) {
promises.value = promises.value.filter(p => p !== res);
if (promises.value.length === 0) {
abortPromises = false;
}
helpers.queuedFiles.value = helpers.queuedFiles.value.concat(files);
files.forEach(f => { helpers.updateFileStatus(f, 'failed'); });
emit('factoryFailed', err, files);
workingThreads.value--;
}
};
res.then(factory => {
if (abortPromises === true) {
failed(new Error('Aborted'));
}
else if (helpers.isAlive() === true) {
promises.value = promises.value.filter(p => p !== res);
performUpload(files, factory);
}
}).catch(failed);
}
else {
performUpload(files, res || {});
}
}
function performUpload (files, factory) {
const
form = new FormData(),
xhr = new XMLHttpRequest();
const getProp = (name, arg) => {
return factory[ name ] !== void 0
? getFn(factory[ name ])(arg)
: xhrProps.value[ name ](arg)
};
const url = getProp('url', files);
if (!url) {
console.error('q-uploader: invalid or no URL specified');
workingThreads.value--;
return
}
const fields = getProp('formFields', files);
fields !== void 0 && fields.forEach(field => {
form.append(field.name, field.value);
});
let
uploadIndex = 0,
uploadIndexSize = 0,
localUploadedSize = 0,
maxUploadSize = 0,
aborted;
xhr.upload.addEventListener('progress', e => {
if (aborted === true) { return }
const loaded = Math.min(maxUploadSize, e.loaded);
helpers.uploadedSize.value += loaded - localUploadedSize;
localUploadedSize = loaded;
let size = localUploadedSize - uploadIndexSize;
for (let i = uploadIndex; size > 0 && i < files.length; i++) {
const
file = files[ i ],
uploaded = size > file.size;
if (uploaded) {
size -= file.size;
uploadIndex++;
uploadIndexSize += file.size;
helpers.updateFileStatus(file, 'uploading', file.size);
}
else {
helpers.updateFileStatus(file, 'uploading', size);
return
}
}
}, false);
xhr.onreadystatechange = () => {
if (xhr.readyState < 4) {
return
}
if (xhr.status && xhr.status < 400) {
helpers.uploadedFiles.value = helpers.uploadedFiles.value.concat(files);
files.forEach(f => { helpers.updateFileStatus(f, 'uploaded'); });
emit('uploaded', { files, xhr });
}
else {
aborted = true;
helpers.uploadedSize.value -= localUploadedSize;
helpers.queuedFiles.value = helpers.queuedFiles.value.concat(files);
files.forEach(f => { helpers.updateFileStatus(f, 'failed'); });
emit('failed', { files, xhr });
}
workingThreads.value--;
xhrs.value = xhrs.value.filter(x => x !== xhr);
};
xhr.open(
getProp('method', files),
url
);
if (getProp('withCredentials', files) === true) {
xhr.withCredentials = true;
}
const headers = getProp('headers', files);
headers !== void 0 && headers.forEach(head => {
xhr.setRequestHeader(head.name, head.value);
});
const sendRaw = getProp('sendRaw', files);
files.forEach(file => {
helpers.updateFileStatus(file, 'uploading', 0);
if (sendRaw !== true) {
form.append(getProp('fieldName', file), file, file.name);
}
file.xhr = xhr;
file.__abort = () => { xhr.abort(); };
maxUploadSize += file.size;
});
emit('uploading', { files, xhr });
xhrs.value.push(xhr);
if (sendRaw === true) {
xhr.send(new Blob(files));
}
else {
xhr.send(form);
}
}
return {
isUploading,
isBusy,
abort,
upload
}
}
var xhrUploaderPlugin = {
name: 'QUploader',
props: props$2,
emits: emits$1,
injectPlugin
};
var QUploader = createUploaderComponent(xhrUploaderPlugin);
var QUploaderAddTrigger = createComponent({
name: 'QUploaderAddTrigger',
setup () {
const $trigger = vue.inject(uploaderKey, emptyRenderFn);
if ($trigger === emptyRenderFn) {
console.error('QUploaderAddTrigger needs to be child of QUploader');
}
return $trigger
}
});
var QVideo = createComponent({
name: 'QVideo',
props: {
...useRatioProps,
src: {
type: String,
required: true
},
title: String,
fetchpriority: {
type: String,
default: 'auto'
},
loading: {
type: String,
default: 'eager'
},
referrerpolicy: {
type: String,
default: 'strict-origin-when-cross-origin'
}
},
setup (props) {
const ratioStyle = useRatio(props);
const classes = vue.computed(() =>
'q-video'
+ (props.ratio !== void 0 ? ' q-video--responsive' : '')
);
return () => vue.h('div', {
class: classes.value,
style: ratioStyle.value
}, [
vue.h('iframe', {
src: props.src,
title: props.title,
fetchpriority: props.fetchpriority,
loading: props.loading,
referrerpolicy: props.referrerpolicy,
frameborder: '0',
allowfullscreen: true
})
])
}
});
var components = /*#__PURE__*/Object.freeze({
__proto__: null,
QAjaxBar: QAjaxBar,
QAvatar: QAvatar,
QBadge: QBadge,
QBanner: QBanner,
QBar: QBar,
QBreadcrumbs: QBreadcrumbs,
QBreadcrumbsEl: QBreadcrumbsEl,
QBtn: QBtn,
QBtnDropdown: QBtnDropdown,
QBtnGroup: QBtnGroup,
QBtnToggle: QBtnToggle,
QCard: QCard,
QCardSection: QCardSection,
QCardActions: QCardActions,
QCarousel: QCarousel,
QCarouselSlide: QCarouselSlide,
QCarouselControl: QCarouselControl,
QChatMessage: QChatMessage,
QCheckbox: QCheckbox,
QChip: QChip,
QCircularProgress: QCircularProgress,
QColor: QColor,
QDate: QDate,
QDialog: QDialog,
QDrawer: QDrawer,
QEditor: QEditor,
QExpansionItem: QExpansionItem,
QFab: QFab,
QFabAction: QFabAction,
QField: QField,
QFile: QFile,
QFooter: QFooter,
QForm: QForm,
QFormChildMixin: QFormChildMixin,
QHeader: QHeader,
QIcon: QIcon,
QImg: QImg,
QInfiniteScroll: QInfiniteScroll,
QInnerLoading: QInnerLoading,
QInput: QInput,
QIntersection: QIntersection,
QList: QList,
QItem: QItem,
QItemSection: QItemSection,
QItemLabel: QItemLabel,
QKnob: QKnob,
QLayout: QLayout,
QMarkupTable: QMarkupTable,
QMenu: QMenu,
QNoSsr: QNoSsr,
QOptionGroup: QOptionGroup,
QPage: QPage,
QPageContainer: QPageContainer,
QPageScroller: QPageScroller,
QPageSticky: QPageSticky,
QPagination: QPagination,
QParallax: QParallax,
QPopupEdit: QPopupEdit,
QPopupProxy: QPopupProxy,
QLinearProgress: QLinearProgress,
QPullToRefresh: QPullToRefresh,
QRadio: QRadio,
QRange: QRange,
QRating: QRating,
QResizeObserver: QResizeObserver,
QResponsive: QResponsive,
QScrollArea: QScrollArea,
QScrollObserver: QScrollObserver,
QSelect: QSelect,
QSeparator: QSeparator,
QSkeleton: QSkeleton,
QSlideItem: QSlideItem,
QSlideTransition: QSlideTransition,
QSlider: QSlider,
QSpace: QSpace,
QSpinner: QSpinner,
QSpinnerAudio: QSpinnerAudio,
QSpinnerBall: QSpinnerBall,
QSpinnerBars: QSpinnerBars,
QSpinnerBox: QSpinnerBox,
QSpinnerClock: QSpinnerClock,
QSpinnerComment: QSpinnerComment,
QSpinnerCube: QSpinnerCube,
QSpinnerDots: QSpinnerDots,
QSpinnerFacebook: QSpinnerFacebook,
QSpinnerGears: QSpinnerGears,
QSpinnerGrid: QSpinnerGrid,
QSpinnerHearts: QSpinnerHearts,
QSpinnerHourglass: QSpinnerHourglass,
QSpinnerInfinity: QSpinnerInfinity,
QSpinnerIos: QSpinnerIos,
QSpinnerOrbit: QSpinnerOrbit,
QSpinnerOval: QSpinnerOval,
QSpinnerPie: QSpinnerPie,
QSpinnerPuff: QSpinnerPuff,
QSpinnerRadio: QSpinnerRadio,
QSpinnerRings: QSpinnerRings,
QSpinnerTail: QSpinnerTail,
QSplitter: QSplitter,
QStep: QStep,
QStepper: QStepper,
QStepperNavigation: QStepperNavigation,
QTabPanels: QTabPanels,
QTabPanel: QTabPanel,
QTable: QTable,
QTh: QTh,
QTr: QTr,
QTd: QTd,
QTabs: QTabs,
QTab: QTab,
QRouteTab: QRouteTab,
QTime: QTime,
QTimeline: QTimeline,
QTimelineEntry: QTimelineEntry,
QToggle: QToggle,
QToolbar: QToolbar,
QToolbarTitle: QToolbarTitle,
QTooltip: QTooltip,
QTree: QTree,
QUploader: QUploader,
QUploaderAddTrigger: QUploaderAddTrigger,
QVideo: QVideo,
QVirtualScroll: QVirtualScroll
});
/*
* depth
* < 0 --> close all chain
* 0 --> disabled
* > 0 --> close chain up to N parent
*/
function getDepth (value) {
if (value === false) {
return 0
}
if (value === true || value === void 0) {
return 1
}
const depth = parseInt(value, 10);
return isNaN(depth) ? 0 : depth
}
var ClosePopup = createDirective({
name: 'close-popup',
beforeMount (el, { value }) {
const ctx = {
depth: getDepth(value),
handler (evt) {
// allow @click to be emitted
ctx.depth !== 0 && setTimeout(() => {
const proxy = getPortalProxy(el);
if (proxy !== void 0) {
closePortals(proxy, evt, ctx.depth);
}
});
},
handlerKey (evt) {
isKeyCode(evt, 13) === true && ctx.handler(evt);
}
};
el.__qclosepopup = ctx;
el.addEventListener('click', ctx.handler);
el.addEventListener('keyup', ctx.handlerKey);
},
updated (el, { value, oldValue }) {
if (value !== oldValue) {
el.__qclosepopup.depth = getDepth(value);
}
},
beforeUnmount (el) {
const ctx = el.__qclosepopup;
el.removeEventListener('click', ctx.handler);
el.removeEventListener('keyup', ctx.handlerKey);
delete el.__qclosepopup;
}
}
);
let id = 0;
let offsetBase = void 0;
function getAbsolutePosition (el, resize) {
if (offsetBase === void 0) {
offsetBase = document.createElement('div');
offsetBase.style.cssText = 'position: absolute; left: 0; top: 0';
document.body.appendChild(offsetBase);
}
const boundingRect = el.getBoundingClientRect();
const baseRect = offsetBase.getBoundingClientRect();
const { marginLeft, marginRight, marginTop, marginBottom } = window.getComputedStyle(el);
const marginH = parseInt(marginLeft, 10) + parseInt(marginRight, 10);
const marginV = parseInt(marginTop, 10) + parseInt(marginBottom, 10);
return {
left: boundingRect.left - baseRect.left,
top: boundingRect.top - baseRect.top,
width: boundingRect.right - boundingRect.left,
height: boundingRect.bottom - boundingRect.top,
widthM: boundingRect.right - boundingRect.left + (resize === true ? 0 : marginH),
heightM: boundingRect.bottom - boundingRect.top + (resize === true ? 0 : marginV),
marginH: resize === true ? marginH : 0,
marginV: resize === true ? marginV : 0
}
}
function getAbsoluteSize (el) {
return {
width: el.scrollWidth,
height: el.scrollHeight
}
}
// firefox rulez
const styleEdges = [ 'Top', 'Right', 'Bottom', 'Left' ];
const styleBorderRadiuses = [ 'borderTopLeftRadius', 'borderTopRightRadius', 'borderBottomRightRadius', 'borderBottomLeftRadius' ];
const reStyleSkipKey = /-block|-inline|block-|inline-/;
const reStyleSkipRule = /(-block|-inline|block-|inline-).*:/;
function getComputedStyle$1 (el, props) {
const style = window.getComputedStyle(el);
const fixed = {};
for (let i = 0; i < props.length; i++) {
const prop = props[ i ];
if (style[ prop ] === '') {
if (prop === 'cssText') {
const styleLen = style.length;
let val = '';
for (let i = 0; i < styleLen; i++) {
if (reStyleSkipKey.test(style[ i ]) !== true) {
val += style[ i ] + ': ' + style[ style[ i ] ] + '; ';
}
}
fixed[ prop ] = val;
}
else if ([ 'borderWidth', 'borderStyle', 'borderColor' ].indexOf(prop) > -1) {
const suffix = prop.replace('border', '');
let val = '';
for (let j = 0; j < styleEdges.length; j++) {
const subProp = 'border' + styleEdges[ j ] + suffix;
val += style[ subProp ] + ' ';
}
fixed[ prop ] = val;
}
else if (prop === 'borderRadius') {
let val1 = '';
let val2 = '';
for (let j = 0; j < styleBorderRadiuses.length; j++) {
const val = style[ styleBorderRadiuses[ j ] ].split(' ');
val1 += val[ 0 ] + ' ';
val2 += (val[ 1 ] === void 0 ? val[ 0 ] : val[ 1 ]) + ' ';
}
fixed[ prop ] = val1 + '/ ' + val2;
}
else {
fixed[ prop ] = style[ prop ];
}
}
else {
if (prop === 'cssText') {
fixed[ prop ] = style[ prop ]
.split(';')
.filter(val => reStyleSkipRule.test(val) !== true)
.join(';');
}
else {
fixed[ prop ] = style[ prop ];
}
}
}
return fixed
}
const zIndexPositions = [ 'absolute', 'fixed', 'relative', 'sticky' ];
function getMaxZIndex (elStart) {
let el = elStart;
let maxIndex = 0;
while (el !== null && el !== document) {
const { position, zIndex } = window.getComputedStyle(el);
const zIndexNum = Number(zIndex);
if (
zIndexNum > maxIndex
&& (el === elStart || zIndexPositions.includes(position) === true)
) {
maxIndex = zIndexNum;
}
el = el.parentNode;
}
return maxIndex
}
function normalizeElements (opts) {
return {
from: opts.from,
to: opts.to !== void 0
? opts.to
: opts.from
}
}
function normalizeOptions (options) {
if (typeof options === 'number') {
options = {
duration: options
};
}
else if (typeof options === 'function') {
options = {
onEnd: options
};
}
return {
...options,
waitFor: options.waitFor === void 0 ? 0 : options.waitFor,
duration: isNaN(options.duration) === true ? 300 : parseInt(options.duration, 10),
easing: typeof options.easing === 'string' && options.easing.length !== 0 ? options.easing : 'ease-in-out',
delay: isNaN(options.delay) === true ? 0 : parseInt(options.delay, 10),
fill: typeof options.fill === 'string' && options.fill.length !== 0 ? options.fill : 'none',
resize: options.resize === true,
// account for UMD too where modifiers will be lowercased to work
useCSS: options.useCSS === true || options.usecss === true,
// account for UMD too where modifiers will be lowercased to work
hideFromClone: options.hideFromClone === true || options.hidefromclone === true,
// account for UMD too where modifiers will be lowercased to work
keepToClone: options.keepToClone === true || options.keeptoclone === true,
tween: options.tween === true,
tweenFromOpacity: isNaN(options.tweenFromOpacity) === true ? 0.6 : parseFloat(options.tweenFromOpacity),
tweenToOpacity: isNaN(options.tweenToOpacity) === true ? 0.5 : parseFloat(options.tweenToOpacity)
}
}
function getElement (element) {
const type = typeof element;
return type === 'function'
? element()
: (
type === 'string'
? document.querySelector(element)
: element
)
}
function isValidElement (element) {
return element
&& element.ownerDocument === document
&& element.parentNode !== null
}
function morph (_options) {
let cancel = () => false;
let cancelStatus = false;
let endElementTo = true;
const elements = normalizeElements(_options);
const options = normalizeOptions(_options);
const elFrom = getElement(elements.from);
if (isValidElement(elFrom) !== true) {
// we return a cancel function that return false, meaning the cancel function failed
return cancel
}
// we clean other morphs running on this element
typeof elFrom.qMorphCancel === 'function' && elFrom.qMorphCancel();
let animationFromClone = void 0;
let animationFromTween = void 0;
let animationToClone = void 0;
let animationTo = void 0;
const elFromParent = elFrom.parentNode;
const elFromNext = elFrom.nextElementSibling;
// we get the dimensions and characteristics
// of the parent of the initial element before changes
const elFromPosition = getAbsolutePosition(elFrom, options.resize);
const {
width: elFromParentWidthBefore,
height: elFromParentHeightBefore
} = getAbsoluteSize(elFromParent);
const {
borderWidth: elFromBorderWidth,
borderStyle: elFromBorderStyle,
borderColor: elFromBorderColor,
borderRadius: elFromBorderRadius,
backgroundColor: elFromBackground,
transform: elFromTransform,
position: elFromPositioningType,
cssText: elFromCssText
} = getComputedStyle$1(elFrom, [ 'borderWidth', 'borderStyle', 'borderColor', 'borderRadius', 'backgroundColor', 'transform', 'position', 'cssText' ]);
const elFromClassSaved = elFrom.classList.toString();
const elFromStyleSaved = elFrom.style.cssText;
// we make a clone of the initial element and
// use it to display until the final element is ready
// and to change the occupied space during animation
const elFromClone = elFrom.cloneNode(true);
const elFromTween = options.tween === true ? elFrom.cloneNode(true) : void 0;
if (elFromTween !== void 0) {
elFromTween.className = elFromTween.classList.toString().split(' ').filter(c => /^bg-/.test(c) === false).join(' ');
}
// if the initial element is not going to be removed do not show the placeholder
options.hideFromClone === true && elFromClone.classList.add('q-morph--internal');
// prevent interaction with placeholder
elFromClone.setAttribute('aria-hidden', 'true');
elFromClone.style.transition = 'none';
elFromClone.style.animation = 'none';
elFromClone.style.pointerEvents = 'none';
elFromParent.insertBefore(elFromClone, elFromNext);
// we mark the element with its cleanup function
elFrom.qMorphCancel = () => {
cancelStatus = true;
// we clean the clone of the initial element
elFromClone.remove();
elFromTween !== void 0 && elFromTween.remove();
options.hideFromClone === true && elFromClone.classList.remove('q-morph--internal');
// we remove the cleanup function from the element
elFrom.qMorphCancel = void 0;
};
// will be called after Vue catches up with the changes done by _options.onToggle() function
const calculateFinalState = () => {
const elTo = getElement(elements.to);
if (cancelStatus === true || isValidElement(elTo) !== true) {
typeof elFrom.qMorphCancel === 'function' && elFrom.qMorphCancel();
return
}
// we clean other morphs running on this element
elFrom !== elTo && typeof elTo.qMorphCancel === 'function' && elTo.qMorphCancel();
// we hide the final element and the clone of the initial element
// we don't hide the final element if we want both it and the animated one visible
options.keepToClone !== true && elTo.classList.add('q-morph--internal');
elFromClone.classList.add('q-morph--internal');
// we get the dimensions of the parent of the initial element after changes
// the difference is how much we should animate the clone
const {
width: elFromParentWidthAfter,
height: elFromParentHeightAfter
} = getAbsoluteSize(elFromParent);
// we get the dimensions of the parent of the final element before changes
const {
width: elToParentWidthBefore,
height: elToParentHeightBefore
} = getAbsoluteSize(elTo.parentNode);
// then we show the clone of the initial element if we don't want it hidden
options.hideFromClone !== true && elFromClone.classList.remove('q-morph--internal');
// we mark the element with its cleanup function
elTo.qMorphCancel = () => {
cancelStatus = true;
// we clean the clone of the initial element
elFromClone.remove();
elFromTween !== void 0 && elFromTween.remove();
options.hideFromClone === true && elFromClone.classList.remove('q-morph--internal');
// we show the final element
options.keepToClone !== true && elTo.classList.remove('q-morph--internal');
// we remove the cleanup function from the elements
elFrom.qMorphCancel = void 0;
elTo.qMorphCancel = void 0;
};
// will be called after waitFor (give time to render the final element)
const animate = () => {
if (cancelStatus === true) {
typeof elTo.qMorphCancel === 'function' && elTo.qMorphCancel();
return
}
// now the animation starts, so we only need the clone
// of the initial element as a spacer
// we also hide it to calculate the dimensions of the
// parent of the final element after the changes
if (options.hideFromClone !== true) {
elFromClone.classList.add('q-morph--internal');
elFromClone.innerHTML = '';
elFromClone.style.left = 0;
elFromClone.style.right = 'unset';
elFromClone.style.top = 0;
elFromClone.style.bottom = 'unset';
elFromClone.style.transform = 'none';
}
// we show the final element
if (options.keepToClone !== true) {
elTo.classList.remove('q-morph--internal');
}
// we get the dimensions of the parent of the final element after changes
// the difference is how much we should animate the clone
const elToParent = elTo.parentNode;
const {
width: elToParentWidthAfter,
height: elToParentHeightAfter
} = getAbsoluteSize(elToParent);
const elToClone = elTo.cloneNode(options.keepToClone);
elToClone.setAttribute('aria-hidden', 'true');
if (options.keepToClone !== true) {
elToClone.style.left = 0;
elToClone.style.right = 'unset';
elToClone.style.top = 0;
elToClone.style.bottom = 'unset';
elToClone.style.transform = 'none';
elToClone.style.pointerEvents = 'none';
}
elToClone.classList.add('q-morph--internal');
// if elFrom is the same as elTo the next element is elFromClone
const elToNext = elTo === elFrom && elFromParent === elToParent ? elFromClone : elTo.nextElementSibling;
elToParent.insertBefore(elToClone, elToNext);
const {
borderWidth: elToBorderWidth,
borderStyle: elToBorderStyle,
borderColor: elToBorderColor,
borderRadius: elToBorderRadius,
backgroundColor: elToBackground,
transform: elToTransform,
position: elToPositioningType,
cssText: elToCssText
} = getComputedStyle$1(elTo, [ 'borderWidth', 'borderStyle', 'borderColor', 'borderRadius', 'backgroundColor', 'transform', 'position', 'cssText' ]);
const elToClassSaved = elTo.classList.toString();
const elToStyleSaved = elTo.style.cssText;
// we set the computed styles on the element (to be able to remove classes)
elTo.style.cssText = elToCssText;
elTo.style.transform = 'none';
elTo.style.animation = 'none';
elTo.style.transition = 'none';
// we strip the background classes (background color can no longer be animated if !important is used)
elTo.className = elToClassSaved.split(' ').filter(c => /^bg-/.test(c) === false).join(' ');
const elToPosition = getAbsolutePosition(elTo, options.resize);
const deltaX = elFromPosition.left - elToPosition.left;
const deltaY = elFromPosition.top - elToPosition.top;
const scaleX = elFromPosition.width / (elToPosition.width > 0 ? elToPosition.width : 10);
const scaleY = elFromPosition.height / (elToPosition.height > 0 ? elToPosition.height : 100);
const elFromParentWidthDiff = elFromParentWidthBefore - elFromParentWidthAfter;
const elFromParentHeightDiff = elFromParentHeightBefore - elFromParentHeightAfter;
const elToParentWidthDiff = elToParentWidthAfter - elToParentWidthBefore;
const elToParentHeightDiff = elToParentHeightAfter - elToParentHeightBefore;
const elFromCloneWidth = Math.max(elFromPosition.widthM, elFromParentWidthDiff);
const elFromCloneHeight = Math.max(elFromPosition.heightM, elFromParentHeightDiff);
const elToCloneWidth = Math.max(elToPosition.widthM, elToParentWidthDiff);
const elToCloneHeight = Math.max(elToPosition.heightM, elToParentHeightDiff);
const elSharedSize = elFrom === elTo
&& [ 'absolute', 'fixed' ].includes(elToPositioningType) === false
&& [ 'absolute', 'fixed' ].includes(elFromPositioningType) === false;
// if the final element has fixed position or if a parent
// has fixed position we need to animate it as fixed
let elToNeedsFixedPosition = elToPositioningType === 'fixed';
let parent = elToParent;
while (elToNeedsFixedPosition !== true && parent !== document) {
elToNeedsFixedPosition = window.getComputedStyle(parent).position === 'fixed';
parent = parent.parentNode;
}
// we show the spacer for the initial element
if (options.hideFromClone !== true) {
elFromClone.style.display = 'block';
elFromClone.style.flex = '0 0 auto';
elFromClone.style.opacity = 0;
elFromClone.style.minWidth = 'unset';
elFromClone.style.maxWidth = 'unset';
elFromClone.style.minHeight = 'unset';
elFromClone.style.maxHeight = 'unset';
elFromClone.classList.remove('q-morph--internal');
}
// we show the spacer for the final element
if (options.keepToClone !== true) {
elToClone.style.display = 'block';
elToClone.style.flex = '0 0 auto';
elToClone.style.opacity = 0;
elToClone.style.minWidth = 'unset';
elToClone.style.maxWidth = 'unset';
elToClone.style.minHeight = 'unset';
elToClone.style.maxHeight = 'unset';
}
elToClone.classList.remove('q-morph--internal');
// we apply classes specified by user
if (typeof options.classes === 'string') {
elTo.className += ' ' + options.classes;
}
// we apply styles specified by user
if (typeof options.style === 'string') {
elTo.style.cssText += ' ' + options.style;
}
else if (isObject(options.style) === true) {
for (const prop in options.style) {
elTo.style[ prop ] = options.style[ prop ];
}
}
const elFromZIndex = getMaxZIndex(elFromClone);
const elToZIndex = getMaxZIndex(elTo);
// we position the morphing element
// if we use fixed position for the final element we need to adjust for scroll
const documentScroll = elToNeedsFixedPosition === true
? document.documentElement
: { scrollLeft: 0, scrollTop: 0 };
elTo.style.position = elToNeedsFixedPosition === true ? 'fixed' : 'absolute';
elTo.style.left = `${ elToPosition.left - documentScroll.scrollLeft }px`;
elTo.style.right = 'unset';
elTo.style.top = `${ elToPosition.top - documentScroll.scrollTop }px`;
elTo.style.margin = 0;
if (options.resize === true) {
elTo.style.minWidth = 'unset';
elTo.style.maxWidth = 'unset';
elTo.style.minHeight = 'unset';
elTo.style.maxHeight = 'unset';
elTo.style.overflow = 'hidden';
elTo.style.overflowX = 'hidden';
elTo.style.overflowY = 'hidden';
}
document.body.appendChild(elTo);
if (elFromTween !== void 0) {
elFromTween.style.cssText = elFromCssText;
elFromTween.style.transform = 'none';
elFromTween.style.animation = 'none';
elFromTween.style.transition = 'none';
elFromTween.style.position = elTo.style.position;
elFromTween.style.left = `${ elFromPosition.left - documentScroll.scrollLeft }px`;
elFromTween.style.right = 'unset';
elFromTween.style.top = `${ elFromPosition.top - documentScroll.scrollTop }px`;
elFromTween.style.margin = 0;
elFromTween.style.pointerEvents = 'none';
if (options.resize === true) {
elFromTween.style.minWidth = 'unset';
elFromTween.style.maxWidth = 'unset';
elFromTween.style.minHeight = 'unset';
elFromTween.style.maxHeight = 'unset';
elFromTween.style.overflow = 'hidden';
elFromTween.style.overflowX = 'hidden';
elFromTween.style.overflowY = 'hidden';
}
document.body.appendChild(elFromTween);
}
const commonCleanup = aborted => {
// we put the element back in it's place
// and restore the styles and classes
if (elFrom === elTo && endElementTo !== true) {
elTo.style.cssText = elFromStyleSaved;
elTo.className = elFromClassSaved;
}
else {
elTo.style.cssText = elToStyleSaved;
elTo.className = elToClassSaved;
}
elToClone.parentNode === elToParent && elToParent.insertBefore(elTo, elToClone);
// we clean the spacers
elFromClone.remove();
elToClone.remove();
elFromTween !== void 0 && elFromTween.remove();
// cancel will be no longer available
cancel = () => false;
elFrom.qMorphCancel = void 0;
elTo.qMorphCancel = void 0;
// we are ready
if (typeof options.onEnd === 'function') {
options.onEnd(endElementTo === true ? 'to' : 'from', aborted === true);
}
};
if (options.useCSS !== true && typeof elTo.animate === 'function') {
const resizeFrom = options.resize === true
? {
transform: `translate(${ deltaX }px, ${ deltaY }px)`,
width: `${ elFromCloneWidth }px`,
height: `${ elFromCloneHeight }px`
}
: {
transform: `translate(${ deltaX }px, ${ deltaY }px) scale(${ scaleX }, ${ scaleY })`
};
const resizeTo = options.resize === true
? {
width: `${ elToCloneWidth }px`,
height: `${ elToCloneHeight }px`
}
: {};
const resizeFromTween = options.resize === true
? {
width: `${ elFromCloneWidth }px`,
height: `${ elFromCloneHeight }px`
}
: {};
const resizeToTween = options.resize === true
? {
transform: `translate(${ -1 * deltaX }px, ${ -1 * deltaY }px)`,
width: `${ elToCloneWidth }px`,
height: `${ elToCloneHeight }px`
}
: {
transform: `translate(${ -1 * deltaX }px, ${ -1 * deltaY }px) scale(${ 1 / scaleX }, ${ 1 / scaleY })`
};
const tweenFrom = elFromTween !== void 0
? { opacity: options.tweenToOpacity }
: { backgroundColor: elFromBackground };
const tweenTo = elFromTween !== void 0
? { opacity: 1 }
: { backgroundColor: elToBackground };
animationTo = elTo.animate([
{
margin: 0,
borderWidth: elFromBorderWidth,
borderStyle: elFromBorderStyle,
borderColor: elFromBorderColor,
borderRadius: elFromBorderRadius,
zIndex: elFromZIndex,
transformOrigin: '0 0',
...resizeFrom,
...tweenFrom
},
{
margin: 0,
borderWidth: elToBorderWidth,
borderStyle: elToBorderStyle,
borderColor: elToBorderColor,
borderRadius: elToBorderRadius,
zIndex: elToZIndex,
transformOrigin: '0 0',
transform: elToTransform,
...resizeTo,
...tweenTo
}
], {
duration: options.duration,
easing: options.easing,
fill: options.fill,
delay: options.delay
});
animationFromTween = elFromTween === void 0 ? void 0 : elFromTween.animate([
{
opacity: options.tweenFromOpacity,
margin: 0,
borderWidth: elFromBorderWidth,
borderStyle: elFromBorderStyle,
borderColor: elFromBorderColor,
borderRadius: elFromBorderRadius,
zIndex: elFromZIndex,
transformOrigin: '0 0',
transform: elFromTransform,
...resizeFromTween
},
{
opacity: 0,
margin: 0,
borderWidth: elToBorderWidth,
borderStyle: elToBorderStyle,
borderColor: elToBorderColor,
borderRadius: elToBorderRadius,
zIndex: elToZIndex,
transformOrigin: '0 0',
...resizeToTween
}
], {
duration: options.duration,
easing: options.easing,
fill: options.fill,
delay: options.delay
});
animationFromClone = options.hideFromClone === true || elSharedSize === true ? void 0 : elFromClone.animate([
{
margin: `${ elFromParentHeightDiff < 0 ? elFromParentHeightDiff / 2 : 0 }px ${ elFromParentWidthDiff < 0 ? elFromParentWidthDiff / 2 : 0 }px`,
width: `${ elFromCloneWidth + elFromPosition.marginH }px`,
height: `${ elFromCloneHeight + elFromPosition.marginV }px`
},
{
margin: 0,
width: 0,
height: 0
}
], {
duration: options.duration,
easing: options.easing,
fill: options.fill,
delay: options.delay
});
animationToClone = options.keepToClone === true ? void 0 : elToClone.animate([
elSharedSize === true
? {
margin: `${ elFromParentHeightDiff < 0 ? elFromParentHeightDiff / 2 : 0 }px ${ elFromParentWidthDiff < 0 ? elFromParentWidthDiff / 2 : 0 }px`,
width: `${ elFromCloneWidth + elFromPosition.marginH }px`,
height: `${ elFromCloneHeight + elFromPosition.marginV }px`
}
: {
margin: 0,
width: 0,
height: 0
},
{
margin: `${ elToParentHeightDiff < 0 ? elToParentHeightDiff / 2 : 0 }px ${ elToParentWidthDiff < 0 ? elToParentWidthDiff / 2 : 0 }px`,
width: `${ elToCloneWidth + elToPosition.marginH }px`,
height: `${ elToCloneHeight + elToPosition.marginV }px`
}
], {
duration: options.duration,
easing: options.easing,
fill: options.fill,
delay: options.delay
});
const cleanup = abort => {
animationFromClone !== void 0 && animationFromClone.cancel();
animationFromTween !== void 0 && animationFromTween.cancel();
animationToClone !== void 0 && animationToClone.cancel();
animationTo.cancel();
animationTo.removeEventListener('finish', cleanup);
animationTo.removeEventListener('cancel', cleanup);
commonCleanup(abort);
// we clean the animations
animationFromClone = void 0;
animationFromTween = void 0;
animationToClone = void 0;
animationTo = void 0;
};
elFrom.qMorphCancel = () => {
elFrom.qMorphCancel = void 0;
cancelStatus = true;
cleanup();
};
elTo.qMorphCancel = () => {
elTo.qMorphCancel = void 0;
cancelStatus = true;
cleanup();
};
animationTo.addEventListener('finish', cleanup);
animationTo.addEventListener('cancel', cleanup);
cancel = abort => {
// we are not in a morph that we can cancel
if (cancelStatus === true || animationTo === void 0) {
return false
}
if (abort === true) {
cleanup(true);
return true
}
endElementTo = endElementTo !== true;
animationFromClone !== void 0 && animationFromClone.reverse();
animationFromTween !== void 0 && animationFromTween.reverse();
animationToClone !== void 0 && animationToClone.reverse();
animationTo.reverse();
return true
};
}
else {
const qAnimId = `q-morph-anim-${ ++id }`;
const style = document.createElement('style');
const resizeFrom = options.resize === true
? `
transform: translate(${ deltaX }px, ${ deltaY }px);
width: ${ elFromCloneWidth }px;
height: ${ elFromCloneHeight }px;
`
: `transform: translate(${ deltaX }px, ${ deltaY }px) scale(${ scaleX }, ${ scaleY });`;
const resizeTo = options.resize === true
? `
width: ${ elToCloneWidth }px;
height: ${ elToCloneHeight }px;
`
: '';
const resizeFromTween = options.resize === true
? `
width: ${ elFromCloneWidth }px;
height: ${ elFromCloneHeight }px;
`
: '';
const resizeToTween = options.resize === true
? `
transform: translate(${ -1 * deltaX }px, ${ -1 * deltaY }px);
width: ${ elToCloneWidth }px;
height: ${ elToCloneHeight }px;
`
: `transform: translate(${ -1 * deltaX }px, ${ -1 * deltaY }px) scale(${ 1 / scaleX }, ${ 1 / scaleY });`;
const tweenFrom = elFromTween !== void 0
? `opacity: ${ options.tweenToOpacity };`
: `background-color: ${ elFromBackground };`;
const tweenTo = elFromTween !== void 0
? 'opacity: 1;'
: `background-color: ${ elToBackground };`;
const keyframesFromTween = elFromTween === void 0
? ''
: `
@keyframes ${ qAnimId }-from-tween {
0% {
opacity: ${ options.tweenFromOpacity };
margin: 0;
border-width: ${ elFromBorderWidth };
border-style: ${ elFromBorderStyle };
border-color: ${ elFromBorderColor };
border-radius: ${ elFromBorderRadius };
z-index: ${ elFromZIndex };
transform-origin: 0 0;
transform: ${ elFromTransform };
${ resizeFromTween }
}
100% {
opacity: 0;
margin: 0;
border-width: ${ elToBorderWidth };
border-style: ${ elToBorderStyle };
border-color: ${ elToBorderColor };
border-radius: ${ elToBorderRadius };
z-index: ${ elToZIndex };
transform-origin: 0 0;
${ resizeToTween }
}
}
`;
const keyframesFrom = options.hideFromClone === true || elSharedSize === true
? ''
: `
@keyframes ${ qAnimId }-from {
0% {
margin: ${ elFromParentHeightDiff < 0 ? elFromParentHeightDiff / 2 : 0 }px ${ elFromParentWidthDiff < 0 ? elFromParentWidthDiff / 2 : 0 }px;
width: ${ elFromCloneWidth + elFromPosition.marginH }px;
height: ${ elFromCloneHeight + elFromPosition.marginV }px;
}
100% {
margin: 0;
width: 0;
height: 0;
}
}
`;
const keyframeToStart = elSharedSize === true
? `
margin: ${ elFromParentHeightDiff < 0 ? elFromParentHeightDiff / 2 : 0 }px ${ elFromParentWidthDiff < 0 ? elFromParentWidthDiff / 2 : 0 }px;
width: ${ elFromCloneWidth + elFromPosition.marginH }px;
height: ${ elFromCloneHeight + elFromPosition.marginV }px;
`
: `
margin: 0;
width: 0;
height: 0;
`;
const keyframesTo = options.keepToClone === true
? ''
: `
@keyframes ${ qAnimId }-to {
0% {
${ keyframeToStart }
}
100% {
margin: ${ elToParentHeightDiff < 0 ? elToParentHeightDiff / 2 : 0 }px ${ elToParentWidthDiff < 0 ? elToParentWidthDiff / 2 : 0 }px;
width: ${ elToCloneWidth + elToPosition.marginH }px;
height: ${ elToCloneHeight + elToPosition.marginV }px;
}
}
`;
style.innerHTML = `
@keyframes ${ qAnimId } {
0% {
margin: 0;
border-width: ${ elFromBorderWidth };
border-style: ${ elFromBorderStyle };
border-color: ${ elFromBorderColor };
border-radius: ${ elFromBorderRadius };
background-color: ${ elFromBackground };
z-index: ${ elFromZIndex };
transform-origin: 0 0;
${ resizeFrom }
${ tweenFrom }
}
100% {
margin: 0;
border-width: ${ elToBorderWidth };
border-style: ${ elToBorderStyle };
border-color: ${ elToBorderColor };
border-radius: ${ elToBorderRadius };
background-color: ${ elToBackground };
z-index: ${ elToZIndex };
transform-origin: 0 0;
transform: ${ elToTransform };
${ resizeTo }
${ tweenTo }
}
}
${ keyframesFrom }
${ keyframesFromTween }
${ keyframesTo }
`;
document.head.appendChild(style);
let animationDirection = 'normal';
elFromClone.style.animation = `${ options.duration }ms ${ options.easing } ${ options.delay }ms ${ animationDirection } ${ options.fill } ${ qAnimId }-from`;
if (elFromTween !== void 0) {
elFromTween.style.animation = `${ options.duration }ms ${ options.easing } ${ options.delay }ms ${ animationDirection } ${ options.fill } ${ qAnimId }-from-tween`;
}
elToClone.style.animation = `${ options.duration }ms ${ options.easing } ${ options.delay }ms ${ animationDirection } ${ options.fill } ${ qAnimId }-to`;
elTo.style.animation = `${ options.duration }ms ${ options.easing } ${ options.delay }ms ${ animationDirection } ${ options.fill } ${ qAnimId }`;
const cleanup = evt => {
if (evt === Object(evt) && evt.animationName !== qAnimId) {
return
}
elTo.removeEventListener('animationend', cleanup);
elTo.removeEventListener('animationcancel', cleanup);
commonCleanup();
// we clean the animations
style.remove();
};
elFrom.qMorphCancel = () => {
elFrom.qMorphCancel = void 0;
cancelStatus = true;
cleanup();
};
elTo.qMorphCancel = () => {
elTo.qMorphCancel = void 0;
cancelStatus = true;
cleanup();
};
elTo.addEventListener('animationend', cleanup);
elTo.addEventListener('animationcancel', cleanup);
cancel = abort => {
// we are not in a morph that we can cancel
if (cancelStatus === true || !elTo || !elFromClone || !elToClone) {
return false
}
if (abort === true) {
cleanup();
return true
}
endElementTo = endElementTo !== true;
animationDirection = animationDirection === 'normal' ? 'reverse' : 'normal';
elFromClone.style.animationDirection = animationDirection;
elFromTween.style.animationDirection = animationDirection;
elToClone.style.animationDirection = animationDirection;
elTo.style.animationDirection = animationDirection;
return true
};
}
};
if (
options.waitFor > 0
|| options.waitFor === 'transitionend'
|| (options.waitFor === Object(options.waitFor) && typeof options.waitFor.then === 'function')
) {
const delayPromise = options.waitFor > 0
? new Promise(resolve => setTimeout(resolve, options.waitFor))
: (
options.waitFor === 'transitionend'
? new Promise(resolve => {
const endFn = () => {
if (timer !== null) {
clearTimeout(timer);
timer = null;
}
if (elTo) {
elTo.removeEventListener('transitionend', endFn);
elTo.removeEventListener('transitioncancel', endFn);
}
resolve();
};
let timer = setTimeout(endFn, 400);
elTo.addEventListener('transitionend', endFn);
elTo.addEventListener('transitioncancel', endFn);
})
: options.waitFor
);
delayPromise
.then(animate)
.catch(() => {
typeof elTo.qMorphCancel === 'function' && elTo.qMorphCancel();
});
}
else {
animate();
}
};
typeof _options.onToggle === 'function' && _options.onToggle();
requestAnimationFrame(calculateFinalState);
// we return the cancel function
// returns:
// false if the cancel cannot be performed (the morph ended already or has not started)
// true else
return abort => cancel(abort)
}
const morphGroups = {};
const props$1 = [
'duration', 'delay', 'easing', 'fill',
'classes', 'style', 'duration', 'resize',
'useCSS', 'hideFromClone', 'keepToClone', 'tween',
'tweenFromOpacity', 'tweenToOpacity',
'waitFor', 'onEnd'
];
const mods = [
'resize', 'useCSS', 'hideFromClone', 'keepToClone', 'tween'
];
function changeClass (ctx, action) {
if (ctx.clsAction !== action) {
ctx.clsAction = action;
ctx.el.classList[ action ]('q-morph--invisible');
}
}
function trigger (group) {
if (group.animating === true || group.queue.length < 2) {
return
}
const [ from, to ] = group.queue;
group.animating = true;
from.animating = true;
to.animating = true;
changeClass(from, 'remove');
changeClass(to, 'remove');
const cancelFn = morph({
from: from.el,
to: to.el,
onToggle () {
changeClass(from, 'add');
changeClass(to, 'remove');
},
...to.opts,
onEnd (dir, aborted) {
to.opts.onEnd !== void 0 && to.opts.onEnd(dir, aborted);
if (aborted === true) {
return
}
from.animating = false;
to.animating = false;
group.animating = false;
group.cancel = void 0;
group.queue.shift();
trigger(group);
}
});
group.cancel = () => {
cancelFn(true); // abort
group.cancel = void 0;
};
}
function updateModifiers (mod, ctx) {
const opts = ctx.opts;
mods.forEach(name => {
opts[ name ] = mod[ name ] === true;
});
}
function insertArgs (arg, ctx) {
const opts = typeof arg === 'string' && arg.length !== 0
? arg.split(':') : [];
ctx.name = opts[ 0 ];
ctx.group = opts[ 1 ];
Object.assign(ctx.opts, {
duration: isNaN(opts[ 2 ]) === true
? 300
: parseFloat(opts[ 2 ]),
waitFor: opts[ 3 ]
});
}
function updateArgs (arg, ctx) {
if (arg.group !== void 0) {
ctx.group = arg.group;
}
if (arg.name !== void 0) {
ctx.name = arg.name;
}
const opts = ctx.opts;
props$1.forEach(name => {
if (arg[ name ] !== void 0) {
opts[ name ] = arg[ name ];
}
});
}
function updateModel (name, ctx) {
if (ctx.name === name) {
const group = morphGroups[ ctx.group ];
// if group is not registered
if (group === void 0) {
morphGroups[ ctx.group ] = {
name: ctx.group,
model: name,
queue: [ ctx ],
animating: false
};
changeClass(ctx, 'remove');
}
// if model changed
else if (group.model !== name) {
group.model = name;
group.queue.push(ctx);
if (group.animating === false && group.queue.length === 2) {
trigger(group);
}
}
return
}
if (ctx.animating === false) {
changeClass(ctx, 'add');
}
}
function updateValue (ctx, value) {
let model;
if (Object(value) === value) {
model = '' + value.model;
updateArgs(value, ctx);
updateModifiers(value, ctx);
}
else {
model = '' + value;
}
if (model !== ctx.model) {
ctx.model = model;
updateModel(model, ctx);
}
else if (ctx.animating === false && ctx.clsAction !== void 0) {
// ensure HMR
ctx.el.classList[ ctx.clsAction ]('q-morph--invisible');
}
}
var Morph = createDirective({
name: 'morph',
mounted (el, binding) {
const ctx = {
el,
animating: false,
opts: {}
};
updateModifiers(binding.modifiers, ctx);
insertArgs(binding.arg, ctx);
updateValue(ctx, binding.value);
el.__qmorph = ctx;
},
updated (el, binding) {
updateValue(el.__qmorph, binding.value);
},
beforeUnmount (el) {
const ctx = el.__qmorph;
const group = morphGroups[ ctx.group ];
if (group !== void 0) {
const index = group.queue.indexOf(ctx);
if (index !== -1) {
group.queue = group.queue.filter(item => item !== ctx);
if (group.queue.length === 0) {
group.cancel !== void 0 && group.cancel();
delete morphGroups[ ctx.group ];
}
}
}
if (ctx.clsAction === 'add') {
el.classList.remove('q-morph--invisible');
}
delete el.__qmorph;
}
}
);
const defaultCfg = {
childList: true,
subtree: true,
attributes: true,
characterData: true,
attributeOldValue: true,
characterDataOldValue: true
};
function update$2 (el, ctx, value) {
ctx.handler = value;
ctx.observer !== void 0 && ctx.observer.disconnect();
ctx.observer = new MutationObserver(list => {
if (typeof ctx.handler === 'function') {
const res = ctx.handler(list);
if (res === false || ctx.once === true) {
destroy(el);
}
}
});
ctx.observer.observe(el, ctx.opts);
}
function destroy (el) {
const ctx = el.__qmutation;
if (ctx !== void 0) {
ctx.observer !== void 0 && ctx.observer.disconnect();
delete el.__qmutation;
}
}
var Mutation = createDirective({
name: 'mutation',
mounted (el, { modifiers: { once, ...mod }, value }) {
const ctx = {
once,
opts: Object.keys(mod).length === 0
? defaultCfg
: mod
};
update$2(el, ctx, value);
el.__qmutation = ctx;
},
updated (el, { oldValue, value }) {
const ctx = el.__qmutation;
if (ctx !== void 0 && oldValue !== value) {
update$2(el, ctx, value);
}
},
beforeUnmount: destroy
}
);
const { passive } = listenOpts;
function update$1 (ctx, { value, oldValue }) {
if (typeof value !== 'function') {
ctx.scrollTarget.removeEventListener('scroll', ctx.scroll, passive);
return
}
ctx.handler = value;
if (typeof oldValue !== 'function') {
ctx.scrollTarget.addEventListener('scroll', ctx.scroll, passive);
ctx.scroll();
}
}
var ScrollFire = createDirective({
name: 'scroll-fire',
mounted (el, binding) {
const ctx = {
scrollTarget: getScrollTarget(el),
scroll: debounce(() => {
let containerBottom, elBottom;
if (ctx.scrollTarget === window) {
elBottom = el.getBoundingClientRect().bottom;
containerBottom = window.innerHeight;
}
else {
elBottom = offset(el).top + height(el);
containerBottom = offset(ctx.scrollTarget).top + height(ctx.scrollTarget);
}
if (elBottom > 0 && elBottom < containerBottom) {
ctx.scrollTarget.removeEventListener('scroll', ctx.scroll, passive);
ctx.handler(el);
}
}, 25)
};
update$1(ctx, binding);
el.__qscrollfire = ctx;
},
updated (el, binding) {
if (binding.value !== binding.oldValue) {
update$1(el.__qscrollfire, binding);
}
},
beforeUnmount (el) {
const ctx = el.__qscrollfire;
ctx.scrollTarget.removeEventListener('scroll', ctx.scroll, passive);
ctx.scroll.cancel();
delete el.__qscrollfire;
}
}
);
function update (ctx, { value, oldValue }) {
if (typeof value !== 'function') {
ctx.scrollTarget.removeEventListener('scroll', ctx.scroll, listenOpts.passive);
return
}
ctx.handler = value;
if (typeof oldValue !== 'function') {
ctx.scrollTarget.addEventListener('scroll', ctx.scroll, listenOpts.passive);
}
}
var Scroll = createDirective({
name: 'scroll',
mounted (el, binding) {
const ctx = {
scrollTarget: getScrollTarget(el),
scroll () {
ctx.handler(
getVerticalScrollPosition(ctx.scrollTarget),
getHorizontalScrollPosition(ctx.scrollTarget)
);
}
};
update(ctx, binding);
el.__qscroll = ctx;
},
updated (el, binding) {
if (el.__qscroll !== void 0 && binding.oldValue !== binding.value) {
update(el.__qscroll, binding);
}
},
beforeUnmount (el) {
const ctx = el.__qscroll;
ctx.scrollTarget.removeEventListener('scroll', ctx.scroll, listenOpts.passive);
delete el.__qscroll;
}
}
);
var TouchHold = createDirective({
name: 'touch-hold',
beforeMount (el, binding) {
const { modifiers } = binding;
// early return, we don't need to do anything
if (modifiers.mouse !== true && client.has.touch !== true) {
return
}
const ctx = {
handler: binding.value,
noop,
mouseStart (evt) {
if (typeof ctx.handler === 'function' && leftClick(evt) === true) {
addEvt(ctx, 'temp', [
[ document, 'mousemove', 'move', 'passiveCapture' ],
[ document, 'click', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt, true);
}
},
touchStart (evt) {
if (evt.target !== void 0 && typeof ctx.handler === 'function') {
const target = evt.target;
addEvt(ctx, 'temp', [
[ target, 'touchmove', 'move', 'passiveCapture' ],
[ target, 'touchcancel', 'end', 'notPassiveCapture' ],
[ target, 'touchend', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt);
}
},
start (evt, mouseEvent) {
ctx.origin = position(evt);
const startTime = Date.now();
if (client.is.mobile === true) {
document.body.classList.add('non-selectable');
clearSelection();
ctx.styleCleanup = withDelay => {
ctx.styleCleanup = void 0;
const remove = () => {
document.body.classList.remove('non-selectable');
};
if (withDelay === true) {
clearSelection();
setTimeout(remove, 10);
}
else { remove(); }
};
}
ctx.triggered = false;
ctx.sensitivity = mouseEvent === true
? ctx.mouseSensitivity
: ctx.touchSensitivity;
ctx.timer = setTimeout(() => {
ctx.timer = void 0;
clearSelection();
ctx.triggered = true;
ctx.handler({
evt,
touch: mouseEvent !== true,
mouse: mouseEvent === true,
position: ctx.origin,
duration: Date.now() - startTime
});
}, ctx.duration);
},
move (evt) {
const { top, left } = position(evt);
if (
ctx.timer !== void 0 && (
Math.abs(left - ctx.origin.left) >= ctx.sensitivity
|| Math.abs(top - ctx.origin.top) >= ctx.sensitivity
)
) {
clearTimeout(ctx.timer);
ctx.timer = void 0;
}
},
end (evt) {
cleanEvt(ctx, 'temp');
// delay needed otherwise selection still occurs
ctx.styleCleanup !== void 0 && ctx.styleCleanup(ctx.triggered);
if (ctx.triggered === true) {
evt !== void 0 && stopAndPrevent(evt);
}
else if (ctx.timer !== void 0) {
clearTimeout(ctx.timer);
ctx.timer = void 0;
}
}
};
// duration in ms, touch in pixels, mouse in pixels
const data = [ 600, 5, 7 ];
if (typeof binding.arg === 'string' && binding.arg.length !== 0) {
binding.arg.split(':').forEach((val, index) => {
const v = parseInt(val, 10);
v && (data[ index ] = v);
});
}
[ ctx.duration, ctx.touchSensitivity, ctx.mouseSensitivity ] = data;
el.__qtouchhold = ctx;
if (modifiers.mouse === true) {
// account for UMD too where modifiers will be lowercased to work
const capture = modifiers.mouseCapture === true || modifiers.mousecapture === true
? 'Capture'
: '';
addEvt(ctx, 'main', [
[ el, 'mousedown', 'mouseStart', `passive${ capture }` ]
]);
}
client.has.touch === true && addEvt(ctx, 'main', [
[ el, 'touchstart', 'touchStart', `passive${ modifiers.capture === true ? 'Capture' : '' }` ],
[ el, 'touchend', 'noop', 'notPassiveCapture' ]
]);
},
updated (el, binding) {
const ctx = el.__qtouchhold;
if (ctx !== void 0 && binding.oldValue !== binding.value) {
typeof binding.value !== 'function' && ctx.end();
ctx.handler = binding.value;
}
},
beforeUnmount (el) {
const ctx = el.__qtouchhold;
if (ctx !== void 0) {
cleanEvt(ctx, 'main');
cleanEvt(ctx, 'temp');
ctx.timer !== void 0 && clearTimeout(ctx.timer);
ctx.styleCleanup !== void 0 && ctx.styleCleanup();
delete el.__qtouchhold;
}
}
}
);
const
keyCodes = {
esc: 27,
tab: 9,
enter: 13,
space: 32,
up: 38,
left: 37,
right: 39,
down: 40,
delete: [ 8, 46 ]
},
keyRegex = new RegExp(`^([\\d+]+|${ Object.keys(keyCodes).join('|') })$`, 'i');
function shouldEnd (evt, origin) {
const { top, left } = position(evt);
return Math.abs(left - origin.left) >= 7
|| Math.abs(top - origin.top) >= 7
}
var TouchRepeat = createDirective({
name: 'touch-repeat',
beforeMount (el, { modifiers, value, arg }) {
const keyboard = Object.keys(modifiers).reduce((acc, key) => {
if (keyRegex.test(key) === true) {
const keyCode = isNaN(parseInt(key, 10)) ? keyCodes[ key.toLowerCase() ] : parseInt(key, 10);
keyCode >= 0 && acc.push(keyCode);
}
return acc
}, []);
// early return, we don't need to do anything
if (
modifiers.mouse !== true
&& client.has.touch !== true
&& keyboard.length === 0
) {
return
}
const durations = typeof arg === 'string' && arg.length !== 0
? arg.split(':').map(val => parseInt(val, 10))
: [ 0, 600, 300 ];
const durationsLast = durations.length - 1;
const ctx = {
keyboard,
handler: value,
noop,
mouseStart (evt) {
if (ctx.event === void 0 && typeof ctx.handler === 'function' && leftClick(evt) === true) {
addEvt(ctx, 'temp', [
[ document, 'mousemove', 'move', 'passiveCapture' ],
[ document, 'click', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt, true);
}
},
keyboardStart (evt) {
if (typeof ctx.handler === 'function' && isKeyCode(evt, keyboard) === true) {
if (durations[ 0 ] === 0 || ctx.event !== void 0) {
stopAndPrevent(evt);
el.focus();
if (ctx.event !== void 0) {
return
}
}
addEvt(ctx, 'temp', [
[ document, 'keyup', 'end', 'notPassiveCapture' ],
[ document, 'click', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt, false, true);
}
},
touchStart (evt) {
if (evt.target !== void 0 && typeof ctx.handler === 'function') {
const target = evt.target;
addEvt(ctx, 'temp', [
[ target, 'touchmove', 'move', 'passiveCapture' ],
[ target, 'touchcancel', 'end', 'notPassiveCapture' ],
[ target, 'touchend', 'end', 'notPassiveCapture' ]
]);
ctx.start(evt);
}
},
start (evt, mouseEvent, keyboardEvent) {
if (keyboardEvent !== true) {
ctx.origin = position(evt);
}
function styleCleanup (withDelay) {
ctx.styleCleanup = void 0;
document.documentElement.style.cursor = '';
const remove = () => {
document.body.classList.remove('non-selectable');
};
if (withDelay === true) {
clearSelection();
setTimeout(remove, 10);
}
else { remove(); }
}
if (client.is.mobile === true) {
document.body.classList.add('non-selectable');
clearSelection();
ctx.styleCleanup = styleCleanup;
}
ctx.event = {
touch: mouseEvent !== true && keyboardEvent !== true,
mouse: mouseEvent === true,
keyboard: keyboardEvent === true,
startTime: Date.now(),
repeatCount: 0
};
const fn = () => {
ctx.timer = void 0;
if (ctx.event === void 0) {
return
}
if (ctx.event.repeatCount === 0) {
ctx.event.evt = evt;
if (keyboardEvent === true) {
ctx.event.keyCode = evt.keyCode;
}
else {
ctx.event.position = position(evt);
}
if (client.is.mobile !== true) {
document.documentElement.style.cursor = 'pointer';
document.body.classList.add('non-selectable');
clearSelection();
ctx.styleCleanup = styleCleanup;
}
}
ctx.event.duration = Date.now() - ctx.event.startTime;
ctx.event.repeatCount += 1;
ctx.handler(ctx.event);
const index = durationsLast < ctx.event.repeatCount
? durationsLast
: ctx.event.repeatCount;
ctx.timer = setTimeout(fn, durations[ index ]);
};
if (durations[ 0 ] === 0) {
fn();
}
else {
ctx.timer = setTimeout(fn, durations[ 0 ]);
}
},
move (evt) {
if (ctx.event !== void 0 && ctx.timer !== void 0 && shouldEnd(evt, ctx.origin) === true) {
clearTimeout(ctx.timer);
ctx.timer = void 0;
}
},
end (evt) {
if (ctx.event === void 0) {
return
}
ctx.styleCleanup !== void 0 && ctx.styleCleanup(true);
evt !== void 0 && ctx.event.repeatCount > 0 && stopAndPrevent(evt);
cleanEvt(ctx, 'temp');
if (ctx.timer !== void 0) {
clearTimeout(ctx.timer);
ctx.timer = void 0;
}
ctx.event = void 0;
}
};
el.__qtouchrepeat = ctx;
if (modifiers.mouse === true) {
// account for UMD too where modifiers will be lowercased to work
const capture = modifiers.mouseCapture === true || modifiers.mousecapture === true
? 'Capture'
: '';
addEvt(ctx, 'main', [
[ el, 'mousedown', 'mouseStart', `passive${ capture }` ]
]);
}
client.has.touch === true && addEvt(ctx, 'main', [
[ el, 'touchstart', 'touchStart', `passive${ modifiers.capture === true ? 'Capture' : '' }` ],
[ el, 'touchend', 'noop', 'passiveCapture' ]
]);
if (keyboard.length !== 0) {
// account for UMD too where modifiers will be lowercased to work
const capture = modifiers.keyCapture === true || modifiers.keycapture === true
? 'Capture'
: '';
addEvt(ctx, 'main', [
[ el, 'keydown', 'keyboardStart', `notPassive${ capture }` ]
]);
}
},
updated (el, { oldValue, value }) {
const ctx = el.__qtouchrepeat;
if (ctx !== void 0 && oldValue !== value) {
typeof value !== 'function' && ctx.end();
ctx.handler = value;
}
},
beforeUnmount (el) {
const ctx = el.__qtouchrepeat;
if (ctx !== void 0) {
ctx.timer !== void 0 && clearTimeout(ctx.timer);
cleanEvt(ctx, 'main');
cleanEvt(ctx, 'temp');
ctx.styleCleanup !== void 0 && ctx.styleCleanup();
delete el.__qtouchrepeat;
}
}
}
);
var directives = /*#__PURE__*/Object.freeze({
__proto__: null,
ClosePopup: ClosePopup,
Intersection: Intersection,
Morph: Morph,
Mutation: Mutation,
Ripple: Ripple,
ScrollFire: ScrollFire,
Scroll: Scroll,
TouchHold: TouchHold,
TouchPan: TouchPan,
TouchRepeat: TouchRepeat,
TouchSwipe: TouchSwipe
});
function getCssVar (propName, element = document.body) {
if (typeof propName !== 'string') {
throw new TypeError('Expected a string as propName')
}
if (!(element instanceof Element)) {
throw new TypeError('Expected a DOM element')
}
return getComputedStyle(element).getPropertyValue(`--q-${ propName }`).trim() || null
}
let metaValue;
function getProp () {
return client.is.winphone
? 'msapplication-navbutton-color'
: (
client.is.safari
? 'apple-mobile-web-app-status-bar-style'
: 'theme-color' // Chrome, Firefox OS, Opera, Vivaldi, ...
)
}
function getMetaTag (v) {
const els = document.getElementsByTagName('META');
for (const i in els) {
if (els[ i ].name === v) {
return els[ i ]
}
}
}
function setColor (hexColor) {
if (metaValue === void 0) {
// cache it
metaValue = getProp();
}
let metaTag = getMetaTag(metaValue);
const newTag = metaTag === void 0;
if (newTag) {
metaTag = document.createElement('meta');
metaTag.setAttribute('name', metaValue);
}
metaTag.setAttribute('content', hexColor);
if (newTag) {
document.head.appendChild(metaTag);
}
}
var AddressbarColor = {
set: client.is.mobile === true && (
client.is.nativeMobile === true
|| client.is.winphone === true || client.is.safari === true
|| client.is.webkit === true || client.is.vivaldi === true
)
? hexColor => {
const val = hexColor || getCssVar('primary');
if (client.is.nativeMobile === true && window.StatusBar) {
window.StatusBar.backgroundColorByHexString(val);
}
else {
setColor(val);
}
}
: noop,
install ({ $q }) {
$q.addressbarColor = this;
$q.config.addressbarColor && this.set($q.config.addressbarColor);
}
};
const prefixes = {};
function assignFn (fn) {
Object.assign(Plugin$6, {
request: fn,
exit: fn,
toggle: fn
});
}
function getFullscreenElement () {
return (
document.fullscreenElement
|| document.mozFullScreenElement
|| document.webkitFullscreenElement
|| document.msFullscreenElement
|| null
)
}
function updateEl () {
const newEl = Plugin$6.activeEl = Plugin$6.isActive === false
? null
: getFullscreenElement();
changeGlobalNodesTarget(
newEl === null || newEl === document.documentElement
? document.body
: newEl
);
}
function togglePluginState () {
Plugin$6.isActive = Plugin$6.isActive === false;
updateEl();
}
// needed for consistency across browsers
function promisify (target, fn) {
try {
const res = target[ fn ]();
return res === void 0
? Promise.resolve()
: res
}
catch (err) {
return Promise.reject(err)
}
}
const Plugin$6 = defineReactivePlugin({
isActive: false,
activeEl: null
}, {
isCapable: false,
install ({ $q }) {
$q.fullscreen = this;
}
});
{
prefixes.request = [
'requestFullscreen',
'msRequestFullscreen', 'mozRequestFullScreen', 'webkitRequestFullscreen'
].find(request => document.documentElement[ request ] !== void 0);
Plugin$6.isCapable = prefixes.request !== void 0;
if (Plugin$6.isCapable === false) {
// it means the browser does NOT support it
assignFn(() => Promise.reject('Not capable'));
}
else {
Object.assign(Plugin$6, {
request (target) {
const el = target || document.documentElement;
const { activeEl } = Plugin$6;
if (el === activeEl) {
return Promise.resolve()
}
const queue = activeEl !== null && el.contains(activeEl) === true
? Plugin$6.exit()
: Promise.resolve();
return queue.finally(() => promisify(el, prefixes.request))
},
exit () {
return Plugin$6.isActive === true
? promisify(document, prefixes.exit)
: Promise.resolve()
},
toggle (target) {
return Plugin$6.isActive === true
? Plugin$6.exit()
: Plugin$6.request(target)
}
});
prefixes.exit = [
'exitFullscreen',
'msExitFullscreen', 'mozCancelFullScreen', 'webkitExitFullscreen'
].find(exit => document[ exit ]);
Plugin$6.isActive = Boolean(getFullscreenElement());
Plugin$6.isActive === true && updateEl()
;[
'onfullscreenchange',
'onmsfullscreenchange', 'onwebkitfullscreenchange'
].forEach(evt => {
document[ evt ] = togglePluginState;
});
}
}
const Plugin$5 = defineReactivePlugin({
appVisible: true
}, {
install ({ $q }) {
injectProp($q, 'appVisible', () => this.appVisible);
}
});
{
let prop, evt;
if (typeof document.hidden !== 'undefined') { // Opera 12.10 and Firefox 18 and later support
prop = 'hidden';
evt = 'visibilitychange';
}
else if (typeof document.msHidden !== 'undefined') {
prop = 'msHidden';
evt = 'msvisibilitychange';
}
else if (typeof document.webkitHidden !== 'undefined') {
prop = 'webkitHidden';
evt = 'webkitvisibilitychange';
}
if (evt && typeof document[ prop ] !== 'undefined') {
const update = () => { Plugin$5.appVisible = !document[ prop ]; };
document.addEventListener(evt, update, false);
}
}
var BottomSheet$1 = createComponent({
name: 'BottomSheetPlugin',
props: {
...useDarkProps,
title: String,
message: String,
actions: Array,
grid: Boolean,
cardClass: [ String, Array, Object ],
cardStyle: [ String, Array, Object ]
},
emits: [ 'ok', 'hide' ],
setup (props, { emit }) {
const { proxy } = vue.getCurrentInstance();
const isDark = useDark(props, proxy.$q);
const dialogRef = vue.ref(null);
function show () {
dialogRef.value.show();
}
function hide () {
dialogRef.value.hide();
}
function onOk (action) {
emit('ok', action);
hide();
}
function onHide () {
emit('hide');
}
function getGrid () {
return props.actions.map(action => {
const img = action.avatar || action.img;
return action.label === void 0
? vue.h(QSeparator, {
class: 'col-all',
dark: isDark.value
})
: vue.h('div', {
class: [
'q-bottom-sheet__item q-hoverable q-focusable cursor-pointer relative-position',
action.class
],
style: action.style,
tabindex: 0,
role: 'listitem',
onClick () { onOk(action); },
onKeyup (e) { e.keyCode === 13 && onOk(action); }
}, [
vue.h('div', { class: 'q-focus-helper' }),
action.icon
? vue.h(QIcon, { name: action.icon, color: action.color })
: (
img
? vue.h('img', {
class: action.avatar ? 'q-bottom-sheet__avatar' : '',
src: img
})
: vue.h('div', { class: 'q-bottom-sheet__empty-icon' })
),
vue.h('div', action.label)
])
})
}
function getList () {
return props.actions.map(action => {
const img = action.avatar || action.img;
return action.label === void 0
? vue.h(QSeparator, { spaced: true, dark: isDark.value })
: vue.h(QItem, {
class: [ 'q-bottom-sheet__item', action.classes ],
style: action.style,
tabindex: 0,
clickable: true,
dark: isDark.value,
onClick () { onOk(action); }
}, () => [
vue.h(
QItemSection,
{ avatar: true },
() => (
action.icon
? vue.h(QIcon, { name: action.icon, color: action.color })
: (
img
? vue.h('img', {
class: action.avatar ? 'q-bottom-sheet__avatar' : '',
src: img
})
: null
)
)
),
vue.h(QItemSection, () => action.label)
])
})
}
function getCardContent () {
const child = [];
props.title && child.push(
vue.h(QCardSection, {
class: 'q-dialog__title'
}, () => props.title)
);
props.message && child.push(
vue.h(QCardSection, {
class: 'q-dialog__message'
}, () => props.message)
);
child.push(
props.grid === true
? vue.h('div', {
class: 'row items-stretch justify-start',
role: 'list'
}, getGrid())
: vue.h('div', {
role: 'list'
}, getList())
);
return child
}
function getContent () {
return [
vue.h(QCard, {
class: [
`q-bottom-sheet q-bottom-sheet--${ props.grid === true ? 'grid' : 'list' }`
+ (isDark.value === true ? ' q-bottom-sheet--dark q-dark' : ''),
props.cardClass
],
style: props.cardStyle
}, getCardContent)
]
}
// expose public methods
Object.assign(proxy, { show, hide });
return () => vue.h(QDialog, {
ref: dialogRef,
position: 'bottom',
onHide
}, getContent)
}
});
function merge (target, source) {
for (const key in source) {
if (key !== 'spinner' && Object(source[ key ]) === source[ key ]) {
target[ key ] = Object(target[ key ]) !== target[ key ]
? {}
: { ...target[ key ] };
merge(target[ key ], source[ key ]);
}
else {
target[ key ] = source[ key ];
}
}
}
function globalDialog (DefaultComponent, supportsCustomComponent, parentApp) {
return pluginProps => {
let DialogComponent, props;
const isCustom = supportsCustomComponent === true
&& pluginProps.component !== void 0;
if (isCustom === true) {
const { component, componentProps } = pluginProps;
DialogComponent = (typeof component === 'string')
? parentApp.component(component)
: component;
props = componentProps || {};
}
else {
const { class: klass, style, ...otherProps } = pluginProps;
DialogComponent = DefaultComponent;
props = otherProps;
klass !== void 0 && (otherProps.cardClass = klass);
style !== void 0 && (otherProps.cardStyle = style);
}
let vm, emittedOK = false;
const dialogRef = vue.ref(null);
const el = createGlobalNode(false, 'dialog');
const applyState = cmd => {
if (dialogRef.value !== null && dialogRef.value[ cmd ] !== void 0) {
dialogRef.value[ cmd ]();
return
}
const target = vm.$.subTree;
if (target && target.component) {
// account for "script setup" way of declaring component
if (target.component.proxy && target.component.proxy[ cmd ]) {
target.component.proxy[ cmd ]();
return
}
// account for "script setup" + async component way of declaring component
if (
target.component.subTree
&& target.component.subTree.component
&& target.component.subTree.component.proxy
&& target.component.subTree.component.proxy[ cmd ]
) {
target.component.subTree.component.proxy[ cmd ]();
return
}
}
console.error('[Quasar] Incorrectly defined Dialog component');
};
const
okFns = [],
cancelFns = [],
API = {
onOk (fn) {
okFns.push(fn);
return API
},
onCancel (fn) {
cancelFns.push(fn);
return API
},
onDismiss (fn) {
okFns.push(fn);
cancelFns.push(fn);
return API
},
hide () {
applyState('hide');
return API
},
update (componentProps) {
if (vm !== null) {
if (isCustom === true) {
Object.assign(props, componentProps);
}
else {
const { class: klass, style, ...cfg } = componentProps;
klass !== void 0 && (cfg.cardClass = klass);
style !== void 0 && (cfg.cardStyle = style);
merge(props, cfg);
}
vm.$forceUpdate();
}
return API
}
};
const onOk = data => {
emittedOK = true;
okFns.forEach(fn => { fn(data); });
};
const onHide = () => {
app.unmount(el);
removeGlobalNode(el);
app = null;
vm = null;
if (emittedOK !== true) {
cancelFns.forEach(fn => { fn(); });
}
};
let app = createChildApp({
name: 'QGlobalDialog',
setup: () => () => vue.h(DialogComponent, {
...props,
ref: dialogRef,
onOk,
onHide,
onVnodeMounted (...args) {
if (typeof props.onVnodeMounted === 'function') {
props.onVnodeMounted(...args);
}
vue.nextTick(() => applyState('show'));
}
})
}, parentApp);
vm = app.mount(el);
return API
}
}
var BottomSheet = {
install ({ $q, parentApp }) {
$q.bottomSheet = globalDialog(BottomSheet$1, false, parentApp);
if (this.__installed !== true) {
this.create = $q.bottomSheet;
}
}
};
function encode$1 (string) {
return encodeURIComponent(string)
}
function decode$1 (string) {
return decodeURIComponent(string)
}
function stringifyCookieValue (value) {
return encode$1(value === Object(value) ? JSON.stringify(value) : '' + value)
}
function read (string) {
if (string === '') {
return string
}
if (string.indexOf('"') === 0) {
// This is a quoted cookie as according to RFC2068, unescape...
string = string.slice(1, -1).replace(/\\"/g, '"').replace(/\\\\/g, '\\');
}
// Replace server-side written pluses with spaces.
// If we can't decode the cookie, ignore it, it's unusable.
// If we can't parse the cookie, ignore it, it's unusable.
string = decode$1(string.replace(/\+/g, ' '));
try {
const parsed = JSON.parse(string);
if (parsed === Object(parsed) || Array.isArray(parsed) === true) {
string = parsed;
}
}
catch (e) {}
return string
}
function getString (msOffset) {
const time = new Date();
time.setMilliseconds(time.getMilliseconds() + msOffset);
return time.toUTCString()
}
function parseExpireString (str) {
let timestamp = 0;
const days = str.match(/(\d+)d/);
const hours = str.match(/(\d+)h/);
const minutes = str.match(/(\d+)m/);
const seconds = str.match(/(\d+)s/);
if (days) { timestamp += days[ 1 ] * 864e+5; }
if (hours) { timestamp += hours[ 1 ] * 36e+5; }
if (minutes) { timestamp += minutes[ 1 ] * 6e+4; }
if (seconds) { timestamp += seconds[ 1 ] * 1000; }
return timestamp === 0
? str
: getString(timestamp)
}
function set (key, val, opts = {}, ssr) {
let expire, expireValue;
if (opts.expires !== void 0) {
// if it's a Date Object
if (Object.prototype.toString.call(opts.expires) === '[object Date]') {
expire = opts.expires.toUTCString();
}
// if it's a String (eg. "15m", "1h", "13d", "1d 15m", "31s")
// possible units: d (days), h (hours), m (minutes), s (seconds)
else if (typeof opts.expires === 'string') {
expire = parseExpireString(opts.expires);
}
// otherwise it must be a Number (defined in days)
else {
expireValue = parseFloat(opts.expires);
expire = isNaN(expireValue) === false
? getString(expireValue * 864e+5)
: opts.expires;
}
}
const keyValue = `${ encode$1(key) }=${ stringifyCookieValue(val) }`;
const cookie = [
keyValue,
expire !== void 0 ? '; Expires=' + expire : '', // use expires attribute, max-age is not supported by IE
opts.path ? '; Path=' + opts.path : '',
opts.domain ? '; Domain=' + opts.domain : '',
opts.sameSite ? '; SameSite=' + opts.sameSite : '',
opts.httpOnly ? '; HttpOnly' : '',
opts.secure ? '; Secure' : '',
opts.other ? '; ' + opts.other : ''
].join('');
if (ssr) {
if (ssr.req.qCookies) {
ssr.req.qCookies.push(cookie);
}
else {
ssr.req.qCookies = [ cookie ];
}
ssr.res.setHeader('Set-Cookie', ssr.req.qCookies);
// make temporary update so future get()
// within same SSR timeframe would return the set value
let all = ssr.req.headers.cookie || '';
if (expire !== void 0 && expireValue < 0) {
const val = get(key, ssr);
if (val !== undefined) {
all = all
.replace(`${ key }=${ val }; `, '')
.replace(`; ${ key }=${ val }`, '')
.replace(`${ key }=${ val }`, '');
}
}
else {
all = all
? `${ keyValue }; ${ all }`
: cookie;
}
ssr.req.headers.cookie = all;
}
else {
document.cookie = cookie;
}
}
function get (key, ssr) {
const
cookieSource = ssr ? ssr.req.headers : document,
cookies = cookieSource.cookie ? cookieSource.cookie.split('; ') : [],
l = cookies.length;
let
result = key ? null : {},
i = 0,
parts,
name,
cookie;
for (; i < l; i++) {
parts = cookies[ i ].split('=');
name = decode$1(parts.shift());
cookie = parts.join('=');
if (!key) {
result[ name ] = cookie;
}
else if (key === name) {
result = read(cookie);
break
}
}
return result
}
function remove (key, options, ssr) {
set(
key,
'',
{ expires: -1, ...options },
ssr
);
}
function has (key, ssr) {
return get(key, ssr) !== null
}
function getObject (ssr) {
return {
get: key => get(key, ssr),
set: (key, val, opts) => set(key, val, opts, ssr),
has: key => has(key, ssr),
remove: (key, options) => remove(key, options, ssr),
getAll: () => get(null, ssr)
}
}
const Plugin$4 = {
install ({ $q, ssrContext }) {
$q.cookies = this;
}
};
{
Object.assign(Plugin$4, getObject());
}
var DialogPlugin = createComponent({
name: 'DialogPlugin',
props: {
...useDarkProps,
title: String,
message: String,
prompt: Object,
options: Object,
progress: [ Boolean, Object ],
html: Boolean,
ok: {
type: [ String, Object, Boolean ],
default: true
},
cancel: [ String, Object, Boolean ],
focus: {
type: String,
default: 'ok',
validator: v => [ 'ok', 'cancel', 'none' ].includes(v)
},
stackButtons: Boolean,
color: String,
cardClass: [ String, Array, Object ],
cardStyle: [ String, Array, Object ]
},
emits: [ 'ok', 'hide' ],
setup (props, { emit }) {
const { proxy } = vue.getCurrentInstance();
const { $q } = proxy;
const isDark = useDark(props, $q);
const dialogRef = vue.ref(null);
const model = vue.ref(
props.prompt !== void 0
? props.prompt.model
: (props.options !== void 0 ? props.options.model : void 0)
);
const classes = vue.computed(() =>
'q-dialog-plugin'
+ (isDark.value === true ? ' q-dialog-plugin--dark q-dark' : '')
+ (props.progress !== false ? ' q-dialog-plugin--progress' : '')
);
const vmColor = vue.computed(() =>
props.color || (isDark.value === true ? 'amber' : 'primary')
);
const spinner = vue.computed(() => (
props.progress === false
? null
: (
isObject(props.progress) === true
? {
component: props.progress.spinner || QSpinner,
props: { color: props.progress.color || vmColor.value }
}
: {
component: QSpinner,
props: { color: vmColor.value }
}
)
));
const hasForm = vue.computed(() =>
props.prompt !== void 0 || props.options !== void 0
);
const formProps = vue.computed(() => {
if (hasForm.value !== true) {
return {}
}
const { model, isValid, items, ...formProps } = props.prompt !== void 0
? props.prompt
: props.options;
return formProps
});
const okLabel = vue.computed(() => (
isObject(props.ok) === true
? $q.lang.label.ok
: (
props.ok === true
? $q.lang.label.ok
: props.ok
)
));
const cancelLabel = vue.computed(() => (
isObject(props.cancel) === true
? $q.lang.label.cancel
: (
props.cancel === true
? $q.lang.label.cancel
: props.cancel
)
));
const okDisabled = vue.computed(() => {
if (props.prompt !== void 0) {
return props.prompt.isValid !== void 0
&& props.prompt.isValid(model.value) !== true
}
if (props.options !== void 0) {
return props.options.isValid !== void 0
&& props.options.isValid(model.value) !== true
}
return false
});
const okProps = vue.computed(() => ({
color: vmColor.value,
label: okLabel.value,
ripple: false,
disable: okDisabled.value,
...(isObject(props.ok) === true ? props.ok : { flat: true }),
'data-autofocus': (props.focus === 'ok' && hasForm.value !== true) || void 0,
onClick: onOk
}));
const cancelProps = vue.computed(() => ({
color: vmColor.value,
label: cancelLabel.value,
ripple: false,
...(isObject(props.cancel) === true ? props.cancel : { flat: true }),
'data-autofocus': (props.focus === 'cancel' && hasForm.value !== true) || void 0,
onClick: onCancel
}));
vue.watch(() => props.prompt && props.prompt.model, onUpdateModel);
vue.watch(() => props.options && props.options.model, onUpdateModel);
function show () {
dialogRef.value.show();
}
function hide () {
dialogRef.value.hide();
}
function onOk () {
emit('ok', vue.toRaw(model.value));
hide();
}
function onCancel () {
hide();
}
function onDialogHide () {
emit('hide');
}
function onUpdateModel (val) {
model.value = val;
}
function onInputKeyup (evt) {
// if ENTER key
if (
okDisabled.value !== true
&& props.prompt.type !== 'textarea'
&& isKeyCode(evt, 13) === true
) {
onOk();
}
}
function getSection (classes, text) {
return props.html === true
? vue.h(QCardSection, {
class: classes,
innerHTML: text
})
: vue.h(QCardSection, { class: classes }, () => text)
}
function getPrompt () {
return [
vue.h(QInput, {
color: vmColor.value,
dense: true,
autofocus: true,
dark: isDark.value,
...formProps.value,
modelValue: model.value,
'onUpdate:modelValue': onUpdateModel,
onKeyup: onInputKeyup
})
]
}
function getOptions () {
return [
vue.h(QOptionGroup, {
color: vmColor.value,
options: props.options.items,
dark: isDark.value,
...formProps.value,
modelValue: model.value,
'onUpdate:modelValue': onUpdateModel
})
]
}
function getButtons () {
const child = [];
props.cancel && child.push(
vue.h(QBtn, cancelProps.value)
);
props.ok && child.push(
vue.h(QBtn, okProps.value)
);
return vue.h(QCardActions, {
class: props.stackButtons === true ? 'items-end' : '',
vertical: props.stackButtons,
align: 'right'
}, () => child)
}
function getCardContent () {
const child = [];
props.title && child.push(
getSection('q-dialog__title', props.title)
);
props.progress !== false && child.push(
vue.h(
QCardSection,
{ class: 'q-dialog__progress' },
() => vue.h(spinner.value.component, spinner.value.props)
)
);
props.message && child.push(
getSection('q-dialog__message', props.message)
);
if (props.prompt !== void 0) {
child.push(
vue.h(
QCardSection,
{ class: 'scroll q-dialog-plugin__form' },
getPrompt
)
);
}
else if (props.options !== void 0) {
child.push(
vue.h(QSeparator, { dark: isDark.value }),
vue.h(
QCardSection,
{ class: 'scroll q-dialog-plugin__form' },
getOptions
),
vue.h(QSeparator, { dark: isDark.value })
);
}
if (props.ok || props.cancel) {
child.push(getButtons());
}
return child
}
function getContent () {
return [
vue.h(QCard, {
class: [
classes.value,
props.cardClass
],
style: props.cardStyle,
dark: isDark.value
}, getCardContent)
]
}
// expose public methods
Object.assign(proxy, { show, hide });
return () => vue.h(QDialog, {
ref: dialogRef,
onHide: onDialogHide
}, getContent)
}
});
var Dialog = {
install ({ $q, parentApp }) {
$q.dialog = globalDialog(DialogPlugin, true, parentApp);
if (this.__installed !== true) {
this.create = $q.dialog;
}
}
};
const barRef = vue.ref(null);
const Plugin$3 = defineReactivePlugin({
isActive: false
}, {
start: noop,
stop: noop,
increment: noop,
setDefaults: noop,
install ({ $q, parentApp }) {
$q.loadingBar = this;
if (this.__installed === true) {
if ($q.config.loadingBar !== void 0) {
this.setDefaults($q.config.loadingBar);
}
return
}
const props = vue.ref(
$q.config.loadingBar !== void 0
? { ...$q.config.loadingBar }
: {}
);
function onStart () {
Plugin$3.isActive = true;
}
function onStop () {
Plugin$3.isActive = false;
}
const el = createGlobalNode('q-loading-bar');
createChildApp({
name: 'LoadingBar',
// hide App from Vue devtools
devtools: { hide: true },
setup: () => () => vue.h(QAjaxBar, { ...props.value, onStart, onStop, ref: barRef })
}, parentApp).mount(el);
Object.assign(this, {
start (speed) {
barRef.value.start(speed);
},
stop () {
barRef.value.stop();
},
increment () {
barRef.value.increment.apply(null, arguments);
},
setDefaults (opts) {
if (isObject(opts) === true) {
Object.assign(props.value, opts);
}
}
});
}
});
let
app,
vm,
uid$1 = 0,
timeout = null,
props = {},
activeGroups = {};
const originalDefaults = {
group: '__default_quasar_group__',
delay: 0,
message: false,
html: false,
spinnerSize: 80,
spinnerColor: '',
messageColor: '',
backgroundColor: '',
boxClass: '',
spinner: QSpinner,
customClass: ''
};
const defaults$1 = { ...originalDefaults };
function registerProps (opts) {
if (opts && opts.group !== void 0 && activeGroups[ opts.group ] !== void 0) {
return Object.assign(activeGroups[ opts.group ], opts)
}
const newProps = isObject(opts) === true && opts.ignoreDefaults === true
? { ...originalDefaults, ...opts }
: { ...defaults$1, ...opts };
activeGroups[ newProps.group ] = newProps;
return newProps
}
const Plugin$2 = defineReactivePlugin({
isActive: false
}, {
show (opts) {
props = registerProps(opts);
const { group } = props;
Plugin$2.isActive = true;
if (app !== void 0) {
props.uid = uid$1;
vm.$forceUpdate();
}
else {
props.uid = ++uid$1;
timeout !== null && clearTimeout(timeout);
timeout = setTimeout(() => {
timeout = null;
const el = createGlobalNode('q-loading');
app = createChildApp({
name: 'QLoading',
setup () {
vue.onMounted(() => {
preventScroll(true);
});
function onAfterLeave () {
// might be called to finalize
// previous leave, even if it was cancelled
if (Plugin$2.isActive !== true && app !== void 0) {
preventScroll(false);
app.unmount(el);
removeGlobalNode(el);
app = void 0;
vm = void 0;
}
}
function getContent () {
if (Plugin$2.isActive !== true) {
return null
}
const content = [
vue.h(props.spinner, {
class: 'q-loading__spinner',
color: props.spinnerColor,
size: props.spinnerSize
})
];
props.message && content.push(
vue.h('div', {
class: 'q-loading__message'
+ (props.messageColor ? ` text-${ props.messageColor }` : ''),
[ props.html === true ? 'innerHTML' : 'textContent' ]: props.message
})
);
return vue.h('div', {
class: 'q-loading fullscreen flex flex-center z-max ' + props.customClass.trim(),
key: props.uid
}, [
vue.h('div', {
class: 'q-loading__backdrop'
+ (props.backgroundColor ? ` bg-${ props.backgroundColor }` : '')
}),
vue.h('div', {
class: 'q-loading__box column items-center ' + props.boxClass
}, content)
])
}
return () => vue.h(vue.Transition, {
name: 'q-transition--fade',
appear: true,
onAfterLeave
}, getContent)
}
}, Plugin$2.__parentApp);
vm = app.mount(el);
}, props.delay);
}
return paramProps => {
// if we don't have params (or not an Object param) then we need to hide this group
if (paramProps === void 0 || Object(paramProps) !== paramProps) {
Plugin$2.hide(group);
return
}
// else we have params so we need to update this group
Plugin$2.show({ ...paramProps, group });
}
},
hide (group) {
if (Plugin$2.isActive === true) {
if (group === void 0) {
// clear out any active groups
activeGroups = {};
}
else if (activeGroups[ group ] === void 0) {
// we've already hidden it so nothing to do
return
}
else {
// remove active group
delete activeGroups[ group ];
const keys = Object.keys(activeGroups);
// if there are other groups registered then
// show last registered one since that one is still active
if (keys.length !== 0) {
// get last registered group
const lastGroup = keys[ keys.length - 1 ];
Plugin$2.show({ group: lastGroup });
return
}
}
if (timeout !== null) {
clearTimeout(timeout);
timeout = null;
}
Plugin$2.isActive = false;
}
},
setDefaults (opts) {
{
isObject(opts) === true && Object.assign(defaults$1, opts);
}
},
install ({ $q, parentApp }) {
$q.loading = this;
{
Plugin$2.__parentApp = parentApp;
if ($q.config.loading !== void 0) {
this.setDefaults($q.config.loading);
}
}
}
});
let updateId = null, currentClientMeta;
const clientList = [];
function normalize (meta) {
if (meta.title) {
meta.title = meta.titleTemplate
? meta.titleTemplate(meta.title)
: meta.title;
delete meta.titleTemplate;
}
[ [ 'meta', 'content' ], [ 'link', 'href' ] ].forEach(type => {
const
metaType = meta[ type[ 0 ] ],
metaProp = type[ 1 ];
for (const name in metaType) {
const metaLink = metaType[ name ];
if (metaLink.template) {
if (Object.keys(metaLink).length === 1) {
delete metaType[ name ];
}
else {
metaLink[ metaProp ] = metaLink.template(metaLink[ metaProp ] || '');
delete metaLink.template;
}
}
}
});
}
function changed (old, def) {
if (Object.keys(old).length !== Object.keys(def).length) {
return true
}
for (const key in old) {
if (old[ key ] !== def[ key ]) {
return true
}
}
}
function bodyFilter (name) {
return [ 'class', 'style' ].includes(name) === false
}
function htmlFilter (name) {
return [ 'lang', 'dir' ].includes(name) === false
}
function diff (meta, other) {
const add = {}, remove = {};
if (meta === void 0) {
return { add: other, remove }
}
if (meta.title !== other.title) {
add.title = other.title;
}
[ 'meta', 'link', 'script', 'htmlAttr', 'bodyAttr' ].forEach(type => {
const old = meta[ type ], cur = other[ type ];
remove[ type ] = [];
if (old === void 0 || old === null) {
add[ type ] = cur;
return
}
add[ type ] = {};
for (const key in old) {
if (cur.hasOwnProperty(key) === false) {
remove[ type ].push(key);
}
}
for (const key in cur) {
if (old.hasOwnProperty(key) === false) {
add[ type ][ key ] = cur[ key ];
}
else if (changed(old[ key ], cur[ key ]) === true) {
remove[ type ].push(key);
add[ type ][ key ] = cur[ key ];
}
}
});
return { add, remove }
}
function apply ({ add, remove }) {
if (add.title) {
document.title = add.title;
}
if (Object.keys(remove).length !== 0) {
[ 'meta', 'link', 'script' ].forEach(type => {
remove[ type ].forEach(name => {
document.head.querySelector(`${ type }[data-qmeta="${ name }"]`).remove();
});
});
remove.htmlAttr.filter(htmlFilter).forEach(name => {
document.documentElement.removeAttribute(name);
});
remove.bodyAttr.filter(bodyFilter).forEach(name => {
document.body.removeAttribute(name);
});
}
[ 'meta', 'link', 'script' ].forEach(type => {
const metaType = add[ type ];
for (const name in metaType) {
const tag = document.createElement(type);
for (const att in metaType[ name ]) {
if (att !== 'innerHTML') {
tag.setAttribute(att, metaType[ name ][ att ]);
}
}
tag.setAttribute('data-qmeta', name);
if (type === 'script') {
tag.innerHTML = metaType[ name ].innerHTML || '';
}
document.head.appendChild(tag);
}
});
Object.keys(add.htmlAttr).filter(htmlFilter).forEach(name => {
document.documentElement.setAttribute(name, add.htmlAttr[ name ] || '');
});
Object.keys(add.bodyAttr).filter(bodyFilter).forEach(name => {
document.body.setAttribute(name, add.bodyAttr[ name ] || '');
});
}
function updateClientMeta () {
updateId = null;
const data = {
title: '',
titleTemplate: null,
meta: {},
link: {},
script: {},
htmlAttr: {},
bodyAttr: {}
};
for (let i = 0; i < clientList.length; i++) {
const { active, val } = clientList[ i ];
if (active === true) {
extend(true, data, val);
}
}
normalize(data);
apply(diff(currentClientMeta, data));
currentClientMeta = data;
}
function planClientUpdate () {
updateId !== null && clearTimeout(updateId);
updateId = setTimeout(updateClientMeta, 50);
}
var Meta = {
install (opts) {
if (this.__installed !== true && isRuntimeSsrPreHydration.value === true) {
currentClientMeta = window.__Q_META__;
document.getElementById('qmeta-init').remove();
}
}
};
let uid = 0;
const defaults = {};
const groups = {};
const notificationsList = {};
const positionClass = {};
const emptyRE = /^\s*$/;
const notifRefs = [];
const positionList = [
'top-left', 'top-right',
'bottom-left', 'bottom-right',
'top', 'bottom', 'left', 'right', 'center'
];
const badgePositions = [
'top-left', 'top-right',
'bottom-left', 'bottom-right'
];
const notifTypes = {
positive: {
icon: $q => $q.iconSet.type.positive,
color: 'positive'
},
negative: {
icon: $q => $q.iconSet.type.negative,
color: 'negative'
},
warning: {
icon: $q => $q.iconSet.type.warning,
color: 'warning',
textColor: 'dark'
},
info: {
icon: $q => $q.iconSet.type.info,
color: 'info'
},
ongoing: {
group: false,
timeout: 0,
spinner: true,
color: 'grey-8'
}
};
function addNotification (config, $q, originalApi) {
if (!config) {
return logError('parameter required')
}
let Api;
const notif = { textColor: 'white' };
if (config.ignoreDefaults !== true) {
Object.assign(notif, defaults);
}
if (isObject(config) === false) {
if (notif.type) {
Object.assign(notif, notifTypes[ notif.type ]);
}
config = { message: config };
}
Object.assign(notif, notifTypes[ config.type || notif.type ], config);
if (typeof notif.icon === 'function') {
notif.icon = notif.icon($q);
}
if (!notif.spinner) {
notif.spinner = false;
}
else {
if (notif.spinner === true) {
notif.spinner = QSpinner;
}
notif.spinner = vue.markRaw(notif.spinner);
}
notif.meta = {
hasMedia: Boolean(notif.spinner !== false || notif.icon || notif.avatar),
hasText: hasContent(notif.message) || hasContent(notif.caption)
};
if (notif.position) {
if (positionList.includes(notif.position) === false) {
return logError('wrong position', config)
}
}
else {
notif.position = 'bottom';
}
if (notif.timeout === void 0) {
notif.timeout = 5000;
}
else {
const t = parseInt(notif.timeout, 10);
if (isNaN(t) || t < 0) {
return logError('wrong timeout', config)
}
notif.timeout = t;
}
if (notif.timeout === 0) {
notif.progress = false;
}
else if (notif.progress === true) {
notif.meta.progressClass = 'q-notification__progress' + (
notif.progressClass
? ` ${ notif.progressClass }`
: ''
);
notif.meta.progressStyle = {
animationDuration: `${ notif.timeout + 1000 }ms`
};
}
const actions = (
Array.isArray(config.actions) === true
? config.actions
: []
).concat(
config.ignoreDefaults !== true && Array.isArray(defaults.actions) === true
? defaults.actions
: []
).concat(
notifTypes[ config.type ] !== void 0 && Array.isArray(notifTypes[ config.type ].actions) === true
? notifTypes[ config.type ].actions
: []
);
const { closeBtn } = notif;
closeBtn && actions.push({
label: typeof closeBtn === 'string'
? closeBtn
: $q.lang.label.close
});
notif.actions = actions.map(({ handler, noDismiss, ...item }) => ({
flat: true,
...item,
onClick: typeof handler === 'function'
? () => {
handler();
noDismiss !== true && dismiss();
}
: () => { dismiss(); }
}));
if (notif.multiLine === void 0) {
notif.multiLine = notif.actions.length > 1;
}
Object.assign(notif.meta, {
class: 'q-notification row items-stretch'
+ ` q-notification--${ notif.multiLine === true ? 'multi-line' : 'standard' }`
+ (notif.color !== void 0 ? ` bg-${ notif.color }` : '')
+ (notif.textColor !== void 0 ? ` text-${ notif.textColor }` : '')
+ (notif.classes !== void 0 ? ` ${ notif.classes }` : ''),
wrapperClass: 'q-notification__wrapper col relative-position border-radius-inherit '
+ (notif.multiLine === true ? 'column no-wrap justify-center' : 'row items-center'),
contentClass: 'q-notification__content row items-center'
+ (notif.multiLine === true ? '' : ' col'),
leftClass: notif.meta.hasText === true ? 'additional' : 'single',
attrs: {
role: 'alert',
...notif.attrs
}
});
if (notif.group === false) {
notif.group = void 0;
notif.meta.group = void 0;
}
else {
if (notif.group === void 0 || notif.group === true) {
// do not replace notifications with different buttons
notif.group = [
notif.message,
notif.caption,
notif.multiline
].concat(
notif.actions.map(props => `${ props.label }*${ props.icon }`)
).join('|');
}
notif.meta.group = notif.group + '|' + notif.position;
}
if (notif.actions.length === 0) {
notif.actions = void 0;
}
else {
notif.meta.actionsClass = 'q-notification__actions row items-center '
+ (notif.multiLine === true ? 'justify-end' : 'col-auto')
+ (notif.meta.hasMedia === true ? ' q-notification__actions--with-media' : '');
}
if (originalApi !== void 0) {
// reset timeout if any
if (originalApi.notif.meta.timer) {
clearTimeout(originalApi.notif.meta.timer);
originalApi.notif.meta.timer = void 0;
}
// retain uid
notif.meta.uid = originalApi.notif.meta.uid;
// replace notif
const index = notificationsList[ notif.position ].value.indexOf(originalApi.notif);
notificationsList[ notif.position ].value[ index ] = notif;
}
else {
const original = groups[ notif.meta.group ];
// woohoo, it's a new notification
if (original === void 0) {
notif.meta.uid = uid++;
notif.meta.badge = 1;
if ([ 'left', 'right', 'center' ].indexOf(notif.position) !== -1) {
notificationsList[ notif.position ].value.splice(
Math.floor(notificationsList[ notif.position ].value.length / 2),
0,
notif
);
}
else {
const action = notif.position.indexOf('top') > -1 ? 'unshift' : 'push';
notificationsList[ notif.position ].value[ action ](notif);
}
if (notif.group !== void 0) {
groups[ notif.meta.group ] = notif;
}
}
// ok, so it's NOT a new one
else {
// reset timeout if any
if (original.meta.timer) {
clearTimeout(original.meta.timer);
original.meta.timer = void 0;
}
if (notif.badgePosition !== void 0) {
if (badgePositions.includes(notif.badgePosition) === false) {
return logError('wrong badgePosition', config)
}
}
else {
notif.badgePosition = `top-${ notif.position.indexOf('left') > -1 ? 'right' : 'left' }`;
}
notif.meta.uid = original.meta.uid;
notif.meta.badge = original.meta.badge + 1;
notif.meta.badgeClass = `q-notification__badge q-notification__badge--${ notif.badgePosition }`
+ (notif.badgeColor !== void 0 ? ` bg-${ notif.badgeColor }` : '')
+ (notif.badgeTextColor !== void 0 ? ` text-${ notif.badgeTextColor }` : '')
+ (notif.badgeClass ? ` ${ notif.badgeClass }` : '');
const index = notificationsList[ notif.position ].value.indexOf(original);
notificationsList[ notif.position ].value[ index ] = groups[ notif.meta.group ] = notif;
}
}
const dismiss = () => {
removeNotification(notif);
Api = void 0;
};
if (notif.timeout > 0) {
notif.meta.timer = setTimeout(() => {
notif.meta.timer = void 0;
dismiss();
}, notif.timeout + /* show duration */ 1000);
}
// only non-groupable can be updated
if (notif.group !== void 0) {
return props => {
if (props !== void 0) {
logError('trying to update a grouped one which is forbidden', config);
}
else {
dismiss();
}
}
}
Api = {
dismiss,
config,
notif
};
if (originalApi !== void 0) {
Object.assign(originalApi, Api);
return
}
return props => {
// if notification wasn't previously dismissed
if (Api !== void 0) {
// if no params, then we must dismiss the notification
if (props === void 0) {
Api.dismiss();
}
// otherwise we're updating it
else {
const newNotif = Object.assign({}, Api.config, props, {
group: false,
position: notif.position
});
addNotification(newNotif, $q, Api);
}
}
}
}
function removeNotification (notif) {
if (notif.meta.timer) {
clearTimeout(notif.meta.timer);
notif.meta.timer = void 0;
}
const index = notificationsList[ notif.position ].value.indexOf(notif);
if (index !== -1) {
if (notif.group !== void 0) {
delete groups[ notif.meta.group ];
}
const el = notifRefs[ '' + notif.meta.uid ];
if (el) {
const { width, height } = getComputedStyle(el);
el.style.left = `${ el.offsetLeft }px`;
el.style.width = width;
el.style.height = height;
}
notificationsList[ notif.position ].value.splice(index, 1);
if (typeof notif.onDismiss === 'function') {
notif.onDismiss();
}
}
}
function hasContent (str) {
return str !== void 0
&& str !== null
&& emptyRE.test(str) !== true
}
function logError (error, config) {
console.error(`Notify: ${ error }`, config);
return false
}
function getComponent () {
return createComponent({
name: 'QNotifications',
// hide App from Vue devtools
devtools: { hide: true },
setup () {
return () => vue.h('div', { class: 'q-notifications' }, positionList.map(pos => {
return vue.h(vue.TransitionGroup, {
key: pos,
class: positionClass[ pos ],
tag: 'div',
name: `q-notification--${ pos }`
}, () => notificationsList[ pos ].value.map(notif => {
const meta = notif.meta;
const mainChild = [];
if (meta.hasMedia === true) {
if (notif.spinner !== false) {
mainChild.push(
vue.h(notif.spinner, {
class: 'q-notification__spinner q-notification__spinner--' + meta.leftClass,
color: notif.spinnerColor,
size: notif.spinnerSize
})
);
}
else if (notif.icon) {
mainChild.push(
vue.h(QIcon, {
class: 'q-notification__icon q-notification__icon--' + meta.leftClass,
name: notif.icon,
color: notif.iconColor,
size: notif.iconSize,
role: 'img'
})
);
}
else if (notif.avatar) {
mainChild.push(
vue.h(QAvatar, {
class: 'q-notification__avatar q-notification__avatar--' + meta.leftClass
}, () => vue.h('img', { src: notif.avatar, 'aria-hidden': 'true' }))
);
}
}
if (meta.hasText === true) {
let msgChild;
const msgData = { class: 'q-notification__message col' };
if (notif.html === true) {
msgData.innerHTML = notif.caption
? `<div>${ notif.message }</div><div class="q-notification__caption">${ notif.caption }</div>`
: notif.message;
}
else {
const msgNode = [ notif.message ];
msgChild = notif.caption
? [
vue.h('div', msgNode),
vue.h('div', { class: 'q-notification__caption' }, [ notif.caption ])
]
: msgNode;
}
mainChild.push(
vue.h('div', msgData, msgChild)
);
}
const child = [
vue.h('div', { class: meta.contentClass }, mainChild)
];
notif.progress === true && child.push(
vue.h('div', {
key: `${ meta.uid }|p|${ meta.badge }`,
class: meta.progressClass,
style: meta.progressStyle
})
);
notif.actions !== void 0 && child.push(
vue.h('div', {
class: meta.actionsClass
}, notif.actions.map(props => vue.h(QBtn, props)))
);
meta.badge > 1 && child.push(
vue.h('div', {
key: `${ meta.uid }|${ meta.badge }`,
class: notif.meta.badgeClass,
style: notif.badgeStyle
}, [ meta.badge ])
);
return vue.h('div', {
ref: el => { notifRefs[ '' + meta.uid ] = el; },
key: meta.uid,
class: meta.class,
...meta.attrs
}, [
vue.h('div', { class: meta.wrapperClass }, child)
])
}))
}))
}
})
}
var Notify = {
setDefaults (opts) {
{
isObject(opts) === true && Object.assign(defaults, opts);
}
},
registerType (typeName, typeOpts) {
if (isObject(typeOpts) === true) {
notifTypes[ typeName ] = typeOpts;
}
},
install ({ $q, parentApp }) {
$q.notify = this.create = opts => addNotification(opts, $q);
$q.notify.setDefaults = this.setDefaults;
$q.notify.registerType = this.registerType;
if ($q.config.notify !== void 0) {
this.setDefaults($q.config.notify);
}
if (this.__installed !== true) {
positionList.forEach(pos => {
notificationsList[ pos ] = vue.ref([]);
const
vert = [ 'left', 'center', 'right' ].includes(pos) === true ? 'center' : (pos.indexOf('top') > -1 ? 'top' : 'bottom'),
align = pos.indexOf('left') > -1 ? 'start' : (pos.indexOf('right') > -1 ? 'end' : 'center'),
classes = [ 'left', 'right' ].includes(pos) ? `items-${ pos === 'left' ? 'start' : 'end' } justify-center` : (pos === 'center' ? 'flex-center' : `items-${ align }`);
positionClass[ pos ] = `q-notifications__list q-notifications__list--${ vert } fixed column no-wrap ${ classes }`;
});
const el = createGlobalNode('q-notify');
createChildApp(getComponent(), parentApp).mount(el);
}
}
};
function encode (value) {
if (isDate(value) === true) {
return '__q_date|' + value.toUTCString()
}
if (isRegexp(value) === true) {
return '__q_expr|' + value.source
}
if (typeof value === 'number') {
return '__q_numb|' + value
}
if (typeof value === 'boolean') {
return '__q_bool|' + (value ? '1' : '0')
}
if (typeof value === 'string') {
return '__q_strn|' + value
}
if (typeof value === 'function') {
return '__q_strn|' + value.toString()
}
if (value === Object(value)) {
return '__q_objt|' + JSON.stringify(value)
}
// hmm, we don't know what to do with it,
// so just return it as is
return value
}
function decode (value) {
const length = value.length;
if (length < 9) {
// then it wasn't encoded by us
return value
}
const type = value.substring(0, 8);
const source = value.substring(9);
switch (type) {
case '__q_date':
return new Date(source)
case '__q_expr':
return new RegExp(source)
case '__q_numb':
return Number(source)
case '__q_bool':
return Boolean(source === '1')
case '__q_strn':
return '' + source
case '__q_objt':
return JSON.parse(source)
default:
// hmm, we reached here, we don't know the type,
// then it means it wasn't encoded by us, so just
// return whatever value it is
return value
}
}
function getEmptyStorage () {
const getVal = () => null;
return {
has: () => false,
getLength: () => 0,
getItem: getVal,
getIndex: getVal,
getKey: getVal,
getAll: () => {},
getAllKeys: () => [],
set: noop,
remove: noop,
clear: noop,
isEmpty: () => true
}
}
function getStorage (type) {
const
webStorage = window[ type + 'Storage' ],
get = key => {
const item = webStorage.getItem(key);
return item
? decode(item)
: null
};
return {
has: key => webStorage.getItem(key) !== null,
getLength: () => webStorage.length,
getItem: get,
getIndex: index => {
return index < webStorage.length
? get(webStorage.key(index))
: null
},
getKey: index => {
return index < webStorage.length
? webStorage.key(index)
: null
},
getAll: () => {
let key;
const result = {}, len = webStorage.length;
for (let i = 0; i < len; i++) {
key = webStorage.key(i);
result[ key ] = get(key);
}
return result
},
getAllKeys: () => {
const result = [], len = webStorage.length;
for (let i = 0; i < len; i++) {
result.push(webStorage.key(i));
}
return result
},
set: (key, value) => { webStorage.setItem(key, encode(value)); },
remove: key => { webStorage.removeItem(key); },
clear: () => { webStorage.clear(); },
isEmpty: () => webStorage.length === 0
}
}
const storage$1 = client.has.webStorage === false
? getEmptyStorage()
: getStorage('local');
const Plugin$1 = {
install ({ $q }) {
$q.localStorage = storage$1;
}
};
Object.assign(Plugin$1, storage$1);
const storage = client.has.webStorage === false
? getEmptyStorage()
: getStorage('session');
const Plugin = {
install ({ $q }) {
$q.sessionStorage = storage;
}
};
Object.assign(Plugin, storage);
var plugins = /*#__PURE__*/Object.freeze({
__proto__: null,
AddressbarColor: AddressbarColor,
AppFullscreen: Plugin$6,
AppVisibility: Plugin$5,
BottomSheet: BottomSheet,
Cookies: Plugin$4,
Dark: Plugin$9,
Dialog: Dialog,
LoadingBar: Plugin$3,
Loading: Plugin$2,
Meta: Meta,
Notify: Notify,
Platform: Platform,
Screen: Screen,
LocalStorage: Plugin$1,
SessionStorage: Plugin
});
function fallback (text) {
const area = document.createElement('textarea');
area.value = text;
area.contentEditable = 'true';
area.style.position = 'fixed'; // avoid scrolling to bottom
const fn = () => {};
addFocusout(fn);
document.body.appendChild(area);
area.focus();
area.select();
const res = document.execCommand('copy');
area.remove();
removeFocusout(fn);
return res
}
function copyToClipboard (text) {
return navigator.clipboard !== void 0
? navigator.clipboard.writeText(text)
: new Promise((resolve, reject) => {
const res = fallback(text);
if (res) {
resolve(true);
}
else {
reject(res);
}
})
}
var createMetaMixin = metaOptions => {
const mixin = {
activated () {
this.__qMeta.active = true;
planClientUpdate();
},
deactivated () {
this.__qMeta.active = false;
planClientUpdate();
},
unmounted () {
clientList.splice(clientList.indexOf(this.__qMeta), 1);
planClientUpdate();
this.__qMeta = void 0;
}
};
if (typeof metaOptions === 'function') {
Object.assign(mixin, {
computed: {
__qMetaOptions () {
return metaOptions.call(this) || {}
}
},
watch: {
__qMetaOptions (val) {
this.__qMeta.val = val;
this.__qMeta.active === true && planClientUpdate();
}
},
created () {
this.__qMeta = { active: true, val: this.__qMetaOptions };
clientList.push(this.__qMeta);
planClientUpdate();
}
});
}
else {
mixin.created = function () {
this.__qMeta = { active: true, val: metaOptions };
clientList.push(this.__qMeta);
planClientUpdate();
};
}
return mixin
};
/**
* Forked from tiny-emitter
* Copyright (c) 2017 Scott Corgan
*/
class EventBus {
constructor () {
this.__stack = {};
}
on (name, callback, ctx) {
(this.__stack[ name ] || (this.__stack[ name ] = [])).push({
fn: callback,
ctx
});
return this // chainable
}
once (name, callback, ctx) {
const listener = () => {
this.off(name, listener);
callback.apply(ctx, arguments);
};
listener.__callback = callback;
return this.on(name, listener, ctx) // chainable
}
emit (name) {
const list = this.__stack[ name ];
if (list !== void 0) {
const params = [].slice.call(arguments, 1);
list.forEach(entry => {
entry.fn.apply(entry.ctx, params);
});
}
return this // chainable
}
off (name, callback) {
const list = this.__stack[ name ];
if (list === void 0) {
return this // chainable
}
if (callback === void 0) {
delete this.__stack[ name ];
return this // chainable
}
const liveEvents = list.filter(
entry => entry.fn !== callback && entry.fn.__callback !== callback
);
if (liveEvents.length !== 0) {
this.__stack[ name ] = liveEvents;
}
else {
delete this.__stack[ name ];
}
return this // chainable
}
}
function clean (link) {
// allow time for iOS
setTimeout(() => {
window.URL.revokeObjectURL(link.href);
}, 10000);
link.remove();
}
/**
* Forces browser to download file with specified content
*
* @param {*} fileName - String
* @param {*} rawData - String | ArrayBuffer | ArrayBufferView | Blob
* @param {*} opts - String (mimeType) or Object
* Object form: { mimeType?: String, byteOrderMark?: String | Uint8Array, encoding?: String }
* @returns Boolean | Error
*
* mimeType - Examples: 'application/octect-stream' (default), 'text/plain', 'application/json',
* 'text/plain;charset=UTF-8', 'video/mp4', 'image/png', 'application/pdf'
* https://developer.mozilla.org/en-US/docs/Web/HTTP/Basics_of_HTTP/MIME_types
*
* byteOrderMark - (BOM) Example: '\uFEFF'
* https://en.wikipedia.org/wiki/Byte_order_mark
*
* encoding - Performs a TextEncoder.encode() over the rawData;
* Example: 'windows-1252' (ANSI, a subset of ISO-8859-1)
* https://developer.mozilla.org/en-US/docs/Web/API/TextEncoder
*/
function exportFile (fileName, rawData, opts = {}) {
const { mimeType, byteOrderMark, encoding } = typeof opts === 'string'
? { mimeType: opts }
: opts;
const data = encoding !== void 0
? (new TextEncoder(encoding)).encode([ rawData ])
: rawData;
const blobData = byteOrderMark !== void 0 ? [ byteOrderMark, data ] : [ data ];
const blob = new Blob(blobData, { type: mimeType || 'application/octet-stream' });
const link = document.createElement('a');
link.href = window.URL.createObjectURL(blob);
link.setAttribute('download', fileName);
// Check for "download" attribute support;
// If not supported, open this in new window
if (typeof link.download === 'undefined') {
link.setAttribute('target', '_blank');
}
link.classList.add('hidden');
link.style.position = 'fixed'; // avoid scrolling to bottom
document.body.appendChild(link);
try {
link.click();
clean(link);
return true
}
catch (err) {
clean(link);
return err
}
}
function parseFeatures (winFeatures) {
const cfg = Object.assign({ noopener: true }, winFeatures);
const feat = [];
for (const key in cfg) {
const value = cfg[ key ];
if (value === true) {
feat.push(key);
}
else if (isNumber(value) || (typeof value === 'string' && value !== '')) {
feat.push(key + '=' + value);
}
}
return feat.join(',')
}
function openWindow (url, reject, windowFeatures) {
let open = window.open;
if (Platform.is.cordova === true) {
if (cordova !== void 0 && cordova.InAppBrowser !== void 0 && cordova.InAppBrowser.open !== void 0) {
open = cordova.InAppBrowser.open;
}
else if (navigator !== void 0 && navigator.app !== void 0) {
return navigator.app.loadUrl(url, {
openExternal: true
})
}
}
const win = open(url, '_blank', parseFeatures(windowFeatures));
if (win) {
Platform.is.desktop && win.focus();
return win
}
else {
reject && reject();
}
}
var openUrl = (url, reject, windowFeatures) => {
if (
Platform.is.ios === true
&& window.SafariViewController !== void 0
) {
window.SafariViewController.isAvailable(available => {
if (available) {
window.SafariViewController.show(
{ url },
noop,
reject
);
}
else {
openWindow(url, reject, windowFeatures);
}
});
return
}
return openWindow(url, reject, windowFeatures)
};
function parsePromises (sequentialPromises) {
const isList = Array.isArray(sequentialPromises);
if (isList === true) {
const totalJobs = sequentialPromises.length;
return {
isList,
totalJobs,
resultAggregator: Array(totalJobs).fill(null)
}
}
const resultKeys = Object.keys(sequentialPromises);
const resultAggregator = {};
resultKeys.forEach(keyName => { resultAggregator[ keyName ] = null; });
return {
isList,
totalJobs: resultKeys.length,
resultAggregator,
resultKeys
}
}
/**
* Run a list of Promises sequentially, optionally on multiple threads.
*
* @param {*} sequentialPromises - Array of Functions or Object with Functions as values
* Array of Function form: [ (resultAggregator: Array) => Promise<any>, ... ]
* Object form: { [key: string]: (resultAggregator: object) => Promise<any>, ... }
* @param {*} opts - Optional options Object
* Object form: { threadsNumber?: number, abortOnFail?: boolean }
* Default: { threadsNumber: 1, abortOnFail: true }
* When configuring threadsNumber AND using http requests, be
* aware of the maximum threads that the hosting browser
* supports (usually 5); any number of threads above that
* won't add any real benefits
* @returns Promise<Array<Object> | Object>
* With opts.abortOnFail set to true (which is default):
* When sequentialPromises param is Array:
* The Promise resolves with an Array of Objects of the following form:
* [ { key: number, status: 'fulfilled', value: any }, ... ]
* The Promise rejects with an Object of the following form:
* { key: number, status: 'rejected', reason: Error, resultAggregator: array }
* When sequentialPromises param is Object:
* The Promise resolves with an Object of the following form:
* { [key: string]: { key: string, status: 'fulfilled', value: any }, ... }
* The Promise rejects with an Object of the following form:
* { key: string, status: 'rejected', reason: Error, resultAggregator: object }
* With opts.abortOnFail set to false:
* The Promise is never rejected (no catch() needed)
* The Promise resolves with:
* An Array of Objects (when sequentialPromises param is also an Array) of the following form:
* [ { key: number, status: 'fulfilled', value: any } | { status: 'rejected', reason: Error }, ... ]
* An Object (when sequentialPromises param is also an Object) of the following form:
* { [key: string]: { key: string, status: 'fulfilled', value: any } | { key: string, status: 'rejected', reason: Error }, ... }
*/
function runSequentialPromises (
sequentialPromises,
{ threadsNumber = 1, abortOnFail = true } = {}
) {
let jobIndex = -1, hasAborted = false;
const { isList, totalJobs, resultAggregator, resultKeys } = parsePromises(sequentialPromises);
const getPromiseThread = () => new Promise((resolve, reject) => {
function runNextPromise () {
const currentJobIndex = ++jobIndex;
if (hasAborted === true || currentJobIndex >= totalJobs) {
resolve();
return
}
const key = isList === true ? currentJobIndex : resultKeys[ currentJobIndex ];
sequentialPromises[ key ](resultAggregator)
.then(value => {
if (hasAborted === true) {
resolve();
return // early exit
}
resultAggregator[ key ] = { key, status: 'fulfilled', value };
// timeout so it doesn't interfere with the .catch() below
setTimeout(runNextPromise);
})
.catch(reason => {
if (hasAborted === true) {
resolve();
return // early exit
}
const result = { key, status: 'rejected', reason };
resultAggregator[ key ] = result;
if (abortOnFail === true) {
hasAborted = true;
reject({ ...result, resultAggregator });
return // early exit
}
// timeout so no interference
setTimeout(runNextPromise);
});
}
runNextPromise();
});
const threads = Array(threadsNumber).fill(getPromiseThread());
return Promise.all(threads).then(() => resultAggregator)
}
var utils = /*#__PURE__*/Object.freeze({
__proto__: null,
clone: cloneDeep,
colors: colors,
copyToClipboard: copyToClipboard,
createMetaMixin: createMetaMixin,
createUploaderComponent: createUploaderComponent,
date: date,
debounce: debounce,
dom: dom,
EventBus: EventBus,
event: event,
exportFile: exportFile,
extend: extend,
format: format,
frameDebounce: frameDebounce,
getCssVar: getCssVar,
noop: noop,
is: is,
morph: morph,
openURL: openUrl,
patterns: patterns,
runSequentialPromises: runSequentialPromises,
scroll: scroll,
setCssVar: setCssVar,
throttle: throttle,
uid: uid$3
});
// To be used for the custom component
// used on a Dialog plugin
function useDialogPluginComponent () {
const { emit, proxy } = vue.getCurrentInstance();
// we need a Vue reference to the QDialog
// component so we can handle it;
// <q-dialog ref="dialogRef" ...
// make sure that the setup() in which this
// function is called returns dialogRef variable
const dialogRef = vue.ref(null);
function show () { dialogRef.value.show(); }
function hide () { dialogRef.value.hide(); }
function onDialogOK (payload) {
emit('ok', payload);
hide();
}
function onDialogHide () { emit('hide'); }
// expose public methods required by Dialog plugin
Object.assign(proxy, { show, hide });
return {
dialogRef,
onDialogHide,
onDialogOK,
onDialogCancel: hide
}
}
// Don't forget to update the types in "ui/types/composables.d.ts"
const emits = [ 'ok', 'hide' ];
useDialogPluginComponent.emits = emits;
useDialogPluginComponent.emitsObject = getEmitsObject(emits);
function useMeta (metaOptions) {
{
const meta = { active: true };
if (typeof metaOptions === 'function') {
const content = vue.computed(metaOptions);
meta.val = content.value;
vue.watch(content, val => {
meta.val = val;
meta.active === true && planClientUpdate();
});
}
else {
meta.val = metaOptions;
}
clientList.push(meta);
planClientUpdate();
vue.onActivated(() => {
meta.active = true;
planClientUpdate();
});
vue.onDeactivated(() => {
meta.active = false;
planClientUpdate();
});
vue.onUnmounted(() => {
clientList.splice(clientList.indexOf(meta), 1);
planClientUpdate();
});
}
}
/**
* Returns the $q instance.
* Equivalent to `this.$q` inside templates.
*/
function useQuasar () {
return vue.inject(quasarKey)
}
var composables = /*#__PURE__*/Object.freeze({
__proto__: null,
useDialogPluginComponent: useDialogPluginComponent,
useFormChild: useFormChild,
useMeta: useMeta,
useQuasar: useQuasar
});
/**
* UMD entry-point
*/
var index_umd = {
version: '2.12.2',
install (app, opts) {
installQuasar(app, {
components,
directives,
plugins,
...opts
});
},
lang: Plugin$8,
iconSet: Plugin$7,
...components,
...directives,
...plugins,
...composables,
...utils
};
return index_umd;
}));