{"version":3,"file":"index-8125cb2d.js","sources":["../../../node_modules/vue-router/dist/vue-router.esm.js","../../../node_modules/vue-class-component/dist/vue-class-component.esm.js","../../../node_modules/vue-property-decorator/lib/decorators/Emit.js","../../../node_modules/vue-property-decorator/lib/decorators/Inject.js","../../../node_modules/vue-property-decorator/lib/helpers/provideInject.js","../../../node_modules/vue-property-decorator/lib/helpers/metadata.js","../../../node_modules/vue-property-decorator/lib/decorators/Prop.js","../../../node_modules/vue-property-decorator/lib/decorators/PropSync.js","../../../node_modules/vue-property-decorator/lib/decorators/Provide.js","../../../node_modules/vue-property-decorator/lib/decorators/Watch.js","../../../node_modules/mitt/dist/mitt.mjs","../../../src/eventbus.ts","../../../node_modules/gsap/gsap-core.js","../../../node_modules/gsap/CSSPlugin.js","../../../node_modules/gsap/index.js","../../../core/logger.ts","../../../core/utils/isNonEmptyString.ts","../../../src/components/widgets/TransformOptions.ts","../../../src/util/browser.ts","../../../src/utils.ts","../../../src/components/LoadingAnimation.vue","../../../node_modules/vue-demi/lib/index.mjs","../../../node_modules/@vue/devtools-api/lib/esm/env.js","../../../node_modules/@vue/devtools-api/lib/esm/const.js","../../../node_modules/@vue/devtools-api/lib/esm/time.js","../../../node_modules/@vue/devtools-api/lib/esm/proxy.js","../../../node_modules/@vue/devtools-api/lib/esm/index.js","../../../node_modules/pinia/dist/pinia.mjs","../../../src/stores/AppState.ts","../../../src/types/data.ts","../../../src/constants.ts","../../../node_modules/axios/lib/helpers/bind.js","../../../node_modules/axios/lib/utils.js","../../../node_modules/axios/lib/helpers/buildURL.js","../../../node_modules/axios/lib/core/InterceptorManager.js","../../../node_modules/axios/lib/helpers/normalizeHeaderName.js","../../../node_modules/axios/lib/core/AxiosError.js","../../../node_modules/axios/lib/defaults/transitional.js","../../../node_modules/axios/lib/helpers/toFormData.js","../../../node_modules/axios/lib/core/settle.js","../../../node_modules/axios/lib/helpers/cookies.js","../../../node_modules/axios/lib/helpers/isAbsoluteURL.js","../../../node_modules/axios/lib/helpers/combineURLs.js","../../../node_modules/axios/lib/core/buildFullPath.js","../../../node_modules/axios/lib/helpers/parseHeaders.js","../../../node_modules/axios/lib/helpers/isURLSameOrigin.js","../../../node_modules/axios/lib/cancel/CanceledError.js","../../../node_modules/axios/lib/helpers/parseProtocol.js","../../../node_modules/axios/lib/adapters/xhr.js","../../../node_modules/axios/lib/helpers/null.js","../../../node_modules/axios/lib/defaults/index.js","../../../node_modules/axios/lib/core/transformData.js","../../../node_modules/axios/lib/cancel/isCancel.js","../../../node_modules/axios/lib/core/dispatchRequest.js","../../../node_modules/axios/lib/core/mergeConfig.js","../../../node_modules/axios/lib/env/data.js","../../../node_modules/axios/lib/helpers/validator.js","../../../node_modules/axios/lib/core/Axios.js","../../../node_modules/axios/lib/cancel/CancelToken.js","../../../node_modules/axios/lib/helpers/spread.js","../../../node_modules/axios/lib/helpers/isAxiosError.js","../../../node_modules/axios/lib/axios.js","../../../node_modules/axios/index.js","../../../src/appSettings.ts","../../../__vite-browser-external","../../../node_modules/crypto-js/core.js","../../../node_modules/crypto-js/sha256.js","../../../node_modules/crypto-js/enc-base64.js","../../../node_modules/crypto-js/enc-utf8.js","../../../node_modules/jwt-decode/build/jwt-decode.esm.js","../../../node_modules/oidc-client-ts/dist/esm/oidc-client-ts.js","../../../src/api/authService.ts","../../../src/api/backend.ts","../../../src/stores/conditionGroups.ts","../../../src/util/conditionsUtils.ts","../../../src/components/widgets/ComponentOptions.ts","../../../node_modules/vue-i18n/dist/vue-i18n.esm.js","../../../node_modules/vue-i18n-composable/dist/index.mjs","../../../src/locales/dateTimeFormats.ts","../../../src/i18n.ts","../../../src/lib/free-transform/lib/point-finder.js","../../../src/lib/free-transform/lib/scale.js","../../../src/lib/free-transform/lib/rotate.js","../../../src/lib/free-transform/lib/translate.js","../../../node_modules/transformation-matrix/build-es/utils.js","../../../node_modules/transformation-matrix/build-es/translate.js","../../../node_modules/transformation-matrix/build-es/transform.js","../../../node_modules/transformation-matrix/build-es/rotate.js","../../../node_modules/transformation-matrix/build-es/scale.js","../../../node_modules/transformation-matrix/build-es/toString.js","../../../src/lib/free-transform/lib/styler.js","../../../src/rendererState.ts","../../../src/components/widgets/Animation.ts","../../../src/components/widgets/Group/GroupOptions.ts","../../../core/utils/isObjectLiteral.ts","../../../core/utils/isEmpty.ts","../../../src/cleanModel.ts","../../../src/migrations.ts","../../../src/stores/appData.ts","../../../src/stores/connections.ts","../../../src/components/data/dataNodeComponents.ts","../../../core/utils/uuid.ts","../../../src/util/dataUtils.ts","../../../src/stores/editableData.ts","../../../src/stores/connectionData.ts","../../../src/stores/connectionEditor.ts","../../../src/components/widgets/BorderOptions.ts","../../../src/components/widgets/BackgroundOptions.ts","../../../src/components/widgets/CyclingOptions.ts","../../../src/components/widgets/Repeater/RepeaterOptions.ts","../../../src/text/index.ts","../../../src/components/widgets/TextStyleOptions.ts","../../../src/components/widgets/ShadowOptions.ts","../../../src/components/widgets/Text/TextOptions.ts","../../../src/textfit.ts","../../../src/text/binding.ts","../../../src/components/widgets/BlurOptions.ts","../../../src/components/widgets/Ellipse/EllipseOptions.ts","../../../src/components/widgets/Image/ImageOptions.ts","../../../src/components/widgets/Rectangle/RectangleDefinition.ts","../../../src/components/widgets/Multiframe/MultiframeOptions.ts","../../../src/components/widgets/Triangle/TriangleOptions.ts","../../../src/components/widgets/Line/LineOptions.ts","../../../src/components/widgets/AnalogClock/AnalogClockOptions.ts","../../../src/components/widgets/PieGraph/PieGraphOptions.ts","../../../src/components/widgets/LineGraph/LineGraphOptions.ts","../../../src/components/widgets/BarGraph/BarGraphOptions.ts","../../../src/components/widgets/ColumnGraph/ColumnGraphOptions.ts","../../../src/components/widgets/ProgressDonut/ProgressDonutOptions.ts","../../../src/components/widgets/ProgressBar/ProgressBarOptions.ts","../../../src/components/widgets/CalendarRoomSchedule/CalendarRoomScheduleOptions.ts","../../../src/components/widgets/CalendarAgenda/CalendarAgendaOptions.ts","../../../node_modules/luxon/src/errors.js","../../../node_modules/luxon/src/impl/formats.js","../../../node_modules/luxon/src/impl/util.js","../../../node_modules/luxon/src/impl/english.js","../../../node_modules/luxon/src/impl/formatter.js","../../../node_modules/luxon/src/impl/invalid.js","../../../node_modules/luxon/src/zone.js","../../../node_modules/luxon/src/zones/localZone.js","../../../node_modules/luxon/src/zones/IANAZone.js","../../../node_modules/luxon/src/zones/fixedOffsetZone.js","../../../node_modules/luxon/src/zones/invalidZone.js","../../../node_modules/luxon/src/impl/zoneUtil.js","../../../node_modules/luxon/src/settings.js","../../../node_modules/luxon/src/impl/locale.js","../../../node_modules/luxon/src/impl/regexParser.js","../../../node_modules/luxon/src/duration.js","../../../node_modules/luxon/src/interval.js","../../../node_modules/luxon/src/info.js","../../../node_modules/luxon/src/impl/diff.js","../../../node_modules/luxon/src/impl/digits.js","../../../node_modules/luxon/src/impl/tokenParser.js","../../../node_modules/luxon/src/impl/conversions.js","../../../node_modules/luxon/src/datetime.js","../../../src/components/widgets/Countdown/CountdownOptions.ts","../../../src/components/widgets/CalendarWeek/CalendarWeekOptions.ts","../../../src/components/widgets/CalendarDay/CalendarDayOptions.ts","../../../src/components/widgets/CalendarEvent/CalendarEventOptions.ts","../../../src/components/widgets/Datetime/DatetimeOptions.ts","../../../src/components/widgets/Svg/SvgOptions.ts","../../../src/components/widgets/StackedGraph/StackedGraphOptions.ts","../../../src/components/widgets/getWidgetPropertyDefault.ts","../../../src/components/widgets/BarGraph/BarGraphPlaceholderData.ts","../../../src/components/widgets/ColumnGraph/ColumnGraphPlaceholderData.ts","../../../src/components/widgets/LineGraph/LineGraphPlaceholderData.ts","../../../src/components/widgets/PieGraph/PieGraphPlaceholderData.ts","../../../src/components/widgets/StackedGraph/StackedGraphPlaceholderData.ts","../../../src/types/calendar.ts","../../../src/components/widgets/CalendarPlaceholderData.ts","../../../src/placeholderData.ts","../../../src/stores/widgetData.ts","../../../src/stores/gatherAssets.ts","../../../src/stores/dragDrop.ts","../../../src/stores/dynamicImages.ts","../../../src/components/widgets/Video/VideoOptions.ts","../../../src/stores/mainApp.ts","../../../src/stores/appEditor.ts","../../../src/App.vue","../../../src/components/Artboard.vue","../../../src/components/CanvasRender.vue","../../../src/keycodes.ts","../../../src/components/WidgetDragger.vue","../../../src/components/ui/icons/PencilIcon.vue","../../../src/components/ui/icons/TrashIcon.vue","../../../src/components/ui/icons/DuplicateIcon.vue","../../../src/components/Tooltip.vue","../../../src/components/WidgetUIControls.vue","../../../src/findWidget.ts","../../../src/components/WidgetResizer.vue","../../../src/components/BoundingBox.vue","../../../src/components/icons/IconSolid.vue","../../../src/components/ToastMessage.vue","../../../src/event-utilities.ts","../../../src/components/CanvasEditor.vue","../../../src/components/TextDataTokenMenu.vue","../../../src/text/DataToken.ts","../../../src/text/schema.ts","../../../src/components/CanvasTextEditor.vue","../../../src/components/CanvasRulers.vue","../../../src/components/icons/IconOpacity.vue","../../../src/components/icons/Icon.vue","../../../src/components/TooltipMixin.ts","../../../src/components/ButtonIcon.vue","../../../src/components/editors/AlignmentEditor.vue","../../../src/components/Icon.vue","../../../src/components/inputs/EditorNumberInput.vue","../../../src/components/ToggleButton.vue","../../../src/components/DarkToggleButton.vue","../../../src/components/inputs/ToggleInput.vue","../../../src/components/editors/DimensionsEditor.vue","../../../src/components/FormLabel.vue","../../../src/components/inputs/SliderInput.vue","../../../src/components/inputs/NumberSliderInput.vue","../../../src/components/ButtonGradient.vue","../../../src/components/editors/RotationEditor.vue","../../../src/components/icons/LightningBoltIcon.vue","../../../src/components/data/LightningBolt.vue","../../../src/components/editors/ChildBindingDropdown.vue","../../../src/components/editors/ScalarBinder.vue","../../../src/components/editors/OpacityEditor.vue","../../../src/components/BaseWidgetEditor.vue","../../../node_modules/picocolors/picocolors.browser.js","../../../node_modules/tailwindcss/lib/util/log.js","../../../node_modules/tailwindcss/lib/public/colors.js","../../../node_modules/tailwindcss/colors.js","../../../src/components/colors.ts","../../../node_modules/tinycolor2/tinycolor.js","../../../src/components/inputs/NumberInput.vue","../../../src/components/inputs/GradientSliderInput.vue","../../../src/components/inputs/ColorPicker.vue","../../../src/components/ColorPanelPalette.vue","../../../src/components/CollapsePanel.vue","../../../src/components/ColorPanel.vue","../../../src/components/icons/animationIcons/AnimationIcon.vue","../../../src/components/AnimationPresetTile.vue","../../../src/components/inputs/NumberInputBar.vue","../../../src/components/icons/animationIcons/EaseIcon.vue","../../../src/components/EditorSelectMenu.vue","../../../src/components/AnimationPreviewButton.vue","../../../src/components/editors/AnimationPropertyEditor.vue","../../../src/components/AnimationPresetRow.vue","../../../src/components/editors/AnimationEditor.vue","../../../src/components/widgets/Group/GroupEditor.vue","../../../src/components/editors/LockEditor.vue","../../../src/components/editors/MultiComponentEditor.vue","../../../src/components/ComponentEditorPanel.vue","../../../src/components/inputs/ColorInput.vue","../../../src/components/inputs/TextInput.vue","../../../src/components/SelectMenu.vue","../../../src/components/editors/BorderEditor.vue","../../../node_modules/sortablejs/modular/sortable.esm.js","../../../node_modules/vuedraggable/dist/vuedraggable.umd.js","../../../src/components/HelpBubble.vue","../../../src/components/Modal.vue","../../../src/components/OutlineButton.vue","../../../src/components/ActionModal.vue","../../../src/assets/conditions-illustration.png","../../../src/assets/conditions-video-thumbnail.png","../../../src/userPreferences.ts","../../../src/components/editors/ConditionsEditorHelpBubbles.vue","../../../src/components/editors/ConditionButton.vue","../../../src/components/editors/ConditionsEditor.vue","../../../src/components/inputs/ColorInputBar.vue","../../../src/components/editors/ShadowEditor.vue","../../../src/components/widgets/animationAllowed.ts","../../../src/components/widgets/Svg/SvgEditor.vue","../../../src/components/editors/BlurEditor.vue","../../../src/components/widgets/Ellipse/EllipseEditor.vue","../../../src/components/widgets/Rectangle/RectangleEditor.vue","../../../src/components/editors/EditorModuleSection.vue","../../../src/components/inputs/EditorRadioInput.vue","../../../src/fonts/google-fonts-no-italics.ts","../../../node_modules/fontfaceobserver/fontfaceobserver.standalone.js","../../../src/fonts/loadFonts.ts","../../../src/fonts/searchFonts.ts","../../../src/fonts/fontOptions.ts","../../../src/components/inputs/FontFamilyInput.vue","../../../src/components/inputs/FontSizeInput.vue","../../../src/components/ComponentEditorButton.vue","../../../src/components/inputs/TextTransformInput.vue","../../../src/fonts/fontHelpers.ts","../../../src/components/inputs/FontVariantInput.vue","../../../src/components/widgets/Text/TextEditor.vue","../../../src/components/icons/LightningBoltIconWhite.vue","../../../src/components/logic/ruleValidator.ts","../../../src/stores/columnInfo.ts","../../../src/stores/filterEditor.ts","../../../src/components/logic/operators.ts","../../../src/components/logic/OperandSelectType.ts","../../../src/components/logic/OperandTypeIcon.vue","../../../src/components/logic/SelectColumnValue.vue","../../../src/stores/conditionEditor.ts","../../../src/components/FormButton.vue","../../../src/components/UiBlocker.vue","../../../src/components/data/DataNodeDateTime.vue","../../../src/components/data/DataNodeValue.vue","../../../src/components/data/DataNodeImage.vue","../../../src/components/data/DataNodeColor.vue","../../../src/components/data/DataNodeEmpty.vue","../../../src/components/data/DataNodeDisplay.vue","../../../src/components/data/DraggableNode.vue","../../../src/components/data/connections/setup/schema/TreeNode.vue","../../../src/components/data/connections/setup/selection/SelectTreeValue.vue","../../../src/components/data/connections/setup/selection/SelectTableValue.vue","../../../src/components/logic/SelectScalarNode.vue","../../../src/components/logic/SelectScalarValue.vue","../../../src/components/logic/SelectDateTimeValue.vue","../../../src/components/logic/StaticInput.vue","../../../src/components/logic/SelectQueryStringValue.vue","../../../src/components/logic/SelectStaticValue.vue","../../../src/components/logic/CheckBox.vue","../../../src/components/logic/OperandEditor.vue","../../../src/components/logic/describeRule.ts","../../../src/components/data/connections/filters/LogicRuleEditor.vue","../../../src/components/data/connections/filters/RuleGroupEditor.vue","../../../src/components/data/connections/filters/FilterModal.vue","../../../src/components/data/connections/filters/FilterManager.vue","../../../src/components/widgets/CalendarDataChooser.vue","../../../src/components/editors/GroupTextFormatEditor.vue","../../../src/components/inputs/VisibilityInput.vue","../../../src/components/widgets/CalendarEvent/CalendarEventEditor.vue","../../../src/components/inputs/DoubleSliderInput.vue","../../../src/components/widgets/CalendarDay/CalendarDayEditor.vue","../../../src/components/widgets/CalendarWeek/CalendarWeekEditor.vue","../../../src/components/widgets/CalendarAgenda/CalendarAgendaEditor.vue","../../../src/components/widgets/CalendarRoomSchedule/CalendarRoomScheduleEditor.vue","../../../src/components/widgets/Countdown/CountdownEditor.vue","../../../src/components/widgets/ProgressBar/ProgressBarEditor.vue","../../../src/components/widgets/ProgressDonut/ProgressDonutEditor.vue","../../../src/components/inputs/EditorTextInput.vue","../../../src/components/widgets/DataSetChooser.vue","../../../src/components/widgets/ColumnGraph/ColumnGraphEditor.vue","../../../src/components/widgets/BarGraph/BarGraphEditor.vue","../../../src/components/widgets/LineGraph/LineGraphEditor.vue","../../../src/components/widgets/PieGraph/PieGraphEditor.vue","../../../src/components/widgets/StackedGraph/StackedGraphEditor.vue","../../../src/components/inputs/RadioInput.vue","../../../src/assets/repeat-grid-flow-row.svg","../../../src/assets/repeat-grid-flow-row-reverse.svg","../../../src/assets/repeat-grid-flow-column.svg","../../../src/assets/repeat-grid-flow-column-reverse.svg","../../../src/components/widgets/Repeater/RepeaterEditor.vue","../../../src/components/widgets/DataColumnChooser.vue","../../../src/components/widgets/Image/ImageEditor.vue","../../../src/datetimeFormats.ts","../../../src/datetimeUtils.ts","../../../src/components/inputs/DatetimeFormatTextInput.vue","../../../src/components/inputs/DatetimeFormatSelect.vue","../../../src/components/inputs/DatetimeFormatInput.vue","../../../src/components/TimeZoneSelect.vue","../../../src/components/widgets/BaseWidget.vue","../../../src/components/widgets/AnimatableWidget.vue","../../../src/components/widgets/Text/TextComponent.vue","../../../src/components/widgets/Datetime/DatetimeComponent.vue","../../../src/components/widgets/Datetime/DatetimeEditor.vue","../../../src/components/widgets/AnalogClock/AnalogClockEditor.vue","../../../src/components/widgets/Triangle/TriangleEditor.vue","../../../src/components/widgets/Line/LineEditor.vue","../../../src/components/widgets/Video/VideoEditor.vue","../../../src/components/ComponentEditor.vue","../../../src/components/Canvas.vue","../../../src/components/InlineInput.vue","../../../src/components/inputs/ToggleInputLabelled.vue","../../../src/stores/apps.ts","../../../src/components/SaveButton.vue","../../../src/components/SubscribeToPublish.vue","../../../src/components/PublishButton.vue","../../../src/util/htmlBranch.ts","../../../src/components/DownloadBundle.vue","../../../src/components/GetAppButton.vue","../../../src/components/LogoButton.vue","../../../src/assets/logo.svg","../../../src/components/AnchorButton.vue","../../../src/UndoRedoMixin.ts","../../../src/components/ZoomControls.vue","../../../src/components/transitions/ExpandTransition.vue","../../../node_modules/@uppy/utils/lib/hasProperty.js","../../../node_modules/@uppy/utils/lib/Translator.js","../../../node_modules/namespace-emitter/index.js","../../../node_modules/cuid/lib/pad.js","../../../node_modules/cuid/lib/fingerprint.browser.js","../../../node_modules/cuid/lib/getRandomValue.browser.js","../../../node_modules/cuid/index.js","../../../node_modules/@transloadit/prettier-bytes/prettierBytes.js","../../../node_modules/wildcard/index.js","../../../node_modules/mime-match/index.js","../../../node_modules/@uppy/store-default/lib/index.js","../../../node_modules/@uppy/utils/lib/getFileNameAndExtension.js","../../../node_modules/@uppy/utils/lib/mimeTypes.js","../../../node_modules/@uppy/utils/lib/getFileType.js","../../../node_modules/@uppy/utils/lib/generateFileID.js","../../../node_modules/@uppy/utils/lib/findIndex.js","../../../node_modules/@uppy/core/lib/supportsUploadProgress.js","../../../node_modules/@uppy/utils/lib/getTimeStamp.js","../../../node_modules/@uppy/core/lib/loggers.js","../../../node_modules/preact/dist/preact.esm.js","../../../node_modules/@uppy/utils/lib/isDOMElement.js","../../../node_modules/@uppy/utils/lib/findDOMElement.js","../../../node_modules/@uppy/core/lib/Plugin.js","../../../node_modules/@uppy/core/lib/index.js","../../../node_modules/classnames/index.js","../../../node_modules/is-shallow-equal/index.js","../../../node_modules/@uppy/dashboard/lib/utils/getFileTypeIcon.js","../../../node_modules/@uppy/dashboard/lib/components/FilePreview.js","../../../node_modules/@uppy/dashboard/lib/components/FileItem/FilePreviewAndLink/index.js","../../../node_modules/@uppy/dashboard/lib/components/FileItem/FileProgress/index.js","../../../node_modules/@uppy/utils/lib/truncateString.js","../../../node_modules/@uppy/dashboard/lib/components/FileItem/FileInfo/index.js","../../../node_modules/@uppy/dashboard/lib/utils/copyToClipboard.js","../../../node_modules/@uppy/dashboard/lib/components/FileItem/Buttons/index.js","../../../node_modules/@uppy/dashboard/lib/components/FileItem/index.js","../../../node_modules/@uppy/dashboard/lib/components/VirtualList.js","../../../node_modules/@uppy/dashboard/lib/components/FileList.js","../../../node_modules/@uppy/dashboard/lib/components/AddFiles.js","../../../node_modules/@uppy/dashboard/lib/components/AddFilesPanel.js","../../../node_modules/@uppy/dashboard/lib/utils/ignoreEvent.js","../../../node_modules/@uppy/dashboard/lib/components/PickerPanelContent.js","../../../node_modules/@uppy/dashboard/lib/components/EditorPanel.js","../../../node_modules/@uppy/dashboard/lib/components/PickerPanelTopBar.js","../../../node_modules/@uppy/dashboard/lib/components/FileCard/index.js","../../../node_modules/@uppy/dashboard/lib/components/Slide.js","../../../node_modules/@uppy/utils/lib/isDragDropSupported.js","../../../node_modules/@uppy/dashboard/lib/components/Dashboard.js","../../../node_modules/@uppy/status-bar/lib/StatusBarStates.js","../../../node_modules/@uppy/utils/lib/secondsToTime.js","../../../node_modules/@uppy/utils/lib/prettyETA.js","../../../node_modules/@uppy/status-bar/lib/StatusBar.js","../../../node_modules/@uppy/utils/lib/getSpeed.js","../../../node_modules/@uppy/utils/lib/getBytesRemaining.js","../../../node_modules/@uppy/utils/lib/getTextDirection.js","../../../node_modules/@uppy/status-bar/lib/index.js","../../../node_modules/@uppy/informer/lib/index.js","../../../node_modules/@uppy/utils/lib/dataURItoBlob.js","../../../node_modules/@uppy/utils/lib/isObjectURL.js","../../../node_modules/@uppy/utils/lib/isPreviewSupported.js","../../../node_modules/math-log2/index.js","../../../node_modules/exifr/dist/mini.legacy.umd.js","../../../node_modules/@uppy/thumbnail-generator/lib/index.js","../../../node_modules/@uppy/utils/lib/findAllDOMElements.js","../../../node_modules/@uppy/utils/lib/toArray.js","../../../node_modules/@uppy/utils/lib/getDroppedFiles/utils/webkitGetAsEntryApi/getRelativePath.js","../../../node_modules/@uppy/utils/lib/getDroppedFiles/utils/webkitGetAsEntryApi/getFilesAndDirectoriesFromDirectory.js","../../../node_modules/@uppy/utils/lib/getDroppedFiles/utils/webkitGetAsEntryApi/index.js","../../../node_modules/@uppy/utils/lib/getDroppedFiles/utils/fallbackApi.js","../../../node_modules/@uppy/utils/lib/getDroppedFiles/index.js","../../../node_modules/@uppy/dashboard/lib/utils/getActiveOverlayEl.js","../../../node_modules/@uppy/utils/lib/FOCUSABLE_ELEMENTS.js","../../../node_modules/@uppy/dashboard/lib/utils/trapFocus.js","../../../node_modules/resize-observer-polyfill/dist/ResizeObserver.es.js","../../../node_modules/@uppy/dashboard/lib/utils/createSuperFocus.js","../../../node_modules/memoize-one/dist/memoize-one.esm.js","../../../node_modules/@uppy/dashboard/lib/index.js","../../../node_modules/@uppy/utils/lib/getFileTypeExtension.js","../../../node_modules/@uppy/utils/lib/canvasToBlob.js","../../../node_modules/@uppy/webcam/lib/supportsMediaRecorder.js","../../../node_modules/@uppy/webcam/lib/CameraIcon.js","../../../node_modules/@uppy/webcam/lib/SnapshotButton.js","../../../node_modules/@uppy/webcam/lib/RecordButton.js","../../../node_modules/@uppy/webcam/lib/formatSeconds.js","../../../node_modules/@uppy/webcam/lib/RecordingLength.js","../../../node_modules/@uppy/webcam/lib/VideoSourceSelect.js","../../../node_modules/@uppy/webcam/lib/SubmitButton.js","../../../node_modules/@uppy/webcam/lib/DiscardButton.js","../../../node_modules/@uppy/webcam/lib/CameraScreen.js","../../../node_modules/@uppy/webcam/lib/PermissionsScreen.js","../../../node_modules/@uppy/webcam/lib/index.js","../../../node_modules/@uppy/companion-client/lib/AuthError.js","../../../node_modules/@uppy/utils/lib/NetworkError.js","../../../node_modules/@uppy/utils/lib/fetchWithNetworkError.js","../../../node_modules/@uppy/companion-client/lib/RequestClient.js","../../../node_modules/qs-stringify/index.js","../../../node_modules/requires-port/index.js","../../../node_modules/querystringify/index.js","../../../node_modules/url-parse/index.js","../../../node_modules/@uppy/companion-client/lib/tokenStorage.js","../../../node_modules/@uppy/companion-client/lib/Provider.js","../../../node_modules/@uppy/companion-client/lib/SearchProvider.js","../../../node_modules/@uppy/companion-client/lib/Socket.js","../../../node_modules/@uppy/companion-client/lib/index.js","../../../node_modules/@uppy/utils/lib/emitSocketProgress.js","../../../node_modules/@uppy/utils/lib/getSocketHost.js","../../../node_modules/@uppy/utils/lib/settle.js","../../../node_modules/@uppy/utils/lib/EventTracker.js","../../../node_modules/@uppy/utils/lib/ProgressTimeout.js","../../../node_modules/@uppy/utils/lib/RateLimitedQueue.js","../../../node_modules/@uppy/utils/lib/isNetworkError.js","../../../node_modules/@uppy/xhr-upload/lib/index.js","../../../src/lib/uppy-custom/xhr-upload-custom/index.js","../../../node_modules/@uppy/url/lib/UrlUI.js","../../../node_modules/@uppy/url/lib/utils/forEachDroppedOrPastedUrl.js","../../../node_modules/@uppy/url/lib/index.js","../../../src/lib/uppy-custom/CustomRequestClient.js","../../../src/lib/uppy-custom/url-custom/index.js","../../../src/uploadAssets.ts","../../../src/components/ImageUpload.vue","../../../src/components/AppSettings.vue","../../../src/components/PreviewLink.vue","../../../src/components/Header.vue","../../../src/stores/user.ts","../../../src/components/UserInfo.vue","../../../src/components/menus/TemplatesMenu.vue","../../../src/components/widgets/Rectangle/RectangleOptions.ts","../../../src/components/icons/componentIcons/ComponentIcon.vue","../../../src/components/menus/ShapesMenu.vue","../../../src/components/menus/data/ConnectionItem.vue","../../../src/components/menus/data/ConnectionsList.vue","../../../src/components/menus/data/LearnMore.vue","../../../src/components/menus/data/ConnectionsEmpty.vue","../../../src/components/menus/data/UsagePrompt.vue","../../../src/components/menus/data/ConnectionHeader.vue","../../../src/components/menus/data/ModerationToggle.vue","../../../src/components/menus/data/RecordIndexControl.vue","../../../src/components/menus/data/DataEmpty.vue","../../../src/stores/assets.ts","../../../src/components/menus/PhotoDragging.vue","../../../src/components/data/DataNodeImageUrl.vue","../../../src/components/menus/data/RecordData.vue","../../../src/components/menus/data/RecordView.vue","../../../src/components/menus/CloseButton.vue","../../../src/components/menus/DataMenu.vue","../../../src/components/menus/PhotosMenu.vue","../../../src/components/VideoPlaceholder.vue","../../../src/components/menus/VideosMenu.vue","../../../src/components/menus/GridsMenu.vue","../../../src/components/menus/WidgetsMenu.vue","../../../src/components/menus/TextMenu.vue","../../../src/components/menus/BackgroundMenu.vue","../../../src/components/menus/DraggableLayers.vue","../../../src/components/menus/LayersMenu.vue","../../../src/components/Sidebar.vue","../../../src/components/HelpConnectData.vue","../../../src/views/Editor.vue","../../../src/views/Dashboard/SearchBar.vue","../../../src/views/Dashboard/AppListViews/AppsHeader.vue","../../../src/views/Dashboard/ScrollView.vue","../../../src/views/Dashboard/AppListViews/ConfirmCopy.vue","../../../src/views/Dashboard/Apps.vue","../../../src/assets/empty_screen_logo.svg","../../../src/views/Dashboard/Collect.vue","../../../src/views/Dashboard/Control.vue","../../../src/stores/templates.ts","../../../src/views/Dashboard/Dashboard.vue","../../../src/views/Dashboard/CreateApp.vue","../../../node_modules/vue-resize/dist/vue-resize.esm.js","../../../node_modules/vue-observe-visibility/dist/vue-observe-visibility.esm.js","../../../node_modules/scrollparent/scrollparent.js","../../../node_modules/vue-virtual-scroller/dist/vue-virtual-scroller.esm.js","../../../src/views/Dashboard/IconButton.vue","../../../src/views/Dashboard/AppListViews/ListItem.vue","../../../src/views/Dashboard/AppListViews/List.vue","../../../src/views/Dashboard/AppListViews/GridItem.vue","../../../src/views/Dashboard/AppListViews/Grid.vue","../../../src/views/Dashboard/AppListViews/Expanded.vue","../../../src/views/Connection/ConnectionDelete.vue","../../../src/views/Connection/components/AppsInConnection.vue","../../../src/views/Connection/ConnectionList.vue","../../../src/stores/connectionSource.ts","../../../src/views/Connection/Connections.vue","../../../node_modules/vue-router/composables.mjs","../../../src/components/BreadcrumbNavigation.vue","../../../src/views/Connection/ConnectionCreateNew.vue","../../../src/components/data/connections/setup/SourceChooser.vue","../../../src/stores/connectionSetup.ts","../../../src/components/Link.vue","../../../src/components/data/connections/setup/provide/SupportedFormats.vue","../../../src/components/data/connections/connectionErrorMessage.ts","../../../src/components/data/connections/setup/provide/FileUpload.vue","../../../src/components/data/connections/setup/provide/Clipboard.vue","../../../src/components/data/connections/setup/provide/Url.vue","../../../src/types/google.ts","../../../src/components/data/connections/setup/AuthSignInButton.vue","../../../src/components/data/connections/setup/ThirdPartyOAuth.vue","../../../src/components/inputs/TabInput.vue","../../../src/components/data/connections/setup/provide/GoogleSheets.vue","../../../src/components/data/connections/setup/provide/SelectAuthorization.vue","../../../src/components/data/connections/setup/provide/GoogleCalendar.vue","../../../src/components/data/connections/setup/provide/MicrosoftSheets.vue","../../../src/components/data/connections/setup/provide/MicrosoftCalendar.vue","../../../src/components/data/connections/setup/SetupNavigation.vue","../../../src/components/data/connections/setup/SetupProvide.vue","../../../src/components/RemapFilterConditionColumnsMixin.ts","../../../src/components/data/connections/setup/Setup.vue","../../../src/components/inputs/CheckboxInput.vue","../../../src/components/data/DataNodePrimativeArray.vue","../../../src/components/data/connections/setup/schema/DefineColumns.vue","../../../src/components/data/connections/setup/SetupStepConfirm.vue","../../../src/components/data/connections/setup/schema/DefineTreeSchema.vue","../../../src/components/data/connections/setup/schema/DefineSpreadsheetSchema.vue","../../../src/components/widgets/TimeDisplay.vue","../../../src/components/widgets/TimeRangeDisplay.vue","../../../src/components/data/connections/setup/schema/DefineCalendarSchema.vue","../../../src/components/data/connections/setup/schema/DefineManualSchema.vue","../../../src/components/data/connections/setup/SetupSchema.vue","../../../src/components/data/connections/setup/SetupSync.vue","../../../src/components/data/connections/setup/SetupName.vue","../../../src/components/data/connections/manage/RemapColumns.vue","../../../src/components/data/connections/setup/SetupReplace.vue","../../../src/components/data/connections/setup/selection/SelectSpreadsheetValue.vue","../../../src/components/data/connections/manage/data/views/ScalarNodeDisplay.vue","../../../src/components/data/connections/setup/SetupSelect.vue","../../../src/components/data/connections/setup/complete/CompleteDefault.vue","../../../src/components/data/connections/setup/complete/CompleteZapier.vue","../../../src/components/data/connections/setup/complete/ApiExample.vue","../../../src/components/data/connections/setup/complete/CompleteRestApi.vue","../../../src/components/data/connections/setup/SetupComplete.vue","../../../src/stores/connectionSetupV2.ts","../../../src/components/data/connections/setup/SetupNavigationV2.vue","../../../src/components/data/connections/setup/provide/components/inputs/validation/useTextValidation.ts","../../../src/components/data/connections/setup/provide/components/inputs/TextInput.vue","../../../src/components/data/connections/setup/SetupProvideV2.vue","../../../src/components/data/connections/setup/schema/DefineTreeSchemaV2.vue","../../../src/components/data/connections/setup/SetupSchemaV2.vue","../../../src/components/data/connections/setup/SetupSyncV2.vue","../../../src/components/data/connections/setup/SetupNameV2.vue","../../../src/components/data/connections/setup/complete/CompleteDefaultV2.vue","../../../src/components/data/connections/setup/SetupCompleteV2.vue","../../../src/components/data/connections/setup/SetupReplaceV2.vue","../../../src/components/data/connections/setup/SetupSelectV2.vue","../../../src/components/data/connections/setup/SetupV2.vue","../../../src/components/data/connections/manage/Manager.vue","../../../src/components/data/connections/manage/data/views/ScalarSelect.vue","../../../src/components/data/connections/manage/data/views/ReadOnlyData.vue","../../../src/components/data/connections/manualSource/helpers.ts","../../../src/components/data/connections/manualSource/DataRowDropdown.vue","../../../src/components/data/connections/manualSource/inputs/InputBase.vue","../../../src/components/data/connections/manualSource/inputs/TextInput.vue","../../../src/components/data/connections/manualSource/inputs/NumberInput.vue","../../../src/components/data/connections/manualSource/inputs/TextAreaInput.vue","../../../src/components/data/connections/manualSource/inputs/DatetimeInput.vue","../../../src/components/data/connections/manualSource/inputs/ImageUploadInput.vue","../../../src/components/data/connections/manualSource/inputs/ColorInput.vue","../../../src/components/data/connections/manualSource/inputs/BooleanInput.vue","../../../src/components/data/connections/manualSource/inputs/CurrencyInput.vue","../../../src/components/data/connections/manualSource/EditDataRowForm.vue","../../../src/components/data/FileDownloader.vue","../../../src/components/data/connections/manage/SaveButton.vue","../../../src/components/data/connections/manualSource/RepositionRowModal.vue","../../../src/components/data/connections/manualSource/EditDataTable.vue","../../../src/components/data/connections/manage/data/Data.vue","../../../src/components/data/connections/manage/Reauthorize.vue","../../../src/components/data/connections/manage/replace/Replace.vue","../../../src/components/data/connections/manage/settings/Settings.vue","../../../src/components/data/connections/manage/api/Api.vue","../../../src/components/data/connections/manualSource/settings/views/ManualSync.vue","../../../src/components/data/connections/manualSource/settings/views/Synchronization.vue","../../../src/components/data/connections/manualSource/settings/views/SchemaColumns.vue","../../../src/components/data/connections/manualSource/settings/views/Name.vue","../../../src/components/PopoverConfirm.vue","../../../src/components/data/connections/manualSource/settings/views/Disconnect.vue","../../../src/components/data/connections/manualSource/settings/views/Moderation.vue","../../../src/components/data/connections/manualSource/settings/views/Configure.vue","../../../src/components/data/connections/manage/ChooseConnectionModal.vue","../../../src/components/data/connections/manage/ChooseConnection.vue","../../../src/components/data/connections/manage/PreviewData.vue","../../../src/components/data/connections/manage/replace/Remap.vue","../../../src/components/conditions/ConditionsModal.vue","../../../src/components/PrimaryButton.vue","../../../src/components/conditions/ConditionSaved.vue","../../../src/components/conditions/LogicRuleEditor.vue","../../../src/components/conditions/RuleGroupEditor.vue","../../../src/components/conditions/ConditionsIntroStep.vue","../../../src/components/conditions/ConditionsIntro.vue","../../../src/components/conditions/ConditionEditor.vue","../../../src/views/Templates/Base.vue","../../../src/views/Templates/ListItem.vue","../../../src/views/Templates/List.vue","../../../src/types/ConnectionSources.ts","../../../src/router.ts","../../../node_modules/portal-vue/dist/portal-vue.esm.js","../../../src/components/widgets/PhotoDropContainer.vue","../../../src/components/ShapeBase.vue","../../../src/components/widgets/Rectangle/RectangleComponent.vue","../../../src/components/widgets/Ellipse/EllipseComponent.vue","../../../src/components/widgets/Image/ImageComponent.vue","../../../src/components/widgets/Group/GroupComponent.vue","../../../src/components/widgets/Repeater/RepeaterComponent.vue","../../../src/components/widgets/Multiframe/MultiframeComponent.vue","../../../src/components/widgets/Svg/SvgComponent.vue","../../../src/components/widgets/AnalogClock/AnalogClockComponent.vue","../../../src/components/widgets/CalendarEmptyMessage.vue","../../../src/components/widgets/CalendarBase.vue","../../../src/components/widgets/CalendarEvent/CalendarEventComponent.vue","../../../src/components/widgets/CalendarDay/CalendarDayComponent.vue","../../../src/components/widgets/CalendarWeek/CalendarWeekComponent.vue","../../../src/components/widgets/Countdown/CountdownComponent.vue","../../../src/components/widgets/CalendarAgenda/CalendarAgendaComponent.vue","../../../src/components/widgets/CalendarRoomSchedule/CalendarRoomScheduleComponent.vue","../../../src/components/widgets/ProgressBar/ProgressBarComponent.vue","../../../src/components/widgets/ProgressDonut/ProgressDonutComponent.vue","../../../src/components/widgets/ColumnGraph/ColumnGraphComponent.vue","../../../src/components/widgets/BarGraph/BarGraphComponent.vue","../../../src/components/widgets/LineGraph/LineGraphComponent.vue","../../../src/components/widgets/PieGraph/PieGraphComponent.vue","../../../src/components/widgets/StackedGraph/StackedGraphComponent.vue","../../../src/components/widgets/Triangle/TriangleComponent.vue","../../../src/components/widgets/Line/LineComponent.vue","../../../src/components/widgets/Video/VideoComponent.vue","../../../src/components/widgets/index.ts","../../../src/components/widgets/Text/TextWrapper.vue","../../../src/components/widgets/CalendarAgenda/CalendarAgendaWrapper.vue","../../../src/components/widgets/CalendarDay/CalendarDayWrapper.vue","../../../src/components/widgets/CalendarEvent/CalendarEventWrapper.vue","../../../src/components/widgets/CalendarWeek/CalendarWeekWrapper.vue","../../../src/components/widgets/Countdown/CountdownWrapper.vue","../../../src/components/widgets/Repeater/RepeaterWrapper.vue","../../../src/components/widgets/Group/GroupWrapper.vue","../../../src/components/widgets/BarGraph/BarGraphWrapper.vue","../../../src/components/widgets/ColumnGraph/ColumnGraphWrapper.vue","../../../src/components/widgets/PieGraph/PieGraphWrapper.vue","../../../src/components/widgets/StackedGraph/StackedGraphWrapper.vue","../../../src/components/widgets/LineGraph/LineGraphWrapper.vue","../../../src/components/widgets/Video/VideoWrapper.vue","../../../src/components/widgets/Datetime/DatetimeWrapper.vue","../../../src/components/widgets/builderWidgets.ts","../../../src/helpers/builder.ts","../../../src/main.ts"],"sourcesContent":["/*!\n * vue-router v3.6.5\n * (c) 2022 Evan You\n * @license MIT\n */\n/* */\n\nfunction assert (condition, message) {\n if (!condition) {\n throw new Error((\"[vue-router] \" + message))\n }\n}\n\nfunction warn (condition, message) {\n if (!condition) {\n typeof console !== 'undefined' && console.warn((\"[vue-router] \" + message));\n }\n}\n\nfunction extend (a, b) {\n for (var key in b) {\n a[key] = b[key];\n }\n return a\n}\n\n/* */\n\nvar encodeReserveRE = /[!'()*]/g;\nvar encodeReserveReplacer = function (c) { return '%' + c.charCodeAt(0).toString(16); };\nvar commaRE = /%2C/g;\n\n// fixed encodeURIComponent which is more conformant to RFC3986:\n// - escapes [!'()*]\n// - preserve commas\nvar encode = function (str) { return encodeURIComponent(str)\n .replace(encodeReserveRE, encodeReserveReplacer)\n .replace(commaRE, ','); };\n\nfunction decode (str) {\n try {\n return decodeURIComponent(str)\n } catch (err) {\n if (process.env.NODE_ENV !== 'production') {\n warn(false, (\"Error decoding \\\"\" + str + \"\\\". Leaving it intact.\"));\n }\n }\n return str\n}\n\nfunction resolveQuery (\n query,\n extraQuery,\n _parseQuery\n) {\n if ( extraQuery === void 0 ) extraQuery = {};\n\n var parse = _parseQuery || parseQuery;\n var parsedQuery;\n try {\n parsedQuery = parse(query || '');\n } catch (e) {\n process.env.NODE_ENV !== 'production' && warn(false, e.message);\n parsedQuery = {};\n }\n for (var key in extraQuery) {\n var value = extraQuery[key];\n parsedQuery[key] = Array.isArray(value)\n ? value.map(castQueryParamValue)\n : castQueryParamValue(value);\n }\n return parsedQuery\n}\n\nvar castQueryParamValue = function (value) { return (value == null || typeof value === 'object' ? value : String(value)); };\n\nfunction parseQuery (query) {\n var res = {};\n\n query = query.trim().replace(/^(\\?|#|&)/, '');\n\n if (!query) {\n return res\n }\n\n query.split('&').forEach(function (param) {\n var parts = param.replace(/\\+/g, ' ').split('=');\n var key = decode(parts.shift());\n var val = parts.length > 0 ? decode(parts.join('=')) : null;\n\n if (res[key] === undefined) {\n res[key] = val;\n } else if (Array.isArray(res[key])) {\n res[key].push(val);\n } else {\n res[key] = [res[key], val];\n }\n });\n\n return res\n}\n\nfunction stringifyQuery (obj) {\n var res = obj\n ? Object.keys(obj)\n .map(function (key) {\n var val = obj[key];\n\n if (val === undefined) {\n return ''\n }\n\n if (val === null) {\n return encode(key)\n }\n\n if (Array.isArray(val)) {\n var result = [];\n val.forEach(function (val2) {\n if (val2 === undefined) {\n return\n }\n if (val2 === null) {\n result.push(encode(key));\n } else {\n result.push(encode(key) + '=' + encode(val2));\n }\n });\n return result.join('&')\n }\n\n return encode(key) + '=' + encode(val)\n })\n .filter(function (x) { return x.length > 0; })\n .join('&')\n : null;\n return res ? (\"?\" + res) : ''\n}\n\n/* */\n\nvar trailingSlashRE = /\\/?$/;\n\nfunction createRoute (\n record,\n location,\n redirectedFrom,\n router\n) {\n var stringifyQuery = router && router.options.stringifyQuery;\n\n var query = location.query || {};\n try {\n query = clone(query);\n } catch (e) {}\n\n var route = {\n name: location.name || (record && record.name),\n meta: (record && record.meta) || {},\n path: location.path || '/',\n hash: location.hash || '',\n query: query,\n params: location.params || {},\n fullPath: getFullPath(location, stringifyQuery),\n matched: record ? formatMatch(record) : []\n };\n if (redirectedFrom) {\n route.redirectedFrom = getFullPath(redirectedFrom, stringifyQuery);\n }\n return Object.freeze(route)\n}\n\nfunction clone (value) {\n if (Array.isArray(value)) {\n return value.map(clone)\n } else if (value && typeof value === 'object') {\n var res = {};\n for (var key in value) {\n res[key] = clone(value[key]);\n }\n return res\n } else {\n return value\n }\n}\n\n// the starting route that represents the initial state\nvar START = createRoute(null, {\n path: '/'\n});\n\nfunction formatMatch (record) {\n var res = [];\n while (record) {\n res.unshift(record);\n record = record.parent;\n }\n return res\n}\n\nfunction getFullPath (\n ref,\n _stringifyQuery\n) {\n var path = ref.path;\n var query = ref.query; if ( query === void 0 ) query = {};\n var hash = ref.hash; if ( hash === void 0 ) hash = '';\n\n var stringify = _stringifyQuery || stringifyQuery;\n return (path || '/') + stringify(query) + hash\n}\n\nfunction isSameRoute (a, b, onlyPath) {\n if (b === START) {\n return a === b\n } else if (!b) {\n return false\n } else if (a.path && b.path) {\n return a.path.replace(trailingSlashRE, '') === b.path.replace(trailingSlashRE, '') && (onlyPath ||\n a.hash === b.hash &&\n isObjectEqual(a.query, b.query))\n } else if (a.name && b.name) {\n return (\n a.name === b.name &&\n (onlyPath || (\n a.hash === b.hash &&\n isObjectEqual(a.query, b.query) &&\n isObjectEqual(a.params, b.params))\n )\n )\n } else {\n return false\n }\n}\n\nfunction isObjectEqual (a, b) {\n if ( a === void 0 ) a = {};\n if ( b === void 0 ) b = {};\n\n // handle null value #1566\n if (!a || !b) { return a === b }\n var aKeys = Object.keys(a).sort();\n var bKeys = Object.keys(b).sort();\n if (aKeys.length !== bKeys.length) {\n return false\n }\n return aKeys.every(function (key, i) {\n var aVal = a[key];\n var bKey = bKeys[i];\n if (bKey !== key) { return false }\n var bVal = b[key];\n // query values can be null and undefined\n if (aVal == null || bVal == null) { return aVal === bVal }\n // check nested equality\n if (typeof aVal === 'object' && typeof bVal === 'object') {\n return isObjectEqual(aVal, bVal)\n }\n return String(aVal) === String(bVal)\n })\n}\n\nfunction isIncludedRoute (current, target) {\n return (\n current.path.replace(trailingSlashRE, '/').indexOf(\n target.path.replace(trailingSlashRE, '/')\n ) === 0 &&\n (!target.hash || current.hash === target.hash) &&\n queryIncludes(current.query, target.query)\n )\n}\n\nfunction queryIncludes (current, target) {\n for (var key in target) {\n if (!(key in current)) {\n return false\n }\n }\n return true\n}\n\nfunction handleRouteEntered (route) {\n for (var i = 0; i < route.matched.length; i++) {\n var record = route.matched[i];\n for (var name in record.instances) {\n var instance = record.instances[name];\n var cbs = record.enteredCbs[name];\n if (!instance || !cbs) { continue }\n delete record.enteredCbs[name];\n for (var i$1 = 0; i$1 < cbs.length; i$1++) {\n if (!instance._isBeingDestroyed) { cbs[i$1](instance); }\n }\n }\n }\n}\n\nvar View = {\n name: 'RouterView',\n functional: true,\n props: {\n name: {\n type: String,\n default: 'default'\n }\n },\n render: function render (_, ref) {\n var props = ref.props;\n var children = ref.children;\n var parent = ref.parent;\n var data = ref.data;\n\n // used by devtools to display a router-view badge\n data.routerView = true;\n\n // directly use parent context's createElement() function\n // so that components rendered by router-view can resolve named slots\n var h = parent.$createElement;\n var name = props.name;\n var route = parent.$route;\n var cache = parent._routerViewCache || (parent._routerViewCache = {});\n\n // determine current view depth, also check to see if the tree\n // has been toggled inactive but kept-alive.\n var depth = 0;\n var inactive = false;\n while (parent && parent._routerRoot !== parent) {\n var vnodeData = parent.$vnode ? parent.$vnode.data : {};\n if (vnodeData.routerView) {\n depth++;\n }\n if (vnodeData.keepAlive && parent._directInactive && parent._inactive) {\n inactive = true;\n }\n parent = parent.$parent;\n }\n data.routerViewDepth = depth;\n\n // render previous view if the tree is inactive and kept-alive\n if (inactive) {\n var cachedData = cache[name];\n var cachedComponent = cachedData && cachedData.component;\n if (cachedComponent) {\n // #2301\n // pass props\n if (cachedData.configProps) {\n fillPropsinData(cachedComponent, data, cachedData.route, cachedData.configProps);\n }\n return h(cachedComponent, data, children)\n } else {\n // render previous empty view\n return h()\n }\n }\n\n var matched = route.matched[depth];\n var component = matched && matched.components[name];\n\n // render empty node if no matched route or no config component\n if (!matched || !component) {\n cache[name] = null;\n return h()\n }\n\n // cache component\n cache[name] = { component: component };\n\n // attach instance registration hook\n // this will be called in the instance's injected lifecycle hooks\n data.registerRouteInstance = function (vm, val) {\n // val could be undefined for unregistration\n var current = matched.instances[name];\n if (\n (val && current !== vm) ||\n (!val && current === vm)\n ) {\n matched.instances[name] = val;\n }\n }\n\n // also register instance in prepatch hook\n // in case the same component instance is reused across different routes\n ;(data.hook || (data.hook = {})).prepatch = function (_, vnode) {\n matched.instances[name] = vnode.componentInstance;\n };\n\n // register instance in init hook\n // in case kept-alive component be actived when routes changed\n data.hook.init = function (vnode) {\n if (vnode.data.keepAlive &&\n vnode.componentInstance &&\n vnode.componentInstance !== matched.instances[name]\n ) {\n matched.instances[name] = vnode.componentInstance;\n }\n\n // if the route transition has already been confirmed then we weren't\n // able to call the cbs during confirmation as the component was not\n // registered yet, so we call it here.\n handleRouteEntered(route);\n };\n\n var configProps = matched.props && matched.props[name];\n // save route and configProps in cache\n if (configProps) {\n extend(cache[name], {\n route: route,\n configProps: configProps\n });\n fillPropsinData(component, data, route, configProps);\n }\n\n return h(component, data, children)\n }\n};\n\nfunction fillPropsinData (component, data, route, configProps) {\n // resolve props\n var propsToPass = data.props = resolveProps(route, configProps);\n if (propsToPass) {\n // clone to prevent mutation\n propsToPass = data.props = extend({}, propsToPass);\n // pass non-declared props as attrs\n var attrs = data.attrs = data.attrs || {};\n for (var key in propsToPass) {\n if (!component.props || !(key in component.props)) {\n attrs[key] = propsToPass[key];\n delete propsToPass[key];\n }\n }\n }\n}\n\nfunction resolveProps (route, config) {\n switch (typeof config) {\n case 'undefined':\n return\n case 'object':\n return config\n case 'function':\n return config(route)\n case 'boolean':\n return config ? route.params : undefined\n default:\n if (process.env.NODE_ENV !== 'production') {\n warn(\n false,\n \"props in \\\"\" + (route.path) + \"\\\" is a \" + (typeof config) + \", \" +\n \"expecting an object, function or boolean.\"\n );\n }\n }\n}\n\n/* */\n\nfunction resolvePath (\n relative,\n base,\n append\n) {\n var firstChar = relative.charAt(0);\n if (firstChar === '/') {\n return relative\n }\n\n if (firstChar === '?' || firstChar === '#') {\n return base + relative\n }\n\n var stack = base.split('/');\n\n // remove trailing segment if:\n // - not appending\n // - appending to trailing slash (last segment is empty)\n if (!append || !stack[stack.length - 1]) {\n stack.pop();\n }\n\n // resolve relative path\n var segments = relative.replace(/^\\//, '').split('/');\n for (var i = 0; i < segments.length; i++) {\n var segment = segments[i];\n if (segment === '..') {\n stack.pop();\n } else if (segment !== '.') {\n stack.push(segment);\n }\n }\n\n // ensure leading slash\n if (stack[0] !== '') {\n stack.unshift('');\n }\n\n return stack.join('/')\n}\n\nfunction parsePath (path) {\n var hash = '';\n var query = '';\n\n var hashIndex = path.indexOf('#');\n if (hashIndex >= 0) {\n hash = path.slice(hashIndex);\n path = path.slice(0, hashIndex);\n }\n\n var queryIndex = path.indexOf('?');\n if (queryIndex >= 0) {\n query = path.slice(queryIndex + 1);\n path = path.slice(0, queryIndex);\n }\n\n return {\n path: path,\n query: query,\n hash: hash\n }\n}\n\nfunction cleanPath (path) {\n return path.replace(/\\/(?:\\s*\\/)+/g, '/')\n}\n\nvar isarray = Array.isArray || function (arr) {\n return Object.prototype.toString.call(arr) == '[object Array]';\n};\n\n/**\n * Expose `pathToRegexp`.\n */\nvar pathToRegexp_1 = pathToRegexp;\nvar parse_1 = parse;\nvar compile_1 = compile;\nvar tokensToFunction_1 = tokensToFunction;\nvar tokensToRegExp_1 = tokensToRegExp;\n\n/**\n * The main path matching regexp utility.\n *\n * @type {RegExp}\n */\nvar PATH_REGEXP = new RegExp([\n // Match escaped characters that would otherwise appear in future matches.\n // This allows the user to escape special characters that won't transform.\n '(\\\\\\\\.)',\n // Match Express-style parameters and un-named parameters with a prefix\n // and optional suffixes. Matches appear as:\n //\n // \"/:test(\\\\d+)?\" => [\"/\", \"test\", \"\\d+\", undefined, \"?\", undefined]\n // \"/route(\\\\d+)\" => [undefined, undefined, undefined, \"\\d+\", undefined, undefined]\n // \"/*\" => [\"/\", undefined, undefined, undefined, undefined, \"*\"]\n '([\\\\/.])?(?:(?:\\\\:(\\\\w+)(?:\\\\(((?:\\\\\\\\.|[^\\\\\\\\()])+)\\\\))?|\\\\(((?:\\\\\\\\.|[^\\\\\\\\()])+)\\\\))([+*?])?|(\\\\*))'\n].join('|'), 'g');\n\n/**\n * Parse a string for the raw tokens.\n *\n * @param {string} str\n * @param {Object=} options\n * @return {!Array}\n */\nfunction parse (str, options) {\n var tokens = [];\n var key = 0;\n var index = 0;\n var path = '';\n var defaultDelimiter = options && options.delimiter || '/';\n var res;\n\n while ((res = PATH_REGEXP.exec(str)) != null) {\n var m = res[0];\n var escaped = res[1];\n var offset = res.index;\n path += str.slice(index, offset);\n index = offset + m.length;\n\n // Ignore already escaped sequences.\n if (escaped) {\n path += escaped[1];\n continue\n }\n\n var next = str[index];\n var prefix = res[2];\n var name = res[3];\n var capture = res[4];\n var group = res[5];\n var modifier = res[6];\n var asterisk = res[7];\n\n // Push the current path onto the tokens.\n if (path) {\n tokens.push(path);\n path = '';\n }\n\n var partial = prefix != null && next != null && next !== prefix;\n var repeat = modifier === '+' || modifier === '*';\n var optional = modifier === '?' || modifier === '*';\n var delimiter = res[2] || defaultDelimiter;\n var pattern = capture || group;\n\n tokens.push({\n name: name || key++,\n prefix: prefix || '',\n delimiter: delimiter,\n optional: optional,\n repeat: repeat,\n partial: partial,\n asterisk: !!asterisk,\n pattern: pattern ? escapeGroup(pattern) : (asterisk ? '.*' : '[^' + escapeString(delimiter) + ']+?')\n });\n }\n\n // Match any characters still remaining.\n if (index < str.length) {\n path += str.substr(index);\n }\n\n // If the path exists, push it onto the end.\n if (path) {\n tokens.push(path);\n }\n\n return tokens\n}\n\n/**\n * Compile a string to a template function for the path.\n *\n * @param {string} str\n * @param {Object=} options\n * @return {!function(Object=, Object=)}\n */\nfunction compile (str, options) {\n return tokensToFunction(parse(str, options), options)\n}\n\n/**\n * Prettier encoding of URI path segments.\n *\n * @param {string}\n * @return {string}\n */\nfunction encodeURIComponentPretty (str) {\n return encodeURI(str).replace(/[\\/?#]/g, function (c) {\n return '%' + c.charCodeAt(0).toString(16).toUpperCase()\n })\n}\n\n/**\n * Encode the asterisk parameter. Similar to `pretty`, but allows slashes.\n *\n * @param {string}\n * @return {string}\n */\nfunction encodeAsterisk (str) {\n return encodeURI(str).replace(/[?#]/g, function (c) {\n return '%' + c.charCodeAt(0).toString(16).toUpperCase()\n })\n}\n\n/**\n * Expose a method for transforming tokens into the path function.\n */\nfunction tokensToFunction (tokens, options) {\n // Compile all the tokens into regexps.\n var matches = new Array(tokens.length);\n\n // Compile all the patterns before compilation.\n for (var i = 0; i < tokens.length; i++) {\n if (typeof tokens[i] === 'object') {\n matches[i] = new RegExp('^(?:' + tokens[i].pattern + ')$', flags(options));\n }\n }\n\n return function (obj, opts) {\n var path = '';\n var data = obj || {};\n var options = opts || {};\n var encode = options.pretty ? encodeURIComponentPretty : encodeURIComponent;\n\n for (var i = 0; i < tokens.length; i++) {\n var token = tokens[i];\n\n if (typeof token === 'string') {\n path += token;\n\n continue\n }\n\n var value = data[token.name];\n var segment;\n\n if (value == null) {\n if (token.optional) {\n // Prepend partial segment prefixes.\n if (token.partial) {\n path += token.prefix;\n }\n\n continue\n } else {\n throw new TypeError('Expected \"' + token.name + '\" to be defined')\n }\n }\n\n if (isarray(value)) {\n if (!token.repeat) {\n throw new TypeError('Expected \"' + token.name + '\" to not repeat, but received `' + JSON.stringify(value) + '`')\n }\n\n if (value.length === 0) {\n if (token.optional) {\n continue\n } else {\n throw new TypeError('Expected \"' + token.name + '\" to not be empty')\n }\n }\n\n for (var j = 0; j < value.length; j++) {\n segment = encode(value[j]);\n\n if (!matches[i].test(segment)) {\n throw new TypeError('Expected all \"' + token.name + '\" to match \"' + token.pattern + '\", but received `' + JSON.stringify(segment) + '`')\n }\n\n path += (j === 0 ? token.prefix : token.delimiter) + segment;\n }\n\n continue\n }\n\n segment = token.asterisk ? encodeAsterisk(value) : encode(value);\n\n if (!matches[i].test(segment)) {\n throw new TypeError('Expected \"' + token.name + '\" to match \"' + token.pattern + '\", but received \"' + segment + '\"')\n }\n\n path += token.prefix + segment;\n }\n\n return path\n }\n}\n\n/**\n * Escape a regular expression string.\n *\n * @param {string} str\n * @return {string}\n */\nfunction escapeString (str) {\n return str.replace(/([.+*?=^!:${}()[\\]|\\/\\\\])/g, '\\\\$1')\n}\n\n/**\n * Escape the capturing group by escaping special characters and meaning.\n *\n * @param {string} group\n * @return {string}\n */\nfunction escapeGroup (group) {\n return group.replace(/([=!:$\\/()])/g, '\\\\$1')\n}\n\n/**\n * Attach the keys as a property of the regexp.\n *\n * @param {!RegExp} re\n * @param {Array} keys\n * @return {!RegExp}\n */\nfunction attachKeys (re, keys) {\n re.keys = keys;\n return re\n}\n\n/**\n * Get the flags for a regexp from the options.\n *\n * @param {Object} options\n * @return {string}\n */\nfunction flags (options) {\n return options && options.sensitive ? '' : 'i'\n}\n\n/**\n * Pull out keys from a regexp.\n *\n * @param {!RegExp} path\n * @param {!Array} keys\n * @return {!RegExp}\n */\nfunction regexpToRegexp (path, keys) {\n // Use a negative lookahead to match only capturing groups.\n var groups = path.source.match(/\\((?!\\?)/g);\n\n if (groups) {\n for (var i = 0; i < groups.length; i++) {\n keys.push({\n name: i,\n prefix: null,\n delimiter: null,\n optional: false,\n repeat: false,\n partial: false,\n asterisk: false,\n pattern: null\n });\n }\n }\n\n return attachKeys(path, keys)\n}\n\n/**\n * Transform an array into a regexp.\n *\n * @param {!Array} path\n * @param {Array} keys\n * @param {!Object} options\n * @return {!RegExp}\n */\nfunction arrayToRegexp (path, keys, options) {\n var parts = [];\n\n for (var i = 0; i < path.length; i++) {\n parts.push(pathToRegexp(path[i], keys, options).source);\n }\n\n var regexp = new RegExp('(?:' + parts.join('|') + ')', flags(options));\n\n return attachKeys(regexp, keys)\n}\n\n/**\n * Create a path regexp from string input.\n *\n * @param {string} path\n * @param {!Array} keys\n * @param {!Object} options\n * @return {!RegExp}\n */\nfunction stringToRegexp (path, keys, options) {\n return tokensToRegExp(parse(path, options), keys, options)\n}\n\n/**\n * Expose a function for taking tokens and returning a RegExp.\n *\n * @param {!Array} tokens\n * @param {(Array|Object)=} keys\n * @param {Object=} options\n * @return {!RegExp}\n */\nfunction tokensToRegExp (tokens, keys, options) {\n if (!isarray(keys)) {\n options = /** @type {!Object} */ (keys || options);\n keys = [];\n }\n\n options = options || {};\n\n var strict = options.strict;\n var end = options.end !== false;\n var route = '';\n\n // Iterate over the tokens and create our regexp string.\n for (var i = 0; i < tokens.length; i++) {\n var token = tokens[i];\n\n if (typeof token === 'string') {\n route += escapeString(token);\n } else {\n var prefix = escapeString(token.prefix);\n var capture = '(?:' + token.pattern + ')';\n\n keys.push(token);\n\n if (token.repeat) {\n capture += '(?:' + prefix + capture + ')*';\n }\n\n if (token.optional) {\n if (!token.partial) {\n capture = '(?:' + prefix + '(' + capture + '))?';\n } else {\n capture = prefix + '(' + capture + ')?';\n }\n } else {\n capture = prefix + '(' + capture + ')';\n }\n\n route += capture;\n }\n }\n\n var delimiter = escapeString(options.delimiter || '/');\n var endsWithDelimiter = route.slice(-delimiter.length) === delimiter;\n\n // In non-strict mode we allow a slash at the end of match. If the path to\n // match already ends with a slash, we remove it for consistency. The slash\n // is valid at the end of a path match, not in the middle. This is important\n // in non-ending mode, where \"/test/\" shouldn't match \"/test//route\".\n if (!strict) {\n route = (endsWithDelimiter ? route.slice(0, -delimiter.length) : route) + '(?:' + delimiter + '(?=$))?';\n }\n\n if (end) {\n route += '$';\n } else {\n // In non-ending mode, we need the capturing groups to match as much as\n // possible by using a positive lookahead to the end or next path segment.\n route += strict && endsWithDelimiter ? '' : '(?=' + delimiter + '|$)';\n }\n\n return attachKeys(new RegExp('^' + route, flags(options)), keys)\n}\n\n/**\n * Normalize the given path string, returning a regular expression.\n *\n * An empty array can be passed in for the keys, which will hold the\n * placeholder key descriptions. For example, using `/user/:id`, `keys` will\n * contain `[{ name: 'id', delimiter: '/', optional: false, repeat: false }]`.\n *\n * @param {(string|RegExp|Array)} path\n * @param {(Array|Object)=} keys\n * @param {Object=} options\n * @return {!RegExp}\n */\nfunction pathToRegexp (path, keys, options) {\n if (!isarray(keys)) {\n options = /** @type {!Object} */ (keys || options);\n keys = [];\n }\n\n options = options || {};\n\n if (path instanceof RegExp) {\n return regexpToRegexp(path, /** @type {!Array} */ (keys))\n }\n\n if (isarray(path)) {\n return arrayToRegexp(/** @type {!Array} */ (path), /** @type {!Array} */ (keys), options)\n }\n\n return stringToRegexp(/** @type {string} */ (path), /** @type {!Array} */ (keys), options)\n}\npathToRegexp_1.parse = parse_1;\npathToRegexp_1.compile = compile_1;\npathToRegexp_1.tokensToFunction = tokensToFunction_1;\npathToRegexp_1.tokensToRegExp = tokensToRegExp_1;\n\n/* */\n\n// $flow-disable-line\nvar regexpCompileCache = Object.create(null);\n\nfunction fillParams (\n path,\n params,\n routeMsg\n) {\n params = params || {};\n try {\n var filler =\n regexpCompileCache[path] ||\n (regexpCompileCache[path] = pathToRegexp_1.compile(path));\n\n // Fix #2505 resolving asterisk routes { name: 'not-found', params: { pathMatch: '/not-found' }}\n // and fix #3106 so that you can work with location descriptor object having params.pathMatch equal to empty string\n if (typeof params.pathMatch === 'string') { params[0] = params.pathMatch; }\n\n return filler(params, { pretty: true })\n } catch (e) {\n if (process.env.NODE_ENV !== 'production') {\n // Fix #3072 no warn if `pathMatch` is string\n warn(typeof params.pathMatch === 'string', (\"missing param for \" + routeMsg + \": \" + (e.message)));\n }\n return ''\n } finally {\n // delete the 0 if it was added\n delete params[0];\n }\n}\n\n/* */\n\nfunction normalizeLocation (\n raw,\n current,\n append,\n router\n) {\n var next = typeof raw === 'string' ? { path: raw } : raw;\n // named target\n if (next._normalized) {\n return next\n } else if (next.name) {\n next = extend({}, raw);\n var params = next.params;\n if (params && typeof params === 'object') {\n next.params = extend({}, params);\n }\n return next\n }\n\n // relative params\n if (!next.path && next.params && current) {\n next = extend({}, next);\n next._normalized = true;\n var params$1 = extend(extend({}, current.params), next.params);\n if (current.name) {\n next.name = current.name;\n next.params = params$1;\n } else if (current.matched.length) {\n var rawPath = current.matched[current.matched.length - 1].path;\n next.path = fillParams(rawPath, params$1, (\"path \" + (current.path)));\n } else if (process.env.NODE_ENV !== 'production') {\n warn(false, \"relative params navigation requires a current route.\");\n }\n return next\n }\n\n var parsedPath = parsePath(next.path || '');\n var basePath = (current && current.path) || '/';\n var path = parsedPath.path\n ? resolvePath(parsedPath.path, basePath, append || next.append)\n : basePath;\n\n var query = resolveQuery(\n parsedPath.query,\n next.query,\n router && router.options.parseQuery\n );\n\n var hash = next.hash || parsedPath.hash;\n if (hash && hash.charAt(0) !== '#') {\n hash = \"#\" + hash;\n }\n\n return {\n _normalized: true,\n path: path,\n query: query,\n hash: hash\n }\n}\n\n/* */\n\n// work around weird flow bug\nvar toTypes = [String, Object];\nvar eventTypes = [String, Array];\n\nvar noop = function () {};\n\nvar warnedCustomSlot;\nvar warnedTagProp;\nvar warnedEventProp;\n\nvar Link = {\n name: 'RouterLink',\n props: {\n to: {\n type: toTypes,\n required: true\n },\n tag: {\n type: String,\n default: 'a'\n },\n custom: Boolean,\n exact: Boolean,\n exactPath: Boolean,\n append: Boolean,\n replace: Boolean,\n activeClass: String,\n exactActiveClass: String,\n ariaCurrentValue: {\n type: String,\n default: 'page'\n },\n event: {\n type: eventTypes,\n default: 'click'\n }\n },\n render: function render (h) {\n var this$1$1 = this;\n\n var router = this.$router;\n var current = this.$route;\n var ref = router.resolve(\n this.to,\n current,\n this.append\n );\n var location = ref.location;\n var route = ref.route;\n var href = ref.href;\n\n var classes = {};\n var globalActiveClass = router.options.linkActiveClass;\n var globalExactActiveClass = router.options.linkExactActiveClass;\n // Support global empty active class\n var activeClassFallback =\n globalActiveClass == null ? 'router-link-active' : globalActiveClass;\n var exactActiveClassFallback =\n globalExactActiveClass == null\n ? 'router-link-exact-active'\n : globalExactActiveClass;\n var activeClass =\n this.activeClass == null ? activeClassFallback : this.activeClass;\n var exactActiveClass =\n this.exactActiveClass == null\n ? exactActiveClassFallback\n : this.exactActiveClass;\n\n var compareTarget = route.redirectedFrom\n ? createRoute(null, normalizeLocation(route.redirectedFrom), null, router)\n : route;\n\n classes[exactActiveClass] = isSameRoute(current, compareTarget, this.exactPath);\n classes[activeClass] = this.exact || this.exactPath\n ? classes[exactActiveClass]\n : isIncludedRoute(current, compareTarget);\n\n var ariaCurrentValue = classes[exactActiveClass] ? this.ariaCurrentValue : null;\n\n var handler = function (e) {\n if (guardEvent(e)) {\n if (this$1$1.replace) {\n router.replace(location, noop);\n } else {\n router.push(location, noop);\n }\n }\n };\n\n var on = { click: guardEvent };\n if (Array.isArray(this.event)) {\n this.event.forEach(function (e) {\n on[e] = handler;\n });\n } else {\n on[this.event] = handler;\n }\n\n var data = { class: classes };\n\n var scopedSlot =\n !this.$scopedSlots.$hasNormal &&\n this.$scopedSlots.default &&\n this.$scopedSlots.default({\n href: href,\n route: route,\n navigate: handler,\n isActive: classes[activeClass],\n isExactActive: classes[exactActiveClass]\n });\n\n if (scopedSlot) {\n if (process.env.NODE_ENV !== 'production' && !this.custom) {\n !warnedCustomSlot && warn(false, 'In Vue Router 4, the v-slot API will by default wrap its content with an element. Use the custom prop to remove this warning:\\n\\n');\n warnedCustomSlot = true;\n }\n if (scopedSlot.length === 1) {\n return scopedSlot[0]\n } else if (scopedSlot.length > 1 || !scopedSlot.length) {\n if (process.env.NODE_ENV !== 'production') {\n warn(\n false,\n (\" with to=\\\"\" + (this.to) + \"\\\" is trying to use a scoped slot but it didn't provide exactly one child. Wrapping the content with a span element.\")\n );\n }\n return scopedSlot.length === 0 ? h() : h('span', {}, scopedSlot)\n }\n }\n\n if (process.env.NODE_ENV !== 'production') {\n if ('tag' in this.$options.propsData && !warnedTagProp) {\n warn(\n false,\n \"'s tag prop is deprecated and has been removed in Vue Router 4. Use the v-slot API to remove this warning: https://next.router.vuejs.org/guide/migration/#removal-of-event-and-tag-props-in-router-link.\"\n );\n warnedTagProp = true;\n }\n if ('event' in this.$options.propsData && !warnedEventProp) {\n warn(\n false,\n \"'s event prop is deprecated and has been removed in Vue Router 4. Use the v-slot API to remove this warning: https://next.router.vuejs.org/guide/migration/#removal-of-event-and-tag-props-in-router-link.\"\n );\n warnedEventProp = true;\n }\n }\n\n if (this.tag === 'a') {\n data.on = on;\n data.attrs = { href: href, 'aria-current': ariaCurrentValue };\n } else {\n // find the first child and apply listener and href\n var a = findAnchor(this.$slots.default);\n if (a) {\n // in case the is a static node\n a.isStatic = false;\n var aData = (a.data = extend({}, a.data));\n aData.on = aData.on || {};\n // transform existing events in both objects into arrays so we can push later\n for (var event in aData.on) {\n var handler$1 = aData.on[event];\n if (event in on) {\n aData.on[event] = Array.isArray(handler$1) ? handler$1 : [handler$1];\n }\n }\n // append new listeners for router-link\n for (var event$1 in on) {\n if (event$1 in aData.on) {\n // on[event] is always a function\n aData.on[event$1].push(on[event$1]);\n } else {\n aData.on[event$1] = handler;\n }\n }\n\n var aAttrs = (a.data.attrs = extend({}, a.data.attrs));\n aAttrs.href = href;\n aAttrs['aria-current'] = ariaCurrentValue;\n } else {\n // doesn't have child, apply listener to self\n data.on = on;\n }\n }\n\n return h(this.tag, data, this.$slots.default)\n }\n};\n\nfunction guardEvent (e) {\n // don't redirect with control keys\n if (e.metaKey || e.altKey || e.ctrlKey || e.shiftKey) { return }\n // don't redirect when preventDefault called\n if (e.defaultPrevented) { return }\n // don't redirect on right click\n if (e.button !== undefined && e.button !== 0) { return }\n // don't redirect if `target=\"_blank\"`\n if (e.currentTarget && e.currentTarget.getAttribute) {\n var target = e.currentTarget.getAttribute('target');\n if (/\\b_blank\\b/i.test(target)) { return }\n }\n // this may be a Weex event which doesn't have this method\n if (e.preventDefault) {\n e.preventDefault();\n }\n return true\n}\n\nfunction findAnchor (children) {\n if (children) {\n var child;\n for (var i = 0; i < children.length; i++) {\n child = children[i];\n if (child.tag === 'a') {\n return child\n }\n if (child.children && (child = findAnchor(child.children))) {\n return child\n }\n }\n }\n}\n\nvar _Vue;\n\nfunction install (Vue) {\n if (install.installed && _Vue === Vue) { return }\n install.installed = true;\n\n _Vue = Vue;\n\n var isDef = function (v) { return v !== undefined; };\n\n var registerInstance = function (vm, callVal) {\n var i = vm.$options._parentVnode;\n if (isDef(i) && isDef(i = i.data) && isDef(i = i.registerRouteInstance)) {\n i(vm, callVal);\n }\n };\n\n Vue.mixin({\n beforeCreate: function beforeCreate () {\n if (isDef(this.$options.router)) {\n this._routerRoot = this;\n this._router = this.$options.router;\n this._router.init(this);\n Vue.util.defineReactive(this, '_route', this._router.history.current);\n } else {\n this._routerRoot = (this.$parent && this.$parent._routerRoot) || this;\n }\n registerInstance(this, this);\n },\n destroyed: function destroyed () {\n registerInstance(this);\n }\n });\n\n Object.defineProperty(Vue.prototype, '$router', {\n get: function get () { return this._routerRoot._router }\n });\n\n Object.defineProperty(Vue.prototype, '$route', {\n get: function get () { return this._routerRoot._route }\n });\n\n Vue.component('RouterView', View);\n Vue.component('RouterLink', Link);\n\n var strats = Vue.config.optionMergeStrategies;\n // use the same hook merging strategy for route hooks\n strats.beforeRouteEnter = strats.beforeRouteLeave = strats.beforeRouteUpdate = strats.created;\n}\n\n/* */\n\nvar inBrowser = typeof window !== 'undefined';\n\n/* */\n\nfunction createRouteMap (\n routes,\n oldPathList,\n oldPathMap,\n oldNameMap,\n parentRoute\n) {\n // the path list is used to control path matching priority\n var pathList = oldPathList || [];\n // $flow-disable-line\n var pathMap = oldPathMap || Object.create(null);\n // $flow-disable-line\n var nameMap = oldNameMap || Object.create(null);\n\n routes.forEach(function (route) {\n addRouteRecord(pathList, pathMap, nameMap, route, parentRoute);\n });\n\n // ensure wildcard routes are always at the end\n for (var i = 0, l = pathList.length; i < l; i++) {\n if (pathList[i] === '*') {\n pathList.push(pathList.splice(i, 1)[0]);\n l--;\n i--;\n }\n }\n\n if (process.env.NODE_ENV === 'development') {\n // warn if routes do not include leading slashes\n var found = pathList\n // check for missing leading slash\n .filter(function (path) { return path && path.charAt(0) !== '*' && path.charAt(0) !== '/'; });\n\n if (found.length > 0) {\n var pathNames = found.map(function (path) { return (\"- \" + path); }).join('\\n');\n warn(false, (\"Non-nested routes must include a leading slash character. Fix the following routes: \\n\" + pathNames));\n }\n }\n\n return {\n pathList: pathList,\n pathMap: pathMap,\n nameMap: nameMap\n }\n}\n\nfunction addRouteRecord (\n pathList,\n pathMap,\n nameMap,\n route,\n parent,\n matchAs\n) {\n var path = route.path;\n var name = route.name;\n if (process.env.NODE_ENV !== 'production') {\n assert(path != null, \"\\\"path\\\" is required in a route configuration.\");\n assert(\n typeof route.component !== 'string',\n \"route config \\\"component\\\" for path: \" + (String(\n path || name\n )) + \" cannot be a \" + \"string id. Use an actual component instead.\"\n );\n\n warn(\n // eslint-disable-next-line no-control-regex\n !/[^\\u0000-\\u007F]+/.test(path),\n \"Route with path \\\"\" + path + \"\\\" contains unencoded characters, make sure \" +\n \"your path is correctly encoded before passing it to the router. Use \" +\n \"encodeURI to encode static segments of your path.\"\n );\n }\n\n var pathToRegexpOptions =\n route.pathToRegexpOptions || {};\n var normalizedPath = normalizePath(path, parent, pathToRegexpOptions.strict);\n\n if (typeof route.caseSensitive === 'boolean') {\n pathToRegexpOptions.sensitive = route.caseSensitive;\n }\n\n var record = {\n path: normalizedPath,\n regex: compileRouteRegex(normalizedPath, pathToRegexpOptions),\n components: route.components || { default: route.component },\n alias: route.alias\n ? typeof route.alias === 'string'\n ? [route.alias]\n : route.alias\n : [],\n instances: {},\n enteredCbs: {},\n name: name,\n parent: parent,\n matchAs: matchAs,\n redirect: route.redirect,\n beforeEnter: route.beforeEnter,\n meta: route.meta || {},\n props:\n route.props == null\n ? {}\n : route.components\n ? route.props\n : { default: route.props }\n };\n\n if (route.children) {\n // Warn if route is named, does not redirect and has a default child route.\n // If users navigate to this route by name, the default child will\n // not be rendered (GH Issue #629)\n if (process.env.NODE_ENV !== 'production') {\n if (\n route.name &&\n !route.redirect &&\n route.children.some(function (child) { return /^\\/?$/.test(child.path); })\n ) {\n warn(\n false,\n \"Named Route '\" + (route.name) + \"' has a default child route. \" +\n \"When navigating to this named route (:to=\\\"{name: '\" + (route.name) + \"'}\\\"), \" +\n \"the default child route will not be rendered. Remove the name from \" +\n \"this route and use the name of the default child route for named \" +\n \"links instead.\"\n );\n }\n }\n route.children.forEach(function (child) {\n var childMatchAs = matchAs\n ? cleanPath((matchAs + \"/\" + (child.path)))\n : undefined;\n addRouteRecord(pathList, pathMap, nameMap, child, record, childMatchAs);\n });\n }\n\n if (!pathMap[record.path]) {\n pathList.push(record.path);\n pathMap[record.path] = record;\n }\n\n if (route.alias !== undefined) {\n var aliases = Array.isArray(route.alias) ? route.alias : [route.alias];\n for (var i = 0; i < aliases.length; ++i) {\n var alias = aliases[i];\n if (process.env.NODE_ENV !== 'production' && alias === path) {\n warn(\n false,\n (\"Found an alias with the same value as the path: \\\"\" + path + \"\\\". You have to remove that alias. It will be ignored in development.\")\n );\n // skip in dev to make it work\n continue\n }\n\n var aliasRoute = {\n path: alias,\n children: route.children\n };\n addRouteRecord(\n pathList,\n pathMap,\n nameMap,\n aliasRoute,\n parent,\n record.path || '/' // matchAs\n );\n }\n }\n\n if (name) {\n if (!nameMap[name]) {\n nameMap[name] = record;\n } else if (process.env.NODE_ENV !== 'production' && !matchAs) {\n warn(\n false,\n \"Duplicate named routes definition: \" +\n \"{ name: \\\"\" + name + \"\\\", path: \\\"\" + (record.path) + \"\\\" }\"\n );\n }\n }\n}\n\nfunction compileRouteRegex (\n path,\n pathToRegexpOptions\n) {\n var regex = pathToRegexp_1(path, [], pathToRegexpOptions);\n if (process.env.NODE_ENV !== 'production') {\n var keys = Object.create(null);\n regex.keys.forEach(function (key) {\n warn(\n !keys[key.name],\n (\"Duplicate param keys in route with path: \\\"\" + path + \"\\\"\")\n );\n keys[key.name] = true;\n });\n }\n return regex\n}\n\nfunction normalizePath (\n path,\n parent,\n strict\n) {\n if (!strict) { path = path.replace(/\\/$/, ''); }\n if (path[0] === '/') { return path }\n if (parent == null) { return path }\n return cleanPath(((parent.path) + \"/\" + path))\n}\n\n/* */\n\n\n\nfunction createMatcher (\n routes,\n router\n) {\n var ref = createRouteMap(routes);\n var pathList = ref.pathList;\n var pathMap = ref.pathMap;\n var nameMap = ref.nameMap;\n\n function addRoutes (routes) {\n createRouteMap(routes, pathList, pathMap, nameMap);\n }\n\n function addRoute (parentOrRoute, route) {\n var parent = (typeof parentOrRoute !== 'object') ? nameMap[parentOrRoute] : undefined;\n // $flow-disable-line\n createRouteMap([route || parentOrRoute], pathList, pathMap, nameMap, parent);\n\n // add aliases of parent\n if (parent && parent.alias.length) {\n createRouteMap(\n // $flow-disable-line route is defined if parent is\n parent.alias.map(function (alias) { return ({ path: alias, children: [route] }); }),\n pathList,\n pathMap,\n nameMap,\n parent\n );\n }\n }\n\n function getRoutes () {\n return pathList.map(function (path) { return pathMap[path]; })\n }\n\n function match (\n raw,\n currentRoute,\n redirectedFrom\n ) {\n var location = normalizeLocation(raw, currentRoute, false, router);\n var name = location.name;\n\n if (name) {\n var record = nameMap[name];\n if (process.env.NODE_ENV !== 'production') {\n warn(record, (\"Route with name '\" + name + \"' does not exist\"));\n }\n if (!record) { return _createRoute(null, location) }\n var paramNames = record.regex.keys\n .filter(function (key) { return !key.optional; })\n .map(function (key) { return key.name; });\n\n if (typeof location.params !== 'object') {\n location.params = {};\n }\n\n if (currentRoute && typeof currentRoute.params === 'object') {\n for (var key in currentRoute.params) {\n if (!(key in location.params) && paramNames.indexOf(key) > -1) {\n location.params[key] = currentRoute.params[key];\n }\n }\n }\n\n location.path = fillParams(record.path, location.params, (\"named route \\\"\" + name + \"\\\"\"));\n return _createRoute(record, location, redirectedFrom)\n } else if (location.path) {\n location.params = {};\n for (var i = 0; i < pathList.length; i++) {\n var path = pathList[i];\n var record$1 = pathMap[path];\n if (matchRoute(record$1.regex, location.path, location.params)) {\n return _createRoute(record$1, location, redirectedFrom)\n }\n }\n }\n // no match\n return _createRoute(null, location)\n }\n\n function redirect (\n record,\n location\n ) {\n var originalRedirect = record.redirect;\n var redirect = typeof originalRedirect === 'function'\n ? originalRedirect(createRoute(record, location, null, router))\n : originalRedirect;\n\n if (typeof redirect === 'string') {\n redirect = { path: redirect };\n }\n\n if (!redirect || typeof redirect !== 'object') {\n if (process.env.NODE_ENV !== 'production') {\n warn(\n false, (\"invalid redirect option: \" + (JSON.stringify(redirect)))\n );\n }\n return _createRoute(null, location)\n }\n\n var re = redirect;\n var name = re.name;\n var path = re.path;\n var query = location.query;\n var hash = location.hash;\n var params = location.params;\n query = re.hasOwnProperty('query') ? re.query : query;\n hash = re.hasOwnProperty('hash') ? re.hash : hash;\n params = re.hasOwnProperty('params') ? re.params : params;\n\n if (name) {\n // resolved named direct\n var targetRecord = nameMap[name];\n if (process.env.NODE_ENV !== 'production') {\n assert(targetRecord, (\"redirect failed: named route \\\"\" + name + \"\\\" not found.\"));\n }\n return match({\n _normalized: true,\n name: name,\n query: query,\n hash: hash,\n params: params\n }, undefined, location)\n } else if (path) {\n // 1. resolve relative redirect\n var rawPath = resolveRecordPath(path, record);\n // 2. resolve params\n var resolvedPath = fillParams(rawPath, params, (\"redirect route with path \\\"\" + rawPath + \"\\\"\"));\n // 3. rematch with existing query and hash\n return match({\n _normalized: true,\n path: resolvedPath,\n query: query,\n hash: hash\n }, undefined, location)\n } else {\n if (process.env.NODE_ENV !== 'production') {\n warn(false, (\"invalid redirect option: \" + (JSON.stringify(redirect))));\n }\n return _createRoute(null, location)\n }\n }\n\n function alias (\n record,\n location,\n matchAs\n ) {\n var aliasedPath = fillParams(matchAs, location.params, (\"aliased route with path \\\"\" + matchAs + \"\\\"\"));\n var aliasedMatch = match({\n _normalized: true,\n path: aliasedPath\n });\n if (aliasedMatch) {\n var matched = aliasedMatch.matched;\n var aliasedRecord = matched[matched.length - 1];\n location.params = aliasedMatch.params;\n return _createRoute(aliasedRecord, location)\n }\n return _createRoute(null, location)\n }\n\n function _createRoute (\n record,\n location,\n redirectedFrom\n ) {\n if (record && record.redirect) {\n return redirect(record, redirectedFrom || location)\n }\n if (record && record.matchAs) {\n return alias(record, location, record.matchAs)\n }\n return createRoute(record, location, redirectedFrom, router)\n }\n\n return {\n match: match,\n addRoute: addRoute,\n getRoutes: getRoutes,\n addRoutes: addRoutes\n }\n}\n\nfunction matchRoute (\n regex,\n path,\n params\n) {\n var m = path.match(regex);\n\n if (!m) {\n return false\n } else if (!params) {\n return true\n }\n\n for (var i = 1, len = m.length; i < len; ++i) {\n var key = regex.keys[i - 1];\n if (key) {\n // Fix #1994: using * with props: true generates a param named 0\n params[key.name || 'pathMatch'] = typeof m[i] === 'string' ? decode(m[i]) : m[i];\n }\n }\n\n return true\n}\n\nfunction resolveRecordPath (path, record) {\n return resolvePath(path, record.parent ? record.parent.path : '/', true)\n}\n\n/* */\n\n// use User Timing api (if present) for more accurate key precision\nvar Time =\n inBrowser && window.performance && window.performance.now\n ? window.performance\n : Date;\n\nfunction genStateKey () {\n return Time.now().toFixed(3)\n}\n\nvar _key = genStateKey();\n\nfunction getStateKey () {\n return _key\n}\n\nfunction setStateKey (key) {\n return (_key = key)\n}\n\n/* */\n\nvar positionStore = Object.create(null);\n\nfunction setupScroll () {\n // Prevent browser scroll behavior on History popstate\n if ('scrollRestoration' in window.history) {\n window.history.scrollRestoration = 'manual';\n }\n // Fix for #1585 for Firefox\n // Fix for #2195 Add optional third attribute to workaround a bug in safari https://bugs.webkit.org/show_bug.cgi?id=182678\n // Fix for #2774 Support for apps loaded from Windows file shares not mapped to network drives: replaced location.origin with\n // window.location.protocol + '//' + window.location.host\n // location.host contains the port and location.hostname doesn't\n var protocolAndPath = window.location.protocol + '//' + window.location.host;\n var absolutePath = window.location.href.replace(protocolAndPath, '');\n // preserve existing history state as it could be overriden by the user\n var stateCopy = extend({}, window.history.state);\n stateCopy.key = getStateKey();\n window.history.replaceState(stateCopy, '', absolutePath);\n window.addEventListener('popstate', handlePopState);\n return function () {\n window.removeEventListener('popstate', handlePopState);\n }\n}\n\nfunction handleScroll (\n router,\n to,\n from,\n isPop\n) {\n if (!router.app) {\n return\n }\n\n var behavior = router.options.scrollBehavior;\n if (!behavior) {\n return\n }\n\n if (process.env.NODE_ENV !== 'production') {\n assert(typeof behavior === 'function', \"scrollBehavior must be a function\");\n }\n\n // wait until re-render finishes before scrolling\n router.app.$nextTick(function () {\n var position = getScrollPosition();\n var shouldScroll = behavior.call(\n router,\n to,\n from,\n isPop ? position : null\n );\n\n if (!shouldScroll) {\n return\n }\n\n if (typeof shouldScroll.then === 'function') {\n shouldScroll\n .then(function (shouldScroll) {\n scrollToPosition((shouldScroll), position);\n })\n .catch(function (err) {\n if (process.env.NODE_ENV !== 'production') {\n assert(false, err.toString());\n }\n });\n } else {\n scrollToPosition(shouldScroll, position);\n }\n });\n}\n\nfunction saveScrollPosition () {\n var key = getStateKey();\n if (key) {\n positionStore[key] = {\n x: window.pageXOffset,\n y: window.pageYOffset\n };\n }\n}\n\nfunction handlePopState (e) {\n saveScrollPosition();\n if (e.state && e.state.key) {\n setStateKey(e.state.key);\n }\n}\n\nfunction getScrollPosition () {\n var key = getStateKey();\n if (key) {\n return positionStore[key]\n }\n}\n\nfunction getElementPosition (el, offset) {\n var docEl = document.documentElement;\n var docRect = docEl.getBoundingClientRect();\n var elRect = el.getBoundingClientRect();\n return {\n x: elRect.left - docRect.left - offset.x,\n y: elRect.top - docRect.top - offset.y\n }\n}\n\nfunction isValidPosition (obj) {\n return isNumber(obj.x) || isNumber(obj.y)\n}\n\nfunction normalizePosition (obj) {\n return {\n x: isNumber(obj.x) ? obj.x : window.pageXOffset,\n y: isNumber(obj.y) ? obj.y : window.pageYOffset\n }\n}\n\nfunction normalizeOffset (obj) {\n return {\n x: isNumber(obj.x) ? obj.x : 0,\n y: isNumber(obj.y) ? obj.y : 0\n }\n}\n\nfunction isNumber (v) {\n return typeof v === 'number'\n}\n\nvar hashStartsWithNumberRE = /^#\\d/;\n\nfunction scrollToPosition (shouldScroll, position) {\n var isObject = typeof shouldScroll === 'object';\n if (isObject && typeof shouldScroll.selector === 'string') {\n // getElementById would still fail if the selector contains a more complicated query like #main[data-attr]\n // but at the same time, it doesn't make much sense to select an element with an id and an extra selector\n var el = hashStartsWithNumberRE.test(shouldScroll.selector) // $flow-disable-line\n ? document.getElementById(shouldScroll.selector.slice(1)) // $flow-disable-line\n : document.querySelector(shouldScroll.selector);\n\n if (el) {\n var offset =\n shouldScroll.offset && typeof shouldScroll.offset === 'object'\n ? shouldScroll.offset\n : {};\n offset = normalizeOffset(offset);\n position = getElementPosition(el, offset);\n } else if (isValidPosition(shouldScroll)) {\n position = normalizePosition(shouldScroll);\n }\n } else if (isObject && isValidPosition(shouldScroll)) {\n position = normalizePosition(shouldScroll);\n }\n\n if (position) {\n // $flow-disable-line\n if ('scrollBehavior' in document.documentElement.style) {\n window.scrollTo({\n left: position.x,\n top: position.y,\n // $flow-disable-line\n behavior: shouldScroll.behavior\n });\n } else {\n window.scrollTo(position.x, position.y);\n }\n }\n}\n\n/* */\n\nvar supportsPushState =\n inBrowser &&\n (function () {\n var ua = window.navigator.userAgent;\n\n if (\n (ua.indexOf('Android 2.') !== -1 || ua.indexOf('Android 4.0') !== -1) &&\n ua.indexOf('Mobile Safari') !== -1 &&\n ua.indexOf('Chrome') === -1 &&\n ua.indexOf('Windows Phone') === -1\n ) {\n return false\n }\n\n return window.history && typeof window.history.pushState === 'function'\n })();\n\nfunction pushState (url, replace) {\n saveScrollPosition();\n // try...catch the pushState call to get around Safari\n // DOM Exception 18 where it limits to 100 pushState calls\n var history = window.history;\n try {\n if (replace) {\n // preserve existing history state as it could be overriden by the user\n var stateCopy = extend({}, history.state);\n stateCopy.key = getStateKey();\n history.replaceState(stateCopy, '', url);\n } else {\n history.pushState({ key: setStateKey(genStateKey()) }, '', url);\n }\n } catch (e) {\n window.location[replace ? 'replace' : 'assign'](url);\n }\n}\n\nfunction replaceState (url) {\n pushState(url, true);\n}\n\n// When changing thing, also edit router.d.ts\nvar NavigationFailureType = {\n redirected: 2,\n aborted: 4,\n cancelled: 8,\n duplicated: 16\n};\n\nfunction createNavigationRedirectedError (from, to) {\n return createRouterError(\n from,\n to,\n NavigationFailureType.redirected,\n (\"Redirected when going from \\\"\" + (from.fullPath) + \"\\\" to \\\"\" + (stringifyRoute(\n to\n )) + \"\\\" via a navigation guard.\")\n )\n}\n\nfunction createNavigationDuplicatedError (from, to) {\n var error = createRouterError(\n from,\n to,\n NavigationFailureType.duplicated,\n (\"Avoided redundant navigation to current location: \\\"\" + (from.fullPath) + \"\\\".\")\n );\n // backwards compatible with the first introduction of Errors\n error.name = 'NavigationDuplicated';\n return error\n}\n\nfunction createNavigationCancelledError (from, to) {\n return createRouterError(\n from,\n to,\n NavigationFailureType.cancelled,\n (\"Navigation cancelled from \\\"\" + (from.fullPath) + \"\\\" to \\\"\" + (to.fullPath) + \"\\\" with a new navigation.\")\n )\n}\n\nfunction createNavigationAbortedError (from, to) {\n return createRouterError(\n from,\n to,\n NavigationFailureType.aborted,\n (\"Navigation aborted from \\\"\" + (from.fullPath) + \"\\\" to \\\"\" + (to.fullPath) + \"\\\" via a navigation guard.\")\n )\n}\n\nfunction createRouterError (from, to, type, message) {\n var error = new Error(message);\n error._isRouter = true;\n error.from = from;\n error.to = to;\n error.type = type;\n\n return error\n}\n\nvar propertiesToLog = ['params', 'query', 'hash'];\n\nfunction stringifyRoute (to) {\n if (typeof to === 'string') { return to }\n if ('path' in to) { return to.path }\n var location = {};\n propertiesToLog.forEach(function (key) {\n if (key in to) { location[key] = to[key]; }\n });\n return JSON.stringify(location, null, 2)\n}\n\nfunction isError (err) {\n return Object.prototype.toString.call(err).indexOf('Error') > -1\n}\n\nfunction isNavigationFailure (err, errorType) {\n return (\n isError(err) &&\n err._isRouter &&\n (errorType == null || err.type === errorType)\n )\n}\n\n/* */\n\nfunction runQueue (queue, fn, cb) {\n var step = function (index) {\n if (index >= queue.length) {\n cb();\n } else {\n if (queue[index]) {\n fn(queue[index], function () {\n step(index + 1);\n });\n } else {\n step(index + 1);\n }\n }\n };\n step(0);\n}\n\n/* */\n\nfunction resolveAsyncComponents (matched) {\n return function (to, from, next) {\n var hasAsync = false;\n var pending = 0;\n var error = null;\n\n flatMapComponents(matched, function (def, _, match, key) {\n // if it's a function and doesn't have cid attached,\n // assume it's an async component resolve function.\n // we are not using Vue's default async resolving mechanism because\n // we want to halt the navigation until the incoming component has been\n // resolved.\n if (typeof def === 'function' && def.cid === undefined) {\n hasAsync = true;\n pending++;\n\n var resolve = once(function (resolvedDef) {\n if (isESModule(resolvedDef)) {\n resolvedDef = resolvedDef.default;\n }\n // save resolved on async factory in case it's used elsewhere\n def.resolved = typeof resolvedDef === 'function'\n ? resolvedDef\n : _Vue.extend(resolvedDef);\n match.components[key] = resolvedDef;\n pending--;\n if (pending <= 0) {\n next();\n }\n });\n\n var reject = once(function (reason) {\n var msg = \"Failed to resolve async component \" + key + \": \" + reason;\n process.env.NODE_ENV !== 'production' && warn(false, msg);\n if (!error) {\n error = isError(reason)\n ? reason\n : new Error(msg);\n next(error);\n }\n });\n\n var res;\n try {\n res = def(resolve, reject);\n } catch (e) {\n reject(e);\n }\n if (res) {\n if (typeof res.then === 'function') {\n res.then(resolve, reject);\n } else {\n // new syntax in Vue 2.3\n var comp = res.component;\n if (comp && typeof comp.then === 'function') {\n comp.then(resolve, reject);\n }\n }\n }\n }\n });\n\n if (!hasAsync) { next(); }\n }\n}\n\nfunction flatMapComponents (\n matched,\n fn\n) {\n return flatten(matched.map(function (m) {\n return Object.keys(m.components).map(function (key) { return fn(\n m.components[key],\n m.instances[key],\n m, key\n ); })\n }))\n}\n\nfunction flatten (arr) {\n return Array.prototype.concat.apply([], arr)\n}\n\nvar hasSymbol =\n typeof Symbol === 'function' &&\n typeof Symbol.toStringTag === 'symbol';\n\nfunction isESModule (obj) {\n return obj.__esModule || (hasSymbol && obj[Symbol.toStringTag] === 'Module')\n}\n\n// in Webpack 2, require.ensure now also returns a Promise\n// so the resolve/reject functions may get called an extra time\n// if the user uses an arrow function shorthand that happens to\n// return that Promise.\nfunction once (fn) {\n var called = false;\n return function () {\n var args = [], len = arguments.length;\n while ( len-- ) args[ len ] = arguments[ len ];\n\n if (called) { return }\n called = true;\n return fn.apply(this, args)\n }\n}\n\n/* */\n\nvar History = function History (router, base) {\n this.router = router;\n this.base = normalizeBase(base);\n // start with a route object that stands for \"nowhere\"\n this.current = START;\n this.pending = null;\n this.ready = false;\n this.readyCbs = [];\n this.readyErrorCbs = [];\n this.errorCbs = [];\n this.listeners = [];\n};\n\nHistory.prototype.listen = function listen (cb) {\n this.cb = cb;\n};\n\nHistory.prototype.onReady = function onReady (cb, errorCb) {\n if (this.ready) {\n cb();\n } else {\n this.readyCbs.push(cb);\n if (errorCb) {\n this.readyErrorCbs.push(errorCb);\n }\n }\n};\n\nHistory.prototype.onError = function onError (errorCb) {\n this.errorCbs.push(errorCb);\n};\n\nHistory.prototype.transitionTo = function transitionTo (\n location,\n onComplete,\n onAbort\n) {\n var this$1$1 = this;\n\n var route;\n // catch redirect option https://github.com/vuejs/vue-router/issues/3201\n try {\n route = this.router.match(location, this.current);\n } catch (e) {\n this.errorCbs.forEach(function (cb) {\n cb(e);\n });\n // Exception should still be thrown\n throw e\n }\n var prev = this.current;\n this.confirmTransition(\n route,\n function () {\n this$1$1.updateRoute(route);\n onComplete && onComplete(route);\n this$1$1.ensureURL();\n this$1$1.router.afterHooks.forEach(function (hook) {\n hook && hook(route, prev);\n });\n\n // fire ready cbs once\n if (!this$1$1.ready) {\n this$1$1.ready = true;\n this$1$1.readyCbs.forEach(function (cb) {\n cb(route);\n });\n }\n },\n function (err) {\n if (onAbort) {\n onAbort(err);\n }\n if (err && !this$1$1.ready) {\n // Initial redirection should not mark the history as ready yet\n // because it's triggered by the redirection instead\n // https://github.com/vuejs/vue-router/issues/3225\n // https://github.com/vuejs/vue-router/issues/3331\n if (!isNavigationFailure(err, NavigationFailureType.redirected) || prev !== START) {\n this$1$1.ready = true;\n this$1$1.readyErrorCbs.forEach(function (cb) {\n cb(err);\n });\n }\n }\n }\n );\n};\n\nHistory.prototype.confirmTransition = function confirmTransition (route, onComplete, onAbort) {\n var this$1$1 = this;\n\n var current = this.current;\n this.pending = route;\n var abort = function (err) {\n // changed after adding errors with\n // https://github.com/vuejs/vue-router/pull/3047 before that change,\n // redirect and aborted navigation would produce an err == null\n if (!isNavigationFailure(err) && isError(err)) {\n if (this$1$1.errorCbs.length) {\n this$1$1.errorCbs.forEach(function (cb) {\n cb(err);\n });\n } else {\n if (process.env.NODE_ENV !== 'production') {\n warn(false, 'uncaught error during route navigation:');\n }\n console.error(err);\n }\n }\n onAbort && onAbort(err);\n };\n var lastRouteIndex = route.matched.length - 1;\n var lastCurrentIndex = current.matched.length - 1;\n if (\n isSameRoute(route, current) &&\n // in the case the route map has been dynamically appended to\n lastRouteIndex === lastCurrentIndex &&\n route.matched[lastRouteIndex] === current.matched[lastCurrentIndex]\n ) {\n this.ensureURL();\n if (route.hash) {\n handleScroll(this.router, current, route, false);\n }\n return abort(createNavigationDuplicatedError(current, route))\n }\n\n var ref = resolveQueue(\n this.current.matched,\n route.matched\n );\n var updated = ref.updated;\n var deactivated = ref.deactivated;\n var activated = ref.activated;\n\n var queue = [].concat(\n // in-component leave guards\n extractLeaveGuards(deactivated),\n // global before hooks\n this.router.beforeHooks,\n // in-component update hooks\n extractUpdateHooks(updated),\n // in-config enter guards\n activated.map(function (m) { return m.beforeEnter; }),\n // async components\n resolveAsyncComponents(activated)\n );\n\n var iterator = function (hook, next) {\n if (this$1$1.pending !== route) {\n return abort(createNavigationCancelledError(current, route))\n }\n try {\n hook(route, current, function (to) {\n if (to === false) {\n // next(false) -> abort navigation, ensure current URL\n this$1$1.ensureURL(true);\n abort(createNavigationAbortedError(current, route));\n } else if (isError(to)) {\n this$1$1.ensureURL(true);\n abort(to);\n } else if (\n typeof to === 'string' ||\n (typeof to === 'object' &&\n (typeof to.path === 'string' || typeof to.name === 'string'))\n ) {\n // next('/') or next({ path: '/' }) -> redirect\n abort(createNavigationRedirectedError(current, route));\n if (typeof to === 'object' && to.replace) {\n this$1$1.replace(to);\n } else {\n this$1$1.push(to);\n }\n } else {\n // confirm transition and pass on the value\n next(to);\n }\n });\n } catch (e) {\n abort(e);\n }\n };\n\n runQueue(queue, iterator, function () {\n // wait until async components are resolved before\n // extracting in-component enter guards\n var enterGuards = extractEnterGuards(activated);\n var queue = enterGuards.concat(this$1$1.router.resolveHooks);\n runQueue(queue, iterator, function () {\n if (this$1$1.pending !== route) {\n return abort(createNavigationCancelledError(current, route))\n }\n this$1$1.pending = null;\n onComplete(route);\n if (this$1$1.router.app) {\n this$1$1.router.app.$nextTick(function () {\n handleRouteEntered(route);\n });\n }\n });\n });\n};\n\nHistory.prototype.updateRoute = function updateRoute (route) {\n this.current = route;\n this.cb && this.cb(route);\n};\n\nHistory.prototype.setupListeners = function setupListeners () {\n // Default implementation is empty\n};\n\nHistory.prototype.teardown = function teardown () {\n // clean up event listeners\n // https://github.com/vuejs/vue-router/issues/2341\n this.listeners.forEach(function (cleanupListener) {\n cleanupListener();\n });\n this.listeners = [];\n\n // reset current history route\n // https://github.com/vuejs/vue-router/issues/3294\n this.current = START;\n this.pending = null;\n};\n\nfunction normalizeBase (base) {\n if (!base) {\n if (inBrowser) {\n // respect tag\n var baseEl = document.querySelector('base');\n base = (baseEl && baseEl.getAttribute('href')) || '/';\n // strip full URL origin\n base = base.replace(/^https?:\\/\\/[^\\/]+/, '');\n } else {\n base = '/';\n }\n }\n // make sure there's the starting slash\n if (base.charAt(0) !== '/') {\n base = '/' + base;\n }\n // remove trailing slash\n return base.replace(/\\/$/, '')\n}\n\nfunction resolveQueue (\n current,\n next\n) {\n var i;\n var max = Math.max(current.length, next.length);\n for (i = 0; i < max; i++) {\n if (current[i] !== next[i]) {\n break\n }\n }\n return {\n updated: next.slice(0, i),\n activated: next.slice(i),\n deactivated: current.slice(i)\n }\n}\n\nfunction extractGuards (\n records,\n name,\n bind,\n reverse\n) {\n var guards = flatMapComponents(records, function (def, instance, match, key) {\n var guard = extractGuard(def, name);\n if (guard) {\n return Array.isArray(guard)\n ? guard.map(function (guard) { return bind(guard, instance, match, key); })\n : bind(guard, instance, match, key)\n }\n });\n return flatten(reverse ? guards.reverse() : guards)\n}\n\nfunction extractGuard (\n def,\n key\n) {\n if (typeof def !== 'function') {\n // extend now so that global mixins are applied.\n def = _Vue.extend(def);\n }\n return def.options[key]\n}\n\nfunction extractLeaveGuards (deactivated) {\n return extractGuards(deactivated, 'beforeRouteLeave', bindGuard, true)\n}\n\nfunction extractUpdateHooks (updated) {\n return extractGuards(updated, 'beforeRouteUpdate', bindGuard)\n}\n\nfunction bindGuard (guard, instance) {\n if (instance) {\n return function boundRouteGuard () {\n return guard.apply(instance, arguments)\n }\n }\n}\n\nfunction extractEnterGuards (\n activated\n) {\n return extractGuards(\n activated,\n 'beforeRouteEnter',\n function (guard, _, match, key) {\n return bindEnterGuard(guard, match, key)\n }\n )\n}\n\nfunction bindEnterGuard (\n guard,\n match,\n key\n) {\n return function routeEnterGuard (to, from, next) {\n return guard(to, from, function (cb) {\n if (typeof cb === 'function') {\n if (!match.enteredCbs[key]) {\n match.enteredCbs[key] = [];\n }\n match.enteredCbs[key].push(cb);\n }\n next(cb);\n })\n }\n}\n\n/* */\n\nvar HTML5History = /*@__PURE__*/(function (History) {\n function HTML5History (router, base) {\n History.call(this, router, base);\n\n this._startLocation = getLocation(this.base);\n }\n\n if ( History ) HTML5History.__proto__ = History;\n HTML5History.prototype = Object.create( History && History.prototype );\n HTML5History.prototype.constructor = HTML5History;\n\n HTML5History.prototype.setupListeners = function setupListeners () {\n var this$1$1 = this;\n\n if (this.listeners.length > 0) {\n return\n }\n\n var router = this.router;\n var expectScroll = router.options.scrollBehavior;\n var supportsScroll = supportsPushState && expectScroll;\n\n if (supportsScroll) {\n this.listeners.push(setupScroll());\n }\n\n var handleRoutingEvent = function () {\n var current = this$1$1.current;\n\n // Avoiding first `popstate` event dispatched in some browsers but first\n // history route not updated since async guard at the same time.\n var location = getLocation(this$1$1.base);\n if (this$1$1.current === START && location === this$1$1._startLocation) {\n return\n }\n\n this$1$1.transitionTo(location, function (route) {\n if (supportsScroll) {\n handleScroll(router, route, current, true);\n }\n });\n };\n window.addEventListener('popstate', handleRoutingEvent);\n this.listeners.push(function () {\n window.removeEventListener('popstate', handleRoutingEvent);\n });\n };\n\n HTML5History.prototype.go = function go (n) {\n window.history.go(n);\n };\n\n HTML5History.prototype.push = function push (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n var ref = this;\n var fromRoute = ref.current;\n this.transitionTo(location, function (route) {\n pushState(cleanPath(this$1$1.base + route.fullPath));\n handleScroll(this$1$1.router, route, fromRoute, false);\n onComplete && onComplete(route);\n }, onAbort);\n };\n\n HTML5History.prototype.replace = function replace (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n var ref = this;\n var fromRoute = ref.current;\n this.transitionTo(location, function (route) {\n replaceState(cleanPath(this$1$1.base + route.fullPath));\n handleScroll(this$1$1.router, route, fromRoute, false);\n onComplete && onComplete(route);\n }, onAbort);\n };\n\n HTML5History.prototype.ensureURL = function ensureURL (push) {\n if (getLocation(this.base) !== this.current.fullPath) {\n var current = cleanPath(this.base + this.current.fullPath);\n push ? pushState(current) : replaceState(current);\n }\n };\n\n HTML5History.prototype.getCurrentLocation = function getCurrentLocation () {\n return getLocation(this.base)\n };\n\n return HTML5History;\n}(History));\n\nfunction getLocation (base) {\n var path = window.location.pathname;\n var pathLowerCase = path.toLowerCase();\n var baseLowerCase = base.toLowerCase();\n // base=\"/a\" shouldn't turn path=\"/app\" into \"/a/pp\"\n // https://github.com/vuejs/vue-router/issues/3555\n // so we ensure the trailing slash in the base\n if (base && ((pathLowerCase === baseLowerCase) ||\n (pathLowerCase.indexOf(cleanPath(baseLowerCase + '/')) === 0))) {\n path = path.slice(base.length);\n }\n return (path || '/') + window.location.search + window.location.hash\n}\n\n/* */\n\nvar HashHistory = /*@__PURE__*/(function (History) {\n function HashHistory (router, base, fallback) {\n History.call(this, router, base);\n // check history fallback deeplinking\n if (fallback && checkFallback(this.base)) {\n return\n }\n ensureSlash();\n }\n\n if ( History ) HashHistory.__proto__ = History;\n HashHistory.prototype = Object.create( History && History.prototype );\n HashHistory.prototype.constructor = HashHistory;\n\n // this is delayed until the app mounts\n // to avoid the hashchange listener being fired too early\n HashHistory.prototype.setupListeners = function setupListeners () {\n var this$1$1 = this;\n\n if (this.listeners.length > 0) {\n return\n }\n\n var router = this.router;\n var expectScroll = router.options.scrollBehavior;\n var supportsScroll = supportsPushState && expectScroll;\n\n if (supportsScroll) {\n this.listeners.push(setupScroll());\n }\n\n var handleRoutingEvent = function () {\n var current = this$1$1.current;\n if (!ensureSlash()) {\n return\n }\n this$1$1.transitionTo(getHash(), function (route) {\n if (supportsScroll) {\n handleScroll(this$1$1.router, route, current, true);\n }\n if (!supportsPushState) {\n replaceHash(route.fullPath);\n }\n });\n };\n var eventType = supportsPushState ? 'popstate' : 'hashchange';\n window.addEventListener(\n eventType,\n handleRoutingEvent\n );\n this.listeners.push(function () {\n window.removeEventListener(eventType, handleRoutingEvent);\n });\n };\n\n HashHistory.prototype.push = function push (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n var ref = this;\n var fromRoute = ref.current;\n this.transitionTo(\n location,\n function (route) {\n pushHash(route.fullPath);\n handleScroll(this$1$1.router, route, fromRoute, false);\n onComplete && onComplete(route);\n },\n onAbort\n );\n };\n\n HashHistory.prototype.replace = function replace (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n var ref = this;\n var fromRoute = ref.current;\n this.transitionTo(\n location,\n function (route) {\n replaceHash(route.fullPath);\n handleScroll(this$1$1.router, route, fromRoute, false);\n onComplete && onComplete(route);\n },\n onAbort\n );\n };\n\n HashHistory.prototype.go = function go (n) {\n window.history.go(n);\n };\n\n HashHistory.prototype.ensureURL = function ensureURL (push) {\n var current = this.current.fullPath;\n if (getHash() !== current) {\n push ? pushHash(current) : replaceHash(current);\n }\n };\n\n HashHistory.prototype.getCurrentLocation = function getCurrentLocation () {\n return getHash()\n };\n\n return HashHistory;\n}(History));\n\nfunction checkFallback (base) {\n var location = getLocation(base);\n if (!/^\\/#/.test(location)) {\n window.location.replace(cleanPath(base + '/#' + location));\n return true\n }\n}\n\nfunction ensureSlash () {\n var path = getHash();\n if (path.charAt(0) === '/') {\n return true\n }\n replaceHash('/' + path);\n return false\n}\n\nfunction getHash () {\n // We can't use window.location.hash here because it's not\n // consistent across browsers - Firefox will pre-decode it!\n var href = window.location.href;\n var index = href.indexOf('#');\n // empty path\n if (index < 0) { return '' }\n\n href = href.slice(index + 1);\n\n return href\n}\n\nfunction getUrl (path) {\n var href = window.location.href;\n var i = href.indexOf('#');\n var base = i >= 0 ? href.slice(0, i) : href;\n return (base + \"#\" + path)\n}\n\nfunction pushHash (path) {\n if (supportsPushState) {\n pushState(getUrl(path));\n } else {\n window.location.hash = path;\n }\n}\n\nfunction replaceHash (path) {\n if (supportsPushState) {\n replaceState(getUrl(path));\n } else {\n window.location.replace(getUrl(path));\n }\n}\n\n/* */\n\nvar AbstractHistory = /*@__PURE__*/(function (History) {\n function AbstractHistory (router, base) {\n History.call(this, router, base);\n this.stack = [];\n this.index = -1;\n }\n\n if ( History ) AbstractHistory.__proto__ = History;\n AbstractHistory.prototype = Object.create( History && History.prototype );\n AbstractHistory.prototype.constructor = AbstractHistory;\n\n AbstractHistory.prototype.push = function push (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n this.transitionTo(\n location,\n function (route) {\n this$1$1.stack = this$1$1.stack.slice(0, this$1$1.index + 1).concat(route);\n this$1$1.index++;\n onComplete && onComplete(route);\n },\n onAbort\n );\n };\n\n AbstractHistory.prototype.replace = function replace (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n this.transitionTo(\n location,\n function (route) {\n this$1$1.stack = this$1$1.stack.slice(0, this$1$1.index).concat(route);\n onComplete && onComplete(route);\n },\n onAbort\n );\n };\n\n AbstractHistory.prototype.go = function go (n) {\n var this$1$1 = this;\n\n var targetIndex = this.index + n;\n if (targetIndex < 0 || targetIndex >= this.stack.length) {\n return\n }\n var route = this.stack[targetIndex];\n this.confirmTransition(\n route,\n function () {\n var prev = this$1$1.current;\n this$1$1.index = targetIndex;\n this$1$1.updateRoute(route);\n this$1$1.router.afterHooks.forEach(function (hook) {\n hook && hook(route, prev);\n });\n },\n function (err) {\n if (isNavigationFailure(err, NavigationFailureType.duplicated)) {\n this$1$1.index = targetIndex;\n }\n }\n );\n };\n\n AbstractHistory.prototype.getCurrentLocation = function getCurrentLocation () {\n var current = this.stack[this.stack.length - 1];\n return current ? current.fullPath : '/'\n };\n\n AbstractHistory.prototype.ensureURL = function ensureURL () {\n // noop\n };\n\n return AbstractHistory;\n}(History));\n\n/* */\n\n\n\nvar VueRouter = function VueRouter (options) {\n if ( options === void 0 ) options = {};\n\n if (process.env.NODE_ENV !== 'production') {\n warn(this instanceof VueRouter, \"Router must be called with the new operator.\");\n }\n this.app = null;\n this.apps = [];\n this.options = options;\n this.beforeHooks = [];\n this.resolveHooks = [];\n this.afterHooks = [];\n this.matcher = createMatcher(options.routes || [], this);\n\n var mode = options.mode || 'hash';\n this.fallback =\n mode === 'history' && !supportsPushState && options.fallback !== false;\n if (this.fallback) {\n mode = 'hash';\n }\n if (!inBrowser) {\n mode = 'abstract';\n }\n this.mode = mode;\n\n switch (mode) {\n case 'history':\n this.history = new HTML5History(this, options.base);\n break\n case 'hash':\n this.history = new HashHistory(this, options.base, this.fallback);\n break\n case 'abstract':\n this.history = new AbstractHistory(this, options.base);\n break\n default:\n if (process.env.NODE_ENV !== 'production') {\n assert(false, (\"invalid mode: \" + mode));\n }\n }\n};\n\nvar prototypeAccessors = { currentRoute: { configurable: true } };\n\nVueRouter.prototype.match = function match (raw, current, redirectedFrom) {\n return this.matcher.match(raw, current, redirectedFrom)\n};\n\nprototypeAccessors.currentRoute.get = function () {\n return this.history && this.history.current\n};\n\nVueRouter.prototype.init = function init (app /* Vue component instance */) {\n var this$1$1 = this;\n\n process.env.NODE_ENV !== 'production' &&\n assert(\n install.installed,\n \"not installed. Make sure to call `Vue.use(VueRouter)` \" +\n \"before creating root instance.\"\n );\n\n this.apps.push(app);\n\n // set up app destroyed handler\n // https://github.com/vuejs/vue-router/issues/2639\n app.$once('hook:destroyed', function () {\n // clean out app from this.apps array once destroyed\n var index = this$1$1.apps.indexOf(app);\n if (index > -1) { this$1$1.apps.splice(index, 1); }\n // ensure we still have a main app or null if no apps\n // we do not release the router so it can be reused\n if (this$1$1.app === app) { this$1$1.app = this$1$1.apps[0] || null; }\n\n if (!this$1$1.app) { this$1$1.history.teardown(); }\n });\n\n // main app previously initialized\n // return as we don't need to set up new history listener\n if (this.app) {\n return\n }\n\n this.app = app;\n\n var history = this.history;\n\n if (history instanceof HTML5History || history instanceof HashHistory) {\n var handleInitialScroll = function (routeOrError) {\n var from = history.current;\n var expectScroll = this$1$1.options.scrollBehavior;\n var supportsScroll = supportsPushState && expectScroll;\n\n if (supportsScroll && 'fullPath' in routeOrError) {\n handleScroll(this$1$1, routeOrError, from, false);\n }\n };\n var setupListeners = function (routeOrError) {\n history.setupListeners();\n handleInitialScroll(routeOrError);\n };\n history.transitionTo(\n history.getCurrentLocation(),\n setupListeners,\n setupListeners\n );\n }\n\n history.listen(function (route) {\n this$1$1.apps.forEach(function (app) {\n app._route = route;\n });\n });\n};\n\nVueRouter.prototype.beforeEach = function beforeEach (fn) {\n return registerHook(this.beforeHooks, fn)\n};\n\nVueRouter.prototype.beforeResolve = function beforeResolve (fn) {\n return registerHook(this.resolveHooks, fn)\n};\n\nVueRouter.prototype.afterEach = function afterEach (fn) {\n return registerHook(this.afterHooks, fn)\n};\n\nVueRouter.prototype.onReady = function onReady (cb, errorCb) {\n this.history.onReady(cb, errorCb);\n};\n\nVueRouter.prototype.onError = function onError (errorCb) {\n this.history.onError(errorCb);\n};\n\nVueRouter.prototype.push = function push (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n // $flow-disable-line\n if (!onComplete && !onAbort && typeof Promise !== 'undefined') {\n return new Promise(function (resolve, reject) {\n this$1$1.history.push(location, resolve, reject);\n })\n } else {\n this.history.push(location, onComplete, onAbort);\n }\n};\n\nVueRouter.prototype.replace = function replace (location, onComplete, onAbort) {\n var this$1$1 = this;\n\n // $flow-disable-line\n if (!onComplete && !onAbort && typeof Promise !== 'undefined') {\n return new Promise(function (resolve, reject) {\n this$1$1.history.replace(location, resolve, reject);\n })\n } else {\n this.history.replace(location, onComplete, onAbort);\n }\n};\n\nVueRouter.prototype.go = function go (n) {\n this.history.go(n);\n};\n\nVueRouter.prototype.back = function back () {\n this.go(-1);\n};\n\nVueRouter.prototype.forward = function forward () {\n this.go(1);\n};\n\nVueRouter.prototype.getMatchedComponents = function getMatchedComponents (to) {\n var route = to\n ? to.matched\n ? to\n : this.resolve(to).route\n : this.currentRoute;\n if (!route) {\n return []\n }\n return [].concat.apply(\n [],\n route.matched.map(function (m) {\n return Object.keys(m.components).map(function (key) {\n return m.components[key]\n })\n })\n )\n};\n\nVueRouter.prototype.resolve = function resolve (\n to,\n current,\n append\n) {\n current = current || this.history.current;\n var location = normalizeLocation(to, current, append, this);\n var route = this.match(location, current);\n var fullPath = route.redirectedFrom || route.fullPath;\n var base = this.history.base;\n var href = createHref(base, fullPath, this.mode);\n return {\n location: location,\n route: route,\n href: href,\n // for backwards compat\n normalizedTo: location,\n resolved: route\n }\n};\n\nVueRouter.prototype.getRoutes = function getRoutes () {\n return this.matcher.getRoutes()\n};\n\nVueRouter.prototype.addRoute = function addRoute (parentOrRoute, route) {\n this.matcher.addRoute(parentOrRoute, route);\n if (this.history.current !== START) {\n this.history.transitionTo(this.history.getCurrentLocation());\n }\n};\n\nVueRouter.prototype.addRoutes = function addRoutes (routes) {\n if (process.env.NODE_ENV !== 'production') {\n warn(false, 'router.addRoutes() is deprecated and has been removed in Vue Router 4. Use router.addRoute() instead.');\n }\n this.matcher.addRoutes(routes);\n if (this.history.current !== START) {\n this.history.transitionTo(this.history.getCurrentLocation());\n }\n};\n\nObject.defineProperties( VueRouter.prototype, prototypeAccessors );\n\nvar VueRouter$1 = VueRouter;\n\nfunction registerHook (list, fn) {\n list.push(fn);\n return function () {\n var i = list.indexOf(fn);\n if (i > -1) { list.splice(i, 1); }\n }\n}\n\nfunction createHref (base, fullPath, mode) {\n var path = mode === 'hash' ? '#' + fullPath : fullPath;\n return base ? cleanPath(base + '/' + path) : path\n}\n\n// We cannot remove this as it would be a breaking change\nVueRouter.install = install;\nVueRouter.version = '3.6.5';\nVueRouter.isNavigationFailure = isNavigationFailure;\nVueRouter.NavigationFailureType = NavigationFailureType;\nVueRouter.START_LOCATION = START;\n\nif (inBrowser && window.Vue) {\n window.Vue.use(VueRouter);\n}\n\nvar version = '3.6.5';\n\nexport { NavigationFailureType, Link as RouterLink, View as RouterView, START as START_LOCATION, VueRouter$1 as default, isNavigationFailure, version };\n","/**\n * vue-class-component v7.2.6\n * (c) 2015-present Evan You\n * @license MIT\n */\nimport Vue from 'vue';\n\nfunction _typeof(obj) {\n if (typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\") {\n _typeof = function (obj) {\n return typeof obj;\n };\n } else {\n _typeof = function (obj) {\n return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj;\n };\n }\n\n return _typeof(obj);\n}\n\nfunction _defineProperty(obj, key, value) {\n if (key in obj) {\n Object.defineProperty(obj, key, {\n value: value,\n enumerable: true,\n configurable: true,\n writable: true\n });\n } else {\n obj[key] = value;\n }\n\n return obj;\n}\n\nfunction _toConsumableArray(arr) {\n return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _nonIterableSpread();\n}\n\nfunction _arrayWithoutHoles(arr) {\n if (Array.isArray(arr)) {\n for (var i = 0, arr2 = new Array(arr.length); i < arr.length; i++) arr2[i] = arr[i];\n\n return arr2;\n }\n}\n\nfunction _iterableToArray(iter) {\n if (Symbol.iterator in Object(iter) || Object.prototype.toString.call(iter) === \"[object Arguments]\") return Array.from(iter);\n}\n\nfunction _nonIterableSpread() {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance\");\n}\n\n// The rational behind the verbose Reflect-feature check below is the fact that there are polyfills\n// which add an implementation for Reflect.defineMetadata but not for Reflect.getOwnMetadataKeys.\n// Without this check consumers will encounter hard to track down runtime errors.\nfunction reflectionIsSupported() {\n return typeof Reflect !== 'undefined' && Reflect.defineMetadata && Reflect.getOwnMetadataKeys;\n}\nfunction copyReflectionMetadata(to, from) {\n forwardMetadata(to, from);\n Object.getOwnPropertyNames(from.prototype).forEach(function (key) {\n forwardMetadata(to.prototype, from.prototype, key);\n });\n Object.getOwnPropertyNames(from).forEach(function (key) {\n forwardMetadata(to, from, key);\n });\n}\n\nfunction forwardMetadata(to, from, propertyKey) {\n var metaKeys = propertyKey ? Reflect.getOwnMetadataKeys(from, propertyKey) : Reflect.getOwnMetadataKeys(from);\n metaKeys.forEach(function (metaKey) {\n var metadata = propertyKey ? Reflect.getOwnMetadata(metaKey, from, propertyKey) : Reflect.getOwnMetadata(metaKey, from);\n\n if (propertyKey) {\n Reflect.defineMetadata(metaKey, metadata, to, propertyKey);\n } else {\n Reflect.defineMetadata(metaKey, metadata, to);\n }\n });\n}\n\nvar fakeArray = {\n __proto__: []\n};\nvar hasProto = fakeArray instanceof Array;\nfunction createDecorator(factory) {\n return function (target, key, index) {\n var Ctor = typeof target === 'function' ? target : target.constructor;\n\n if (!Ctor.__decorators__) {\n Ctor.__decorators__ = [];\n }\n\n if (typeof index !== 'number') {\n index = undefined;\n }\n\n Ctor.__decorators__.push(function (options) {\n return factory(options, key, index);\n });\n };\n}\nfunction mixins() {\n for (var _len = arguments.length, Ctors = new Array(_len), _key = 0; _key < _len; _key++) {\n Ctors[_key] = arguments[_key];\n }\n\n return Vue.extend({\n mixins: Ctors\n });\n}\nfunction isPrimitive(value) {\n var type = _typeof(value);\n\n return value == null || type !== 'object' && type !== 'function';\n}\nfunction warn(message) {\n if (typeof console !== 'undefined') {\n console.warn('[vue-class-component] ' + message);\n }\n}\n\nfunction collectDataFromConstructor(vm, Component) {\n // override _init to prevent to init as Vue instance\n var originalInit = Component.prototype._init;\n\n Component.prototype._init = function () {\n var _this = this;\n\n // proxy to actual vm\n var keys = Object.getOwnPropertyNames(vm); // 2.2.0 compat (props are no longer exposed as self properties)\n\n if (vm.$options.props) {\n for (var key in vm.$options.props) {\n if (!vm.hasOwnProperty(key)) {\n keys.push(key);\n }\n }\n }\n\n keys.forEach(function (key) {\n Object.defineProperty(_this, key, {\n get: function get() {\n return vm[key];\n },\n set: function set(value) {\n vm[key] = value;\n },\n configurable: true\n });\n });\n }; // should be acquired class property values\n\n\n var data = new Component(); // restore original _init to avoid memory leak (#209)\n\n Component.prototype._init = originalInit; // create plain data object\n\n var plainData = {};\n Object.keys(data).forEach(function (key) {\n if (data[key] !== undefined) {\n plainData[key] = data[key];\n }\n });\n\n if (process.env.NODE_ENV !== 'production') {\n if (!(Component.prototype instanceof Vue) && Object.keys(plainData).length > 0) {\n warn('Component class must inherit Vue or its descendant class ' + 'when class property is used.');\n }\n }\n\n return plainData;\n}\n\nvar $internalHooks = ['data', 'beforeCreate', 'created', 'beforeMount', 'mounted', 'beforeDestroy', 'destroyed', 'beforeUpdate', 'updated', 'activated', 'deactivated', 'render', 'errorCaptured', 'serverPrefetch' // 2.6\n];\nfunction componentFactory(Component) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};\n options.name = options.name || Component._componentTag || Component.name; // prototype props.\n\n var proto = Component.prototype;\n Object.getOwnPropertyNames(proto).forEach(function (key) {\n if (key === 'constructor') {\n return;\n } // hooks\n\n\n if ($internalHooks.indexOf(key) > -1) {\n options[key] = proto[key];\n return;\n }\n\n var descriptor = Object.getOwnPropertyDescriptor(proto, key);\n\n if (descriptor.value !== void 0) {\n // methods\n if (typeof descriptor.value === 'function') {\n (options.methods || (options.methods = {}))[key] = descriptor.value;\n } else {\n // typescript decorated data\n (options.mixins || (options.mixins = [])).push({\n data: function data() {\n return _defineProperty({}, key, descriptor.value);\n }\n });\n }\n } else if (descriptor.get || descriptor.set) {\n // computed properties\n (options.computed || (options.computed = {}))[key] = {\n get: descriptor.get,\n set: descriptor.set\n };\n }\n });\n (options.mixins || (options.mixins = [])).push({\n data: function data() {\n return collectDataFromConstructor(this, Component);\n }\n }); // decorate options\n\n var decorators = Component.__decorators__;\n\n if (decorators) {\n decorators.forEach(function (fn) {\n return fn(options);\n });\n delete Component.__decorators__;\n } // find super\n\n\n var superProto = Object.getPrototypeOf(Component.prototype);\n var Super = superProto instanceof Vue ? superProto.constructor : Vue;\n var Extended = Super.extend(options);\n forwardStaticMembers(Extended, Component, Super);\n\n if (reflectionIsSupported()) {\n copyReflectionMetadata(Extended, Component);\n }\n\n return Extended;\n}\nvar reservedPropertyNames = [// Unique id\n'cid', // Super Vue constructor\n'super', // Component options that will be used by the component\n'options', 'superOptions', 'extendOptions', 'sealedOptions', // Private assets\n'component', 'directive', 'filter'];\nvar shouldIgnore = {\n prototype: true,\n arguments: true,\n callee: true,\n caller: true\n};\n\nfunction forwardStaticMembers(Extended, Original, Super) {\n // We have to use getOwnPropertyNames since Babel registers methods as non-enumerable\n Object.getOwnPropertyNames(Original).forEach(function (key) {\n // Skip the properties that should not be overwritten\n if (shouldIgnore[key]) {\n return;\n } // Some browsers does not allow reconfigure built-in properties\n\n\n var extendedDescriptor = Object.getOwnPropertyDescriptor(Extended, key);\n\n if (extendedDescriptor && !extendedDescriptor.configurable) {\n return;\n }\n\n var descriptor = Object.getOwnPropertyDescriptor(Original, key); // If the user agent does not support `__proto__` or its family (IE <= 10),\n // the sub class properties may be inherited properties from the super class in TypeScript.\n // We need to exclude such properties to prevent to overwrite\n // the component options object which stored on the extended constructor (See #192).\n // If the value is a referenced value (object or function),\n // we can check equality of them and exclude it if they have the same reference.\n // If it is a primitive value, it will be forwarded for safety.\n\n if (!hasProto) {\n // Only `cid` is explicitly exluded from property forwarding\n // because we cannot detect whether it is a inherited property or not\n // on the no `__proto__` environment even though the property is reserved.\n if (key === 'cid') {\n return;\n }\n\n var superDescriptor = Object.getOwnPropertyDescriptor(Super, key);\n\n if (!isPrimitive(descriptor.value) && superDescriptor && superDescriptor.value === descriptor.value) {\n return;\n }\n } // Warn if the users manually declare reserved properties\n\n\n if (process.env.NODE_ENV !== 'production' && reservedPropertyNames.indexOf(key) >= 0) {\n warn(\"Static property name '\".concat(key, \"' declared on class '\").concat(Original.name, \"' \") + 'conflicts with reserved property name of Vue internal. ' + 'It may cause unexpected behavior of the component. Consider renaming the property.');\n }\n\n Object.defineProperty(Extended, key, descriptor);\n });\n}\n\nfunction Component(options) {\n if (typeof options === 'function') {\n return componentFactory(options);\n }\n\n return function (Component) {\n return componentFactory(Component, options);\n };\n}\n\nComponent.registerHooks = function registerHooks(keys) {\n $internalHooks.push.apply($internalHooks, _toConsumableArray(keys));\n};\n\nexport default Component;\nexport { createDecorator, mixins };\n","var __spreadArrays = (this && this.__spreadArrays) || function () {\n for (var s = 0, i = 0, il = arguments.length; i < il; i++) s += arguments[i].length;\n for (var r = Array(s), k = 0, i = 0; i < il; i++)\n for (var a = arguments[i], j = 0, jl = a.length; j < jl; j++, k++)\n r[k] = a[j];\n return r;\n};\n// Code copied from Vue/src/shared/util.js\nvar hyphenateRE = /\\B([A-Z])/g;\nvar hyphenate = function (str) { return str.replace(hyphenateRE, '-$1').toLowerCase(); };\n/**\n * decorator of an event-emitter function\n * @param event The name of the event\n * @return MethodDecorator\n */\nexport function Emit(event) {\n return function (_target, propertyKey, descriptor) {\n var key = hyphenate(propertyKey);\n var original = descriptor.value;\n descriptor.value = function emitter() {\n var _this = this;\n var args = [];\n for (var _i = 0; _i < arguments.length; _i++) {\n args[_i] = arguments[_i];\n }\n var emit = function (returnValue) {\n var emitName = event || key;\n if (returnValue === undefined) {\n if (args.length === 0) {\n _this.$emit(emitName);\n }\n else if (args.length === 1) {\n _this.$emit(emitName, args[0]);\n }\n else {\n _this.$emit.apply(_this, __spreadArrays([emitName], args));\n }\n }\n else {\n args.unshift(returnValue);\n _this.$emit.apply(_this, __spreadArrays([emitName], args));\n }\n };\n var returnValue = original.apply(this, args);\n if (isPromise(returnValue)) {\n returnValue.then(emit);\n }\n else {\n emit(returnValue);\n }\n return returnValue;\n };\n };\n}\nfunction isPromise(obj) {\n return obj instanceof Promise || (obj && typeof obj.then === 'function');\n}\n","import { createDecorator } from 'vue-class-component';\n/**\n * decorator of an inject\n * @param from key\n * @return PropertyDecorator\n */\nexport function Inject(options) {\n return createDecorator(function (componentOptions, key) {\n if (typeof componentOptions.inject === 'undefined') {\n componentOptions.inject = {};\n }\n if (!Array.isArray(componentOptions.inject)) {\n componentOptions.inject[key] = options || key;\n }\n });\n}\n","export function needToProduceProvide(original) {\n return (typeof original !== 'function' ||\n (!original.managed && !original.managedReactive));\n}\nexport function produceProvide(original) {\n var provide = function () {\n var _this = this;\n var rv = typeof original === 'function' ? original.call(this) : original;\n rv = Object.create(rv || null);\n // set reactive services (propagates previous services if necessary)\n rv[reactiveInjectKey] = Object.create(this[reactiveInjectKey] || {});\n for (var i in provide.managed) {\n rv[provide.managed[i]] = this[i];\n }\n var _loop_1 = function (i) {\n rv[provide.managedReactive[i]] = this_1[i]; // Duplicates the behavior of `@Provide`\n Object.defineProperty(rv[reactiveInjectKey], provide.managedReactive[i], {\n enumerable: true,\n configurable: true,\n get: function () { return _this[i]; },\n });\n };\n var this_1 = this;\n for (var i in provide.managedReactive) {\n _loop_1(i);\n }\n return rv;\n };\n provide.managed = {};\n provide.managedReactive = {};\n return provide;\n}\n/** Used for keying reactive provide/inject properties */\nexport var reactiveInjectKey = '__reactiveInject__';\nexport function inheritInjected(componentOptions) {\n // inject parent reactive services (if any)\n if (!Array.isArray(componentOptions.inject)) {\n componentOptions.inject = componentOptions.inject || {};\n componentOptions.inject[reactiveInjectKey] = {\n from: reactiveInjectKey,\n default: {},\n };\n }\n}\n","/** @see {@link https://github.com/vuejs/vue-class-component/blob/master/src/reflect.ts} */\nvar reflectMetadataIsSupported = typeof Reflect !== 'undefined' && typeof Reflect.getMetadata !== 'undefined';\nexport function applyMetadata(options, target, key) {\n if (reflectMetadataIsSupported) {\n if (!Array.isArray(options) &&\n typeof options !== 'function' &&\n !options.hasOwnProperty('type') &&\n typeof options.type === 'undefined') {\n var type = Reflect.getMetadata('design:type', target, key);\n if (type !== Object) {\n options.type = type;\n }\n }\n }\n}\n","import { createDecorator } from 'vue-class-component';\nimport { applyMetadata } from '../helpers/metadata';\n/**\n * decorator of a prop\n * @param options the options for the prop\n * @return PropertyDecorator | void\n */\nexport function Prop(options) {\n if (options === void 0) { options = {}; }\n return function (target, key) {\n applyMetadata(options, target, key);\n createDecorator(function (componentOptions, k) {\n ;\n (componentOptions.props || (componentOptions.props = {}))[k] = options;\n })(target, key);\n };\n}\n","import { createDecorator } from 'vue-class-component';\nimport { applyMetadata } from '../helpers/metadata';\n/**\n * decorator of a synced prop\n * @param propName the name to interface with from outside, must be different from decorated property\n * @param options the options for the synced prop\n * @return PropertyDecorator | void\n */\nexport function PropSync(propName, options) {\n if (options === void 0) { options = {}; }\n return function (target, key) {\n applyMetadata(options, target, key);\n createDecorator(function (componentOptions, k) {\n ;\n (componentOptions.props || (componentOptions.props = {}))[propName] = options;\n (componentOptions.computed || (componentOptions.computed = {}))[k] = {\n get: function () {\n return this[propName];\n },\n set: function (value) {\n this.$emit(\"update:\" + propName, value);\n },\n };\n })(target, key);\n };\n}\n","import { createDecorator } from 'vue-class-component';\nimport { inheritInjected, needToProduceProvide, produceProvide, } from '../helpers/provideInject';\n/**\n * decorator of a provide\n * @param key key\n * @return PropertyDecorator | void\n */\nexport function Provide(key) {\n return createDecorator(function (componentOptions, k) {\n var provide = componentOptions.provide;\n inheritInjected(componentOptions);\n if (needToProduceProvide(provide)) {\n provide = componentOptions.provide = produceProvide(provide);\n }\n provide.managed[k] = key || k;\n });\n}\n","import { createDecorator } from 'vue-class-component';\n/**\n * decorator of a watch function\n * @param path the path or the expression to observe\n * @param WatchOption\n * @return MethodDecorator\n */\nexport function Watch(path, options) {\n if (options === void 0) { options = {}; }\n var _a = options.deep, deep = _a === void 0 ? false : _a, _b = options.immediate, immediate = _b === void 0 ? false : _b;\n return createDecorator(function (componentOptions, handler) {\n if (typeof componentOptions.watch !== 'object') {\n componentOptions.watch = Object.create(null);\n }\n var watch = componentOptions.watch;\n if (typeof watch[path] === 'object' && !Array.isArray(watch[path])) {\n watch[path] = [watch[path]];\n }\n else if (typeof watch[path] === 'undefined') {\n watch[path] = [];\n }\n watch[path].push({ handler: handler, deep: deep, immediate: immediate });\n });\n}\n","export default function(n){return{all:n=n||new Map,on:function(t,e){var i=n.get(t);i?i.push(e):n.set(t,[e])},off:function(t,e){var i=n.get(t);i&&(e?i.splice(i.indexOf(e)>>>0,1):n.set(t,[]))},emit:function(t,e){var i=n.get(t);i&&i.slice().map(function(n){n(e)}),(i=n.get(\"*\"))&&i.slice().map(function(n){n(t,e)})}}}\n//# sourceMappingURL=mitt.mjs.map\n","import mitt, { Emitter } from \"mitt\";\nimport { SnapPair } from \"./types/snapping\";\nimport { AppListAction } from \"./types\";\nimport { DraggingInfo } from \"./stores/dragDrop\";\n\ntype Events = {\n WINDOW_RESIZE: undefined;\n APP_CHANGES_SAVED: undefined;\n CANVAS_BOX_UPDATE: undefined;\n COLOR_PICKER_MOUNTED: undefined;\n AWAITING_SERVER: boolean;\n SAVING_APP: boolean;\n APP_EDITOR_INITIALIZED: { appId: string };\n APP_SAVE_ERROR: boolean;\n SNAP_PAIRS: SnapPair[];\n CLICK_FROM_HEX_INPUT: boolean;\n UPDATE_FONT_SIZE: number;\n COMPLETE_CORNER_RESIZE: string;\n FONTS_LOADED: string;\n RESET_ARTBOARD: undefined;\n DATA_CONNECTION_CREATED: { uuid: string };\n DATA_CONNECTION_UPDATED: { uuid: string };\n DATA_CONNECTION_SYNCHRONIZED: { uuid: string };\n DATA_TOKEN_DROPPED: {\n widgetId: string;\n conditionId: string;\n draggingInfo: DraggingInfo;\n parentWidgetId: string;\n };\n DATA_TOKEN_ADDED: undefined;\n APPS_LIST_VIEW_VISIBLE: string;\n APPS_LIST_ACTION: { action: AppListAction; uuid: string };\n OPEN_DASHBOARD_APP_MODAL: {\n action: AppListAction;\n uuid: string;\n name?: string;\n };\n CLEAR_UNDO_STACK: undefined;\n CONDITION_CREATED: string;\n DUPLICATE_WIDGET_ACTION: undefined;\n PLAYBACK_BEGIN: undefined;\n PHOTO_DRAG_ENTER: {\n widgetId: string;\n cellIndex?: number;\n };\n PHOTO_DRAG_LEAVE: undefined;\n UNDO_FIRED: undefined;\n FIRE_SAVE: undefined;\n OPEN_DATA_PANEL: undefined;\n TURN_OFF_DROPDOWN: undefined;\n IGNORE_TEXT_WIDGET_LISTENERS: boolean;\n};\n\nexport const EventBus: Emitter = mitt();\n","function _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return self; }\n\nfunction _inheritsLoose(subClass, superClass) { subClass.prototype = Object.create(superClass.prototype); subClass.prototype.constructor = subClass; subClass.__proto__ = superClass; }\n\n/*!\n * GSAP 3.11.1\n * https://greensock.com\n *\n * @license Copyright 2008-2022, GreenSock. All rights reserved.\n * Subject to the terms at https://greensock.com/standard-license or for\n * Club GreenSock members, the agreement issued with that membership.\n * @author: Jack Doyle, jack@greensock.com\n*/\n\n/* eslint-disable */\nvar _config = {\n autoSleep: 120,\n force3D: \"auto\",\n nullTargetWarn: 1,\n units: {\n lineHeight: \"\"\n }\n},\n _defaults = {\n duration: .5,\n overwrite: false,\n delay: 0\n},\n _suppressOverwrites,\n _reverting,\n _context,\n _bigNum = 1e8,\n _tinyNum = 1 / _bigNum,\n _2PI = Math.PI * 2,\n _HALF_PI = _2PI / 4,\n _gsID = 0,\n _sqrt = Math.sqrt,\n _cos = Math.cos,\n _sin = Math.sin,\n _isString = function _isString(value) {\n return typeof value === \"string\";\n},\n _isFunction = function _isFunction(value) {\n return typeof value === \"function\";\n},\n _isNumber = function _isNumber(value) {\n return typeof value === \"number\";\n},\n _isUndefined = function _isUndefined(value) {\n return typeof value === \"undefined\";\n},\n _isObject = function _isObject(value) {\n return typeof value === \"object\";\n},\n _isNotFalse = function _isNotFalse(value) {\n return value !== false;\n},\n _windowExists = function _windowExists() {\n return typeof window !== \"undefined\";\n},\n _isFuncOrString = function _isFuncOrString(value) {\n return _isFunction(value) || _isString(value);\n},\n _isTypedArray = typeof ArrayBuffer === \"function\" && ArrayBuffer.isView || function () {},\n // note: IE10 has ArrayBuffer, but NOT ArrayBuffer.isView().\n_isArray = Array.isArray,\n _strictNumExp = /(?:-?\\.?\\d|\\.)+/gi,\n //only numbers (including negatives and decimals) but NOT relative values.\n_numExp = /[-+=.]*\\d+[.e\\-+]*\\d*[e\\-+]*\\d*/g,\n //finds any numbers, including ones that start with += or -=, negative numbers, and ones in scientific notation like 1e-8.\n_numWithUnitExp = /[-+=.]*\\d+[.e-]*\\d*[a-z%]*/g,\n _complexStringNumExp = /[-+=.]*\\d+\\.?\\d*(?:e-|e\\+)?\\d*/gi,\n //duplicate so that while we're looping through matches from exec(), it doesn't contaminate the lastIndex of _numExp which we use to search for colors too.\n_relExp = /[+-]=-?[.\\d]+/,\n _delimitedValueExp = /[^,'\"\\[\\]\\s]+/gi,\n // previously /[#\\-+.]*\\b[a-z\\d\\-=+%.]+/gi but didn't catch special characters.\n_unitExp = /^[+\\-=e\\s\\d]*\\d+[.\\d]*([a-z]*|%)\\s*$/i,\n _globalTimeline,\n _win,\n _coreInitted,\n _doc,\n _globals = {},\n _installScope = {},\n _coreReady,\n _install = function _install(scope) {\n return (_installScope = _merge(scope, _globals)) && gsap;\n},\n _missingPlugin = function _missingPlugin(property, value) {\n return console.warn(\"Invalid property\", property, \"set to\", value, \"Missing plugin? gsap.registerPlugin()\");\n},\n _warn = function _warn(message, suppress) {\n return !suppress && console.warn(message);\n},\n _addGlobal = function _addGlobal(name, obj) {\n return name && (_globals[name] = obj) && _installScope && (_installScope[name] = obj) || _globals;\n},\n _emptyFunc = function _emptyFunc() {\n return 0;\n},\n _startAtRevertConfig = {\n suppressEvents: true,\n isStart: true\n},\n _revertConfig = {\n suppressEvents: true\n},\n _reservedProps = {},\n _lazyTweens = [],\n _lazyLookup = {},\n _lastRenderedFrame,\n _plugins = {},\n _effects = {},\n _nextGCFrame = 30,\n _harnessPlugins = [],\n _callbackNames = \"\",\n _harness = function _harness(targets) {\n var target = targets[0],\n harnessPlugin,\n i;\n _isObject(target) || _isFunction(target) || (targets = [targets]);\n\n if (!(harnessPlugin = (target._gsap || {}).harness)) {\n // find the first target with a harness. We assume targets passed into an animation will be of similar type, meaning the same kind of harness can be used for them all (performance optimization)\n i = _harnessPlugins.length;\n\n while (i-- && !_harnessPlugins[i].targetTest(target)) {}\n\n harnessPlugin = _harnessPlugins[i];\n }\n\n i = targets.length;\n\n while (i--) {\n targets[i] && (targets[i]._gsap || (targets[i]._gsap = new GSCache(targets[i], harnessPlugin))) || targets.splice(i, 1);\n }\n\n return targets;\n},\n _getCache = function _getCache(target) {\n return target._gsap || _harness(toArray(target))[0]._gsap;\n},\n _getProperty = function _getProperty(target, property, v) {\n return (v = target[property]) && _isFunction(v) ? target[property]() : _isUndefined(v) && target.getAttribute && target.getAttribute(property) || v;\n},\n _forEachName = function _forEachName(names, func) {\n return (names = names.split(\",\")).forEach(func) || names;\n},\n //split a comma-delimited list of names into an array, then run a forEach() function and return the split array (this is just a way to consolidate/shorten some code).\n_round = function _round(value) {\n return Math.round(value * 100000) / 100000 || 0;\n},\n _roundPrecise = function _roundPrecise(value) {\n return Math.round(value * 10000000) / 10000000 || 0;\n},\n // increased precision mostly for timing values.\n_parseRelative = function _parseRelative(start, value) {\n var operator = value.charAt(0),\n end = parseFloat(value.substr(2));\n start = parseFloat(start);\n return operator === \"+\" ? start + end : operator === \"-\" ? start - end : operator === \"*\" ? start * end : start / end;\n},\n _arrayContainsAny = function _arrayContainsAny(toSearch, toFind) {\n //searches one array to find matches for any of the items in the toFind array. As soon as one is found, it returns true. It does NOT return all the matches; it's simply a boolean search.\n var l = toFind.length,\n i = 0;\n\n for (; toSearch.indexOf(toFind[i]) < 0 && ++i < l;) {}\n\n return i < l;\n},\n _lazyRender = function _lazyRender() {\n var l = _lazyTweens.length,\n a = _lazyTweens.slice(0),\n i,\n tween;\n\n _lazyLookup = {};\n _lazyTweens.length = 0;\n\n for (i = 0; i < l; i++) {\n tween = a[i];\n tween && tween._lazy && (tween.render(tween._lazy[0], tween._lazy[1], true)._lazy = 0);\n }\n},\n _lazySafeRender = function _lazySafeRender(animation, time, suppressEvents, force) {\n _lazyTweens.length && _lazyRender();\n animation.render(time, suppressEvents, force || _reverting);\n _lazyTweens.length && _lazyRender(); //in case rendering caused any tweens to lazy-init, we should render them because typically when someone calls seek() or time() or progress(), they expect an immediate render.\n},\n _numericIfPossible = function _numericIfPossible(value) {\n var n = parseFloat(value);\n return (n || n === 0) && (value + \"\").match(_delimitedValueExp).length < 2 ? n : _isString(value) ? value.trim() : value;\n},\n _passThrough = function _passThrough(p) {\n return p;\n},\n _setDefaults = function _setDefaults(obj, defaults) {\n for (var p in defaults) {\n p in obj || (obj[p] = defaults[p]);\n }\n\n return obj;\n},\n _setKeyframeDefaults = function _setKeyframeDefaults(excludeDuration) {\n return function (obj, defaults) {\n for (var p in defaults) {\n p in obj || p === \"duration\" && excludeDuration || p === \"ease\" || (obj[p] = defaults[p]);\n }\n };\n},\n _merge = function _merge(base, toMerge) {\n for (var p in toMerge) {\n base[p] = toMerge[p];\n }\n\n return base;\n},\n _mergeDeep = function _mergeDeep(base, toMerge) {\n for (var p in toMerge) {\n p !== \"__proto__\" && p !== \"constructor\" && p !== \"prototype\" && (base[p] = _isObject(toMerge[p]) ? _mergeDeep(base[p] || (base[p] = {}), toMerge[p]) : toMerge[p]);\n }\n\n return base;\n},\n _copyExcluding = function _copyExcluding(obj, excluding) {\n var copy = {},\n p;\n\n for (p in obj) {\n p in excluding || (copy[p] = obj[p]);\n }\n\n return copy;\n},\n _inheritDefaults = function _inheritDefaults(vars) {\n var parent = vars.parent || _globalTimeline,\n func = vars.keyframes ? _setKeyframeDefaults(_isArray(vars.keyframes)) : _setDefaults;\n\n if (_isNotFalse(vars.inherit)) {\n while (parent) {\n func(vars, parent.vars.defaults);\n parent = parent.parent || parent._dp;\n }\n }\n\n return vars;\n},\n _arraysMatch = function _arraysMatch(a1, a2) {\n var i = a1.length,\n match = i === a2.length;\n\n while (match && i-- && a1[i] === a2[i]) {}\n\n return i < 0;\n},\n _addLinkedListItem = function _addLinkedListItem(parent, child, firstProp, lastProp, sortBy) {\n if (firstProp === void 0) {\n firstProp = \"_first\";\n }\n\n if (lastProp === void 0) {\n lastProp = \"_last\";\n }\n\n var prev = parent[lastProp],\n t;\n\n if (sortBy) {\n t = child[sortBy];\n\n while (prev && prev[sortBy] > t) {\n prev = prev._prev;\n }\n }\n\n if (prev) {\n child._next = prev._next;\n prev._next = child;\n } else {\n child._next = parent[firstProp];\n parent[firstProp] = child;\n }\n\n if (child._next) {\n child._next._prev = child;\n } else {\n parent[lastProp] = child;\n }\n\n child._prev = prev;\n child.parent = child._dp = parent;\n return child;\n},\n _removeLinkedListItem = function _removeLinkedListItem(parent, child, firstProp, lastProp) {\n if (firstProp === void 0) {\n firstProp = \"_first\";\n }\n\n if (lastProp === void 0) {\n lastProp = \"_last\";\n }\n\n var prev = child._prev,\n next = child._next;\n\n if (prev) {\n prev._next = next;\n } else if (parent[firstProp] === child) {\n parent[firstProp] = next;\n }\n\n if (next) {\n next._prev = prev;\n } else if (parent[lastProp] === child) {\n parent[lastProp] = prev;\n }\n\n child._next = child._prev = child.parent = null; // don't delete the _dp just so we can revert if necessary. But parent should be null to indicate the item isn't in a linked list.\n},\n _removeFromParent = function _removeFromParent(child, onlyIfParentHasAutoRemove) {\n child.parent && (!onlyIfParentHasAutoRemove || child.parent.autoRemoveChildren) && child.parent.remove(child);\n child._act = 0;\n},\n _uncache = function _uncache(animation, child) {\n if (animation && (!child || child._end > animation._dur || child._start < 0)) {\n // performance optimization: if a child animation is passed in we should only uncache if that child EXTENDS the animation (its end time is beyond the end)\n var a = animation;\n\n while (a) {\n a._dirty = 1;\n a = a.parent;\n }\n }\n\n return animation;\n},\n _recacheAncestors = function _recacheAncestors(animation) {\n var parent = animation.parent;\n\n while (parent && parent.parent) {\n //sometimes we must force a re-sort of all children and update the duration/totalDuration of all ancestor timelines immediately in case, for example, in the middle of a render loop, one tween alters another tween's timeScale which shoves its startTime before 0, forcing the parent timeline to shift around and shiftChildren() which could affect that next tween's render (startTime). Doesn't matter for the root timeline though.\n parent._dirty = 1;\n parent.totalDuration();\n parent = parent.parent;\n }\n\n return animation;\n},\n _rewindStartAt = function _rewindStartAt(tween, totalTime, suppressEvents, force) {\n return tween._startAt && (_reverting ? tween._startAt.revert(_revertConfig) : tween.vars.immediateRender && !tween.vars.autoRevert || tween._startAt.render(totalTime, true, force));\n},\n _hasNoPausedAncestors = function _hasNoPausedAncestors(animation) {\n return !animation || animation._ts && _hasNoPausedAncestors(animation.parent);\n},\n _elapsedCycleDuration = function _elapsedCycleDuration(animation) {\n return animation._repeat ? _animationCycle(animation._tTime, animation = animation.duration() + animation._rDelay) * animation : 0;\n},\n // feed in the totalTime and cycleDuration and it'll return the cycle (iteration minus 1) and if the playhead is exactly at the very END, it will NOT bump up to the next cycle.\n_animationCycle = function _animationCycle(tTime, cycleDuration) {\n var whole = Math.floor(tTime /= cycleDuration);\n return tTime && whole === tTime ? whole - 1 : whole;\n},\n _parentToChildTotalTime = function _parentToChildTotalTime(parentTime, child) {\n return (parentTime - child._start) * child._ts + (child._ts >= 0 ? 0 : child._dirty ? child.totalDuration() : child._tDur);\n},\n _setEnd = function _setEnd(animation) {\n return animation._end = _roundPrecise(animation._start + (animation._tDur / Math.abs(animation._ts || animation._rts || _tinyNum) || 0));\n},\n _alignPlayhead = function _alignPlayhead(animation, totalTime) {\n // adjusts the animation's _start and _end according to the provided totalTime (only if the parent's smoothChildTiming is true and the animation isn't paused). It doesn't do any rendering or forcing things back into parent timelines, etc. - that's what totalTime() is for.\n var parent = animation._dp;\n\n if (parent && parent.smoothChildTiming && animation._ts) {\n animation._start = _roundPrecise(parent._time - (animation._ts > 0 ? totalTime / animation._ts : ((animation._dirty ? animation.totalDuration() : animation._tDur) - totalTime) / -animation._ts));\n\n _setEnd(animation);\n\n parent._dirty || _uncache(parent, animation); //for performance improvement. If the parent's cache is already dirty, it already took care of marking the ancestors as dirty too, so skip the function call here.\n }\n\n return animation;\n},\n\n/*\n_totalTimeToTime = (clampedTotalTime, duration, repeat, repeatDelay, yoyo) => {\n\tlet cycleDuration = duration + repeatDelay,\n\t\ttime = _round(clampedTotalTime % cycleDuration);\n\tif (time > duration) {\n\t\ttime = duration;\n\t}\n\treturn (yoyo && (~~(clampedTotalTime / cycleDuration) & 1)) ? duration - time : time;\n},\n*/\n_postAddChecks = function _postAddChecks(timeline, child) {\n var t;\n\n if (child._time || child._initted && !child._dur) {\n //in case, for example, the _start is moved on a tween that has already rendered. Imagine it's at its end state, then the startTime is moved WAY later (after the end of this timeline), it should render at its beginning.\n t = _parentToChildTotalTime(timeline.rawTime(), child);\n\n if (!child._dur || _clamp(0, child.totalDuration(), t) - child._tTime > _tinyNum) {\n child.render(t, true);\n }\n } //if the timeline has already ended but the inserted tween/timeline extends the duration, we should enable this timeline again so that it renders properly. We should also align the playhead with the parent timeline's when appropriate.\n\n\n if (_uncache(timeline, child)._dp && timeline._initted && timeline._time >= timeline._dur && timeline._ts) {\n //in case any of the ancestors had completed but should now be enabled...\n if (timeline._dur < timeline.duration()) {\n t = timeline;\n\n while (t._dp) {\n t.rawTime() >= 0 && t.totalTime(t._tTime); //moves the timeline (shifts its startTime) if necessary, and also enables it. If it's currently zero, though, it may not be scheduled to render until later so there's no need to force it to align with the current playhead position. Only move to catch up with the playhead.\n\n t = t._dp;\n }\n }\n\n timeline._zTime = -_tinyNum; // helps ensure that the next render() will be forced (crossingStart = true in render()), even if the duration hasn't changed (we're adding a child which would need to get rendered). Definitely an edge case. Note: we MUST do this AFTER the loop above where the totalTime() might trigger a render() because this _addToTimeline() method gets called from the Animation constructor, BEFORE tweens even record their targets, etc. so we wouldn't want things to get triggered in the wrong order.\n }\n},\n _addToTimeline = function _addToTimeline(timeline, child, position, skipChecks) {\n child.parent && _removeFromParent(child);\n child._start = _roundPrecise((_isNumber(position) ? position : position || timeline !== _globalTimeline ? _parsePosition(timeline, position, child) : timeline._time) + child._delay);\n child._end = _roundPrecise(child._start + (child.totalDuration() / Math.abs(child.timeScale()) || 0));\n\n _addLinkedListItem(timeline, child, \"_first\", \"_last\", timeline._sort ? \"_start\" : 0);\n\n _isFromOrFromStart(child) || (timeline._recent = child);\n skipChecks || _postAddChecks(timeline, child);\n timeline._ts < 0 && _alignPlayhead(timeline, timeline._tTime); // if the timeline is reversed and the new child makes it longer, we may need to adjust the parent's _start (push it back)\n\n return timeline;\n},\n _scrollTrigger = function _scrollTrigger(animation, trigger) {\n return (_globals.ScrollTrigger || _missingPlugin(\"scrollTrigger\", trigger)) && _globals.ScrollTrigger.create(trigger, animation);\n},\n _attemptInitTween = function _attemptInitTween(tween, totalTime, force, suppressEvents) {\n _initTween(tween, totalTime);\n\n if (!tween._initted) {\n return 1;\n }\n\n if (!force && tween._pt && (tween._dur && tween.vars.lazy !== false || !tween._dur && tween.vars.lazy) && _lastRenderedFrame !== _ticker.frame) {\n _lazyTweens.push(tween);\n\n tween._lazy = [totalTime, suppressEvents];\n return 1;\n }\n},\n _parentPlayheadIsBeforeStart = function _parentPlayheadIsBeforeStart(_ref) {\n var parent = _ref.parent;\n return parent && parent._ts && parent._initted && !parent._lock && (parent.rawTime() < 0 || _parentPlayheadIsBeforeStart(parent));\n},\n // check parent's _lock because when a timeline repeats/yoyos and does its artificial wrapping, we shouldn't force the ratio back to 0\n_isFromOrFromStart = function _isFromOrFromStart(_ref2) {\n var data = _ref2.data;\n return data === \"isFromStart\" || data === \"isStart\";\n},\n _renderZeroDurationTween = function _renderZeroDurationTween(tween, totalTime, suppressEvents, force) {\n var prevRatio = tween.ratio,\n ratio = totalTime < 0 || !totalTime && (!tween._start && _parentPlayheadIsBeforeStart(tween) && !(!tween._initted && _isFromOrFromStart(tween)) || (tween._ts < 0 || tween._dp._ts < 0) && !_isFromOrFromStart(tween)) ? 0 : 1,\n // if the tween or its parent is reversed and the totalTime is 0, we should go to a ratio of 0. Edge case: if a from() or fromTo() stagger tween is placed later in a timeline, the \"startAt\" zero-duration tween could initially render at a time when the parent timeline's playhead is technically BEFORE where this tween is, so make sure that any \"from\" and \"fromTo\" startAt tweens are rendered the first time at a ratio of 1.\n repeatDelay = tween._rDelay,\n tTime = 0,\n pt,\n iteration,\n prevIteration;\n\n if (repeatDelay && tween._repeat) {\n // in case there's a zero-duration tween that has a repeat with a repeatDelay\n tTime = _clamp(0, tween._tDur, totalTime);\n iteration = _animationCycle(tTime, repeatDelay);\n tween._yoyo && iteration & 1 && (ratio = 1 - ratio);\n\n if (iteration !== _animationCycle(tween._tTime, repeatDelay)) {\n // if iteration changed\n prevRatio = 1 - ratio;\n tween.vars.repeatRefresh && tween._initted && tween.invalidate();\n }\n }\n\n if (ratio !== prevRatio || _reverting || force || tween._zTime === _tinyNum || !totalTime && tween._zTime) {\n if (!tween._initted && _attemptInitTween(tween, totalTime, force, suppressEvents)) {\n // if we render the very beginning (time == 0) of a fromTo(), we must force the render (normal tweens wouldn't need to render at a time of 0 when the prevTime was also 0). This is also mandatory to make sure overwriting kicks in immediately.\n return;\n }\n\n prevIteration = tween._zTime;\n tween._zTime = totalTime || (suppressEvents ? _tinyNum : 0); // when the playhead arrives at EXACTLY time 0 (right on top) of a zero-duration tween, we need to discern if events are suppressed so that when the playhead moves again (next time), it'll trigger the callback. If events are NOT suppressed, obviously the callback would be triggered in this render. Basically, the callback should fire either when the playhead ARRIVES or LEAVES this exact spot, not both. Imagine doing a timeline.seek(0) and there's a callback that sits at 0. Since events are suppressed on that seek() by default, nothing will fire, but when the playhead moves off of that position, the callback should fire. This behavior is what people intuitively expect.\n\n suppressEvents || (suppressEvents = totalTime && !prevIteration); // if it was rendered previously at exactly 0 (_zTime) and now the playhead is moving away, DON'T fire callbacks otherwise they'll seem like duplicates.\n\n tween.ratio = ratio;\n tween._from && (ratio = 1 - ratio);\n tween._time = 0;\n tween._tTime = tTime;\n pt = tween._pt;\n\n while (pt) {\n pt.r(ratio, pt.d);\n pt = pt._next;\n }\n\n totalTime < 0 && _rewindStartAt(tween, totalTime, suppressEvents, true);\n tween._onUpdate && !suppressEvents && _callback(tween, \"onUpdate\");\n tTime && tween._repeat && !suppressEvents && tween.parent && _callback(tween, \"onRepeat\");\n\n if ((totalTime >= tween._tDur || totalTime < 0) && tween.ratio === ratio) {\n ratio && _removeFromParent(tween, 1);\n\n if (!suppressEvents && !_reverting) {\n _callback(tween, ratio ? \"onComplete\" : \"onReverseComplete\", true);\n\n tween._prom && tween._prom();\n }\n }\n } else if (!tween._zTime) {\n tween._zTime = totalTime;\n }\n},\n _findNextPauseTween = function _findNextPauseTween(animation, prevTime, time) {\n var child;\n\n if (time > prevTime) {\n child = animation._first;\n\n while (child && child._start <= time) {\n if (child.data === \"isPause\" && child._start > prevTime) {\n return child;\n }\n\n child = child._next;\n }\n } else {\n child = animation._last;\n\n while (child && child._start >= time) {\n if (child.data === \"isPause\" && child._start < prevTime) {\n return child;\n }\n\n child = child._prev;\n }\n }\n},\n _setDuration = function _setDuration(animation, duration, skipUncache, leavePlayhead) {\n var repeat = animation._repeat,\n dur = _roundPrecise(duration) || 0,\n totalProgress = animation._tTime / animation._tDur;\n totalProgress && !leavePlayhead && (animation._time *= dur / animation._dur);\n animation._dur = dur;\n animation._tDur = !repeat ? dur : repeat < 0 ? 1e10 : _roundPrecise(dur * (repeat + 1) + animation._rDelay * repeat);\n totalProgress > 0 && !leavePlayhead ? _alignPlayhead(animation, animation._tTime = animation._tDur * totalProgress) : animation.parent && _setEnd(animation);\n skipUncache || _uncache(animation.parent, animation);\n return animation;\n},\n _onUpdateTotalDuration = function _onUpdateTotalDuration(animation) {\n return animation instanceof Timeline ? _uncache(animation) : _setDuration(animation, animation._dur);\n},\n _zeroPosition = {\n _start: 0,\n endTime: _emptyFunc,\n totalDuration: _emptyFunc\n},\n _parsePosition = function _parsePosition(animation, position, percentAnimation) {\n var labels = animation.labels,\n recent = animation._recent || _zeroPosition,\n clippedDuration = animation.duration() >= _bigNum ? recent.endTime(false) : animation._dur,\n //in case there's a child that infinitely repeats, users almost never intend for the insertion point of a new child to be based on a SUPER long value like that so we clip it and assume the most recently-added child's endTime should be used instead.\n i,\n offset,\n isPercent;\n\n if (_isString(position) && (isNaN(position) || position in labels)) {\n //if the string is a number like \"1\", check to see if there's a label with that name, otherwise interpret it as a number (absolute value).\n offset = position.charAt(0);\n isPercent = position.substr(-1) === \"%\";\n i = position.indexOf(\"=\");\n\n if (offset === \"<\" || offset === \">\") {\n i >= 0 && (position = position.replace(/=/, \"\"));\n return (offset === \"<\" ? recent._start : recent.endTime(recent._repeat >= 0)) + (parseFloat(position.substr(1)) || 0) * (isPercent ? (i < 0 ? recent : percentAnimation).totalDuration() / 100 : 1);\n }\n\n if (i < 0) {\n position in labels || (labels[position] = clippedDuration);\n return labels[position];\n }\n\n offset = parseFloat(position.charAt(i - 1) + position.substr(i + 1));\n\n if (isPercent && percentAnimation) {\n offset = offset / 100 * (_isArray(percentAnimation) ? percentAnimation[0] : percentAnimation).totalDuration();\n }\n\n return i > 1 ? _parsePosition(animation, position.substr(0, i - 1), percentAnimation) + offset : clippedDuration + offset;\n }\n\n return position == null ? clippedDuration : +position;\n},\n _createTweenType = function _createTweenType(type, params, timeline) {\n var isLegacy = _isNumber(params[1]),\n varsIndex = (isLegacy ? 2 : 1) + (type < 2 ? 0 : 1),\n vars = params[varsIndex],\n irVars,\n parent;\n\n isLegacy && (vars.duration = params[1]);\n vars.parent = timeline;\n\n if (type) {\n irVars = vars;\n parent = timeline;\n\n while (parent && !(\"immediateRender\" in irVars)) {\n // inheritance hasn't happened yet, but someone may have set a default in an ancestor timeline. We could do vars.immediateRender = _isNotFalse(_inheritDefaults(vars).immediateRender) but that'd exact a slight performance penalty because _inheritDefaults() also runs in the Tween constructor. We're paying a small kb price here to gain speed.\n irVars = parent.vars.defaults || {};\n parent = _isNotFalse(parent.vars.inherit) && parent.parent;\n }\n\n vars.immediateRender = _isNotFalse(irVars.immediateRender);\n type < 2 ? vars.runBackwards = 1 : vars.startAt = params[varsIndex - 1]; // \"from\" vars\n }\n\n return new Tween(params[0], vars, params[varsIndex + 1]);\n},\n _conditionalReturn = function _conditionalReturn(value, func) {\n return value || value === 0 ? func(value) : func;\n},\n _clamp = function _clamp(min, max, value) {\n return value < min ? min : value > max ? max : value;\n},\n getUnit = function getUnit(value, v) {\n return !_isString(value) || !(v = _unitExp.exec(value)) ? \"\" : v[1];\n},\n // note: protect against padded numbers as strings, like \"100.100\". That shouldn't return \"00\" as the unit. If it's numeric, return no unit.\nclamp = function clamp(min, max, value) {\n return _conditionalReturn(value, function (v) {\n return _clamp(min, max, v);\n });\n},\n _slice = [].slice,\n _isArrayLike = function _isArrayLike(value, nonEmpty) {\n return value && _isObject(value) && \"length\" in value && (!nonEmpty && !value.length || value.length - 1 in value && _isObject(value[0])) && !value.nodeType && value !== _win;\n},\n _flatten = function _flatten(ar, leaveStrings, accumulator) {\n if (accumulator === void 0) {\n accumulator = [];\n }\n\n return ar.forEach(function (value) {\n var _accumulator;\n\n return _isString(value) && !leaveStrings || _isArrayLike(value, 1) ? (_accumulator = accumulator).push.apply(_accumulator, toArray(value)) : accumulator.push(value);\n }) || accumulator;\n},\n //takes any value and returns an array. If it's a string (and leaveStrings isn't true), it'll use document.querySelectorAll() and convert that to an array. It'll also accept iterables like jQuery objects.\ntoArray = function toArray(value, scope, leaveStrings) {\n return _context && !scope && _context.selector ? _context.selector(value) : _isString(value) && !leaveStrings && (_coreInitted || !_wake()) ? _slice.call((scope || _doc).querySelectorAll(value), 0) : _isArray(value) ? _flatten(value, leaveStrings) : _isArrayLike(value) ? _slice.call(value, 0) : value ? [value] : [];\n},\n selector = function selector(value) {\n value = toArray(value)[0] || _warn(\"Invalid scope\") || {};\n return function (v) {\n var el = value.current || value.nativeElement || value;\n return toArray(v, el.querySelectorAll ? el : el === value ? _warn(\"Invalid scope\") || _doc.createElement(\"div\") : value);\n };\n},\n shuffle = function shuffle(a) {\n return a.sort(function () {\n return .5 - Math.random();\n });\n},\n // alternative that's a bit faster and more reliably diverse but bigger: for (let j, v, i = a.length; i; j = Math.floor(Math.random() * i), v = a[--i], a[i] = a[j], a[j] = v); return a;\n//for distributing values across an array. Can accept a number, a function or (most commonly) a function which can contain the following properties: {base, amount, from, ease, grid, axis, length, each}. Returns a function that expects the following parameters: index, target, array. Recognizes the following\ndistribute = function distribute(v) {\n if (_isFunction(v)) {\n return v;\n }\n\n var vars = _isObject(v) ? v : {\n each: v\n },\n //n:1 is just to indicate v was a number; we leverage that later to set v according to the length we get. If a number is passed in, we treat it like the old stagger value where 0.1, for example, would mean that things would be distributed with 0.1 between each element in the array rather than a total \"amount\" that's chunked out among them all.\n ease = _parseEase(vars.ease),\n from = vars.from || 0,\n base = parseFloat(vars.base) || 0,\n cache = {},\n isDecimal = from > 0 && from < 1,\n ratios = isNaN(from) || isDecimal,\n axis = vars.axis,\n ratioX = from,\n ratioY = from;\n\n if (_isString(from)) {\n ratioX = ratioY = {\n center: .5,\n edges: .5,\n end: 1\n }[from] || 0;\n } else if (!isDecimal && ratios) {\n ratioX = from[0];\n ratioY = from[1];\n }\n\n return function (i, target, a) {\n var l = (a || vars).length,\n distances = cache[l],\n originX,\n originY,\n x,\n y,\n d,\n j,\n max,\n min,\n wrapAt;\n\n if (!distances) {\n wrapAt = vars.grid === \"auto\" ? 0 : (vars.grid || [1, _bigNum])[1];\n\n if (!wrapAt) {\n max = -_bigNum;\n\n while (max < (max = a[wrapAt++].getBoundingClientRect().left) && wrapAt < l) {}\n\n wrapAt--;\n }\n\n distances = cache[l] = [];\n originX = ratios ? Math.min(wrapAt, l) * ratioX - .5 : from % wrapAt;\n originY = wrapAt === _bigNum ? 0 : ratios ? l * ratioY / wrapAt - .5 : from / wrapAt | 0;\n max = 0;\n min = _bigNum;\n\n for (j = 0; j < l; j++) {\n x = j % wrapAt - originX;\n y = originY - (j / wrapAt | 0);\n distances[j] = d = !axis ? _sqrt(x * x + y * y) : Math.abs(axis === \"y\" ? y : x);\n d > max && (max = d);\n d < min && (min = d);\n }\n\n from === \"random\" && shuffle(distances);\n distances.max = max - min;\n distances.min = min;\n distances.v = l = (parseFloat(vars.amount) || parseFloat(vars.each) * (wrapAt > l ? l - 1 : !axis ? Math.max(wrapAt, l / wrapAt) : axis === \"y\" ? l / wrapAt : wrapAt) || 0) * (from === \"edges\" ? -1 : 1);\n distances.b = l < 0 ? base - l : base;\n distances.u = getUnit(vars.amount || vars.each) || 0; //unit\n\n ease = ease && l < 0 ? _invertEase(ease) : ease;\n }\n\n l = (distances[i] - distances.min) / distances.max || 0;\n return _roundPrecise(distances.b + (ease ? ease(l) : l) * distances.v) + distances.u; //round in order to work around floating point errors\n };\n},\n _roundModifier = function _roundModifier(v) {\n //pass in 0.1 get a function that'll round to the nearest tenth, or 5 to round to the closest 5, or 0.001 to the closest 1000th, etc.\n var p = Math.pow(10, ((v + \"\").split(\".\")[1] || \"\").length); //to avoid floating point math errors (like 24 * 0.1 == 2.4000000000000004), we chop off at a specific number of decimal places (much faster than toFixed())\n\n return function (raw) {\n var n = _roundPrecise(Math.round(parseFloat(raw) / v) * v * p);\n\n return (n - n % 1) / p + (_isNumber(raw) ? 0 : getUnit(raw)); // n - n % 1 replaces Math.floor() in order to handle negative values properly. For example, Math.floor(-150.00000000000003) is 151!\n };\n},\n snap = function snap(snapTo, value) {\n var isArray = _isArray(snapTo),\n radius,\n is2D;\n\n if (!isArray && _isObject(snapTo)) {\n radius = isArray = snapTo.radius || _bigNum;\n\n if (snapTo.values) {\n snapTo = toArray(snapTo.values);\n\n if (is2D = !_isNumber(snapTo[0])) {\n radius *= radius; //performance optimization so we don't have to Math.sqrt() in the loop.\n }\n } else {\n snapTo = _roundModifier(snapTo.increment);\n }\n }\n\n return _conditionalReturn(value, !isArray ? _roundModifier(snapTo) : _isFunction(snapTo) ? function (raw) {\n is2D = snapTo(raw);\n return Math.abs(is2D - raw) <= radius ? is2D : raw;\n } : function (raw) {\n var x = parseFloat(is2D ? raw.x : raw),\n y = parseFloat(is2D ? raw.y : 0),\n min = _bigNum,\n closest = 0,\n i = snapTo.length,\n dx,\n dy;\n\n while (i--) {\n if (is2D) {\n dx = snapTo[i].x - x;\n dy = snapTo[i].y - y;\n dx = dx * dx + dy * dy;\n } else {\n dx = Math.abs(snapTo[i] - x);\n }\n\n if (dx < min) {\n min = dx;\n closest = i;\n }\n }\n\n closest = !radius || min <= radius ? snapTo[closest] : raw;\n return is2D || closest === raw || _isNumber(raw) ? closest : closest + getUnit(raw);\n });\n},\n random = function random(min, max, roundingIncrement, returnFunction) {\n return _conditionalReturn(_isArray(min) ? !max : roundingIncrement === true ? !!(roundingIncrement = 0) : !returnFunction, function () {\n return _isArray(min) ? min[~~(Math.random() * min.length)] : (roundingIncrement = roundingIncrement || 1e-5) && (returnFunction = roundingIncrement < 1 ? Math.pow(10, (roundingIncrement + \"\").length - 2) : 1) && Math.floor(Math.round((min - roundingIncrement / 2 + Math.random() * (max - min + roundingIncrement * .99)) / roundingIncrement) * roundingIncrement * returnFunction) / returnFunction;\n });\n},\n pipe = function pipe() {\n for (var _len = arguments.length, functions = new Array(_len), _key = 0; _key < _len; _key++) {\n functions[_key] = arguments[_key];\n }\n\n return function (value) {\n return functions.reduce(function (v, f) {\n return f(v);\n }, value);\n };\n},\n unitize = function unitize(func, unit) {\n return function (value) {\n return func(parseFloat(value)) + (unit || getUnit(value));\n };\n},\n normalize = function normalize(min, max, value) {\n return mapRange(min, max, 0, 1, value);\n},\n _wrapArray = function _wrapArray(a, wrapper, value) {\n return _conditionalReturn(value, function (index) {\n return a[~~wrapper(index)];\n });\n},\n wrap = function wrap(min, max, value) {\n // NOTE: wrap() CANNOT be an arrow function! A very odd compiling bug causes problems (unrelated to GSAP).\n var range = max - min;\n return _isArray(min) ? _wrapArray(min, wrap(0, min.length), max) : _conditionalReturn(value, function (value) {\n return (range + (value - min) % range) % range + min;\n });\n},\n wrapYoyo = function wrapYoyo(min, max, value) {\n var range = max - min,\n total = range * 2;\n return _isArray(min) ? _wrapArray(min, wrapYoyo(0, min.length - 1), max) : _conditionalReturn(value, function (value) {\n value = (total + (value - min) % total) % total || 0;\n return min + (value > range ? total - value : value);\n });\n},\n _replaceRandom = function _replaceRandom(value) {\n //replaces all occurrences of random(...) in a string with the calculated random value. can be a range like random(-100, 100, 5) or an array like random([0, 100, 500])\n var prev = 0,\n s = \"\",\n i,\n nums,\n end,\n isArray;\n\n while (~(i = value.indexOf(\"random(\", prev))) {\n end = value.indexOf(\")\", i);\n isArray = value.charAt(i + 7) === \"[\";\n nums = value.substr(i + 7, end - i - 7).match(isArray ? _delimitedValueExp : _strictNumExp);\n s += value.substr(prev, i - prev) + random(isArray ? nums : +nums[0], isArray ? 0 : +nums[1], +nums[2] || 1e-5);\n prev = end + 1;\n }\n\n return s + value.substr(prev, value.length - prev);\n},\n mapRange = function mapRange(inMin, inMax, outMin, outMax, value) {\n var inRange = inMax - inMin,\n outRange = outMax - outMin;\n return _conditionalReturn(value, function (value) {\n return outMin + ((value - inMin) / inRange * outRange || 0);\n });\n},\n interpolate = function interpolate(start, end, progress, mutate) {\n var func = isNaN(start + end) ? 0 : function (p) {\n return (1 - p) * start + p * end;\n };\n\n if (!func) {\n var isString = _isString(start),\n master = {},\n p,\n i,\n interpolators,\n l,\n il;\n\n progress === true && (mutate = 1) && (progress = null);\n\n if (isString) {\n start = {\n p: start\n };\n end = {\n p: end\n };\n } else if (_isArray(start) && !_isArray(end)) {\n interpolators = [];\n l = start.length;\n il = l - 2;\n\n for (i = 1; i < l; i++) {\n interpolators.push(interpolate(start[i - 1], start[i])); //build the interpolators up front as a performance optimization so that when the function is called many times, it can just reuse them.\n }\n\n l--;\n\n func = function func(p) {\n p *= l;\n var i = Math.min(il, ~~p);\n return interpolators[i](p - i);\n };\n\n progress = end;\n } else if (!mutate) {\n start = _merge(_isArray(start) ? [] : {}, start);\n }\n\n if (!interpolators) {\n for (p in end) {\n _addPropTween.call(master, start, p, \"get\", end[p]);\n }\n\n func = function func(p) {\n return _renderPropTweens(p, master) || (isString ? start.p : start);\n };\n }\n }\n\n return _conditionalReturn(progress, func);\n},\n _getLabelInDirection = function _getLabelInDirection(timeline, fromTime, backward) {\n //used for nextLabel() and previousLabel()\n var labels = timeline.labels,\n min = _bigNum,\n p,\n distance,\n label;\n\n for (p in labels) {\n distance = labels[p] - fromTime;\n\n if (distance < 0 === !!backward && distance && min > (distance = Math.abs(distance))) {\n label = p;\n min = distance;\n }\n }\n\n return label;\n},\n _callback = function _callback(animation, type, executeLazyFirst) {\n var v = animation.vars,\n callback = v[type],\n prevContext = _context,\n context = animation._ctx,\n params,\n scope,\n result;\n\n if (!callback) {\n return;\n }\n\n params = v[type + \"Params\"];\n scope = v.callbackScope || animation;\n executeLazyFirst && _lazyTweens.length && _lazyRender(); //in case rendering caused any tweens to lazy-init, we should render them because typically when a timeline finishes, users expect things to have rendered fully. Imagine an onUpdate on a timeline that reports/checks tweened values.\n\n context && (_context = context);\n result = params ? callback.apply(scope, params) : callback.call(scope);\n _context = prevContext;\n return result;\n},\n _interrupt = function _interrupt(animation) {\n _removeFromParent(animation);\n\n animation.scrollTrigger && animation.scrollTrigger.kill(false);\n animation.progress() < 1 && _callback(animation, \"onInterrupt\");\n return animation;\n},\n _quickTween,\n _createPlugin = function _createPlugin(config) {\n config = !config.name && config[\"default\"] || config; //UMD packaging wraps things oddly, so for example MotionPathHelper becomes {MotionPathHelper:MotionPathHelper, default:MotionPathHelper}.\n\n var name = config.name,\n isFunc = _isFunction(config),\n Plugin = name && !isFunc && config.init ? function () {\n this._props = [];\n } : config,\n //in case someone passes in an object that's not a plugin, like CustomEase\n instanceDefaults = {\n init: _emptyFunc,\n render: _renderPropTweens,\n add: _addPropTween,\n kill: _killPropTweensOf,\n modifier: _addPluginModifier,\n rawVars: 0\n },\n statics = {\n targetTest: 0,\n get: 0,\n getSetter: _getSetter,\n aliases: {},\n register: 0\n };\n\n _wake();\n\n if (config !== Plugin) {\n if (_plugins[name]) {\n return;\n }\n\n _setDefaults(Plugin, _setDefaults(_copyExcluding(config, instanceDefaults), statics)); //static methods\n\n\n _merge(Plugin.prototype, _merge(instanceDefaults, _copyExcluding(config, statics))); //instance methods\n\n\n _plugins[Plugin.prop = name] = Plugin;\n\n if (config.targetTest) {\n _harnessPlugins.push(Plugin);\n\n _reservedProps[name] = 1;\n }\n\n name = (name === \"css\" ? \"CSS\" : name.charAt(0).toUpperCase() + name.substr(1)) + \"Plugin\"; //for the global name. \"motionPath\" should become MotionPathPlugin\n }\n\n _addGlobal(name, Plugin);\n\n config.register && config.register(gsap, Plugin, PropTween);\n},\n\n/*\n * --------------------------------------------------------------------------------------\n * COLORS\n * --------------------------------------------------------------------------------------\n */\n_255 = 255,\n _colorLookup = {\n aqua: [0, _255, _255],\n lime: [0, _255, 0],\n silver: [192, 192, 192],\n black: [0, 0, 0],\n maroon: [128, 0, 0],\n teal: [0, 128, 128],\n blue: [0, 0, _255],\n navy: [0, 0, 128],\n white: [_255, _255, _255],\n olive: [128, 128, 0],\n yellow: [_255, _255, 0],\n orange: [_255, 165, 0],\n gray: [128, 128, 128],\n purple: [128, 0, 128],\n green: [0, 128, 0],\n red: [_255, 0, 0],\n pink: [_255, 192, 203],\n cyan: [0, _255, _255],\n transparent: [_255, _255, _255, 0]\n},\n // possible future idea to replace the hard-coded color name values - put this in the ticker.wake() where we set the _doc:\n// let ctx = _doc.createElement(\"canvas\").getContext(\"2d\");\n// _forEachName(\"aqua,lime,silver,black,maroon,teal,blue,navy,white,olive,yellow,orange,gray,purple,green,red,pink,cyan\", color => {ctx.fillStyle = color; _colorLookup[color] = splitColor(ctx.fillStyle)});\n_hue = function _hue(h, m1, m2) {\n h += h < 0 ? 1 : h > 1 ? -1 : 0;\n return (h * 6 < 1 ? m1 + (m2 - m1) * h * 6 : h < .5 ? m2 : h * 3 < 2 ? m1 + (m2 - m1) * (2 / 3 - h) * 6 : m1) * _255 + .5 | 0;\n},\n splitColor = function splitColor(v, toHSL, forceAlpha) {\n var a = !v ? _colorLookup.black : _isNumber(v) ? [v >> 16, v >> 8 & _255, v & _255] : 0,\n r,\n g,\n b,\n h,\n s,\n l,\n max,\n min,\n d,\n wasHSL;\n\n if (!a) {\n if (v.substr(-1) === \",\") {\n //sometimes a trailing comma is included and we should chop it off (typically from a comma-delimited list of values like a textShadow:\"2px 2px 2px blue, 5px 5px 5px rgb(255,0,0)\" - in this example \"blue,\" has a trailing comma. We could strip it out inside parseComplex() but we'd need to do it to the beginning and ending values plus it wouldn't provide protection from other potential scenarios like if the user passes in a similar value.\n v = v.substr(0, v.length - 1);\n }\n\n if (_colorLookup[v]) {\n a = _colorLookup[v];\n } else if (v.charAt(0) === \"#\") {\n if (v.length < 6) {\n //for shorthand like #9F0 or #9F0F (could have alpha)\n r = v.charAt(1);\n g = v.charAt(2);\n b = v.charAt(3);\n v = \"#\" + r + r + g + g + b + b + (v.length === 5 ? v.charAt(4) + v.charAt(4) : \"\");\n }\n\n if (v.length === 9) {\n // hex with alpha, like #fd5e53ff\n a = parseInt(v.substr(1, 6), 16);\n return [a >> 16, a >> 8 & _255, a & _255, parseInt(v.substr(7), 16) / 255];\n }\n\n v = parseInt(v.substr(1), 16);\n a = [v >> 16, v >> 8 & _255, v & _255];\n } else if (v.substr(0, 3) === \"hsl\") {\n a = wasHSL = v.match(_strictNumExp);\n\n if (!toHSL) {\n h = +a[0] % 360 / 360;\n s = +a[1] / 100;\n l = +a[2] / 100;\n g = l <= .5 ? l * (s + 1) : l + s - l * s;\n r = l * 2 - g;\n a.length > 3 && (a[3] *= 1); //cast as number\n\n a[0] = _hue(h + 1 / 3, r, g);\n a[1] = _hue(h, r, g);\n a[2] = _hue(h - 1 / 3, r, g);\n } else if (~v.indexOf(\"=\")) {\n //if relative values are found, just return the raw strings with the relative prefixes in place.\n a = v.match(_numExp);\n forceAlpha && a.length < 4 && (a[3] = 1);\n return a;\n }\n } else {\n a = v.match(_strictNumExp) || _colorLookup.transparent;\n }\n\n a = a.map(Number);\n }\n\n if (toHSL && !wasHSL) {\n r = a[0] / _255;\n g = a[1] / _255;\n b = a[2] / _255;\n max = Math.max(r, g, b);\n min = Math.min(r, g, b);\n l = (max + min) / 2;\n\n if (max === min) {\n h = s = 0;\n } else {\n d = max - min;\n s = l > 0.5 ? d / (2 - max - min) : d / (max + min);\n h = max === r ? (g - b) / d + (g < b ? 6 : 0) : max === g ? (b - r) / d + 2 : (r - g) / d + 4;\n h *= 60;\n }\n\n a[0] = ~~(h + .5);\n a[1] = ~~(s * 100 + .5);\n a[2] = ~~(l * 100 + .5);\n }\n\n forceAlpha && a.length < 4 && (a[3] = 1);\n return a;\n},\n _colorOrderData = function _colorOrderData(v) {\n // strips out the colors from the string, finds all the numeric slots (with units) and returns an array of those. The Array also has a \"c\" property which is an Array of the index values where the colors belong. This is to help work around issues where there's a mis-matched order of color/numeric data like drop-shadow(#f00 0px 1px 2px) and drop-shadow(0x 1px 2px #f00). This is basically a helper function used in _formatColors()\n var values = [],\n c = [],\n i = -1;\n v.split(_colorExp).forEach(function (v) {\n var a = v.match(_numWithUnitExp) || [];\n values.push.apply(values, a);\n c.push(i += a.length + 1);\n });\n values.c = c;\n return values;\n},\n _formatColors = function _formatColors(s, toHSL, orderMatchData) {\n var result = \"\",\n colors = (s + result).match(_colorExp),\n type = toHSL ? \"hsla(\" : \"rgba(\",\n i = 0,\n c,\n shell,\n d,\n l;\n\n if (!colors) {\n return s;\n }\n\n colors = colors.map(function (color) {\n return (color = splitColor(color, toHSL, 1)) && type + (toHSL ? color[0] + \",\" + color[1] + \"%,\" + color[2] + \"%,\" + color[3] : color.join(\",\")) + \")\";\n });\n\n if (orderMatchData) {\n d = _colorOrderData(s);\n c = orderMatchData.c;\n\n if (c.join(result) !== d.c.join(result)) {\n shell = s.replace(_colorExp, \"1\").split(_numWithUnitExp);\n l = shell.length - 1;\n\n for (; i < l; i++) {\n result += shell[i] + (~c.indexOf(i) ? colors.shift() || type + \"0,0,0,0)\" : (d.length ? d : colors.length ? colors : orderMatchData).shift());\n }\n }\n }\n\n if (!shell) {\n shell = s.split(_colorExp);\n l = shell.length - 1;\n\n for (; i < l; i++) {\n result += shell[i] + colors[i];\n }\n }\n\n return result + shell[l];\n},\n _colorExp = function () {\n var s = \"(?:\\\\b(?:(?:rgb|rgba|hsl|hsla)\\\\(.+?\\\\))|\\\\B#(?:[0-9a-f]{3,4}){1,2}\\\\b\",\n //we'll dynamically build this Regular Expression to conserve file size. After building it, it will be able to find rgb(), rgba(), # (hexadecimal), and named color values like red, blue, purple, etc.,\n p;\n\n for (p in _colorLookup) {\n s += \"|\" + p + \"\\\\b\";\n }\n\n return new RegExp(s + \")\", \"gi\");\n}(),\n _hslExp = /hsl[a]?\\(/,\n _colorStringFilter = function _colorStringFilter(a) {\n var combined = a.join(\" \"),\n toHSL;\n _colorExp.lastIndex = 0;\n\n if (_colorExp.test(combined)) {\n toHSL = _hslExp.test(combined);\n a[1] = _formatColors(a[1], toHSL);\n a[0] = _formatColors(a[0], toHSL, _colorOrderData(a[1])); // make sure the order of numbers/colors match with the END value.\n\n return true;\n }\n},\n\n/*\n * --------------------------------------------------------------------------------------\n * TICKER\n * --------------------------------------------------------------------------------------\n */\n_tickerActive,\n _ticker = function () {\n var _getTime = Date.now,\n _lagThreshold = 500,\n _adjustedLag = 33,\n _startTime = _getTime(),\n _lastUpdate = _startTime,\n _gap = 1000 / 240,\n _nextTime = _gap,\n _listeners = [],\n _id,\n _req,\n _raf,\n _self,\n _delta,\n _i,\n _tick = function _tick(v) {\n var elapsed = _getTime() - _lastUpdate,\n manual = v === true,\n overlap,\n dispatch,\n time,\n frame;\n\n elapsed > _lagThreshold && (_startTime += elapsed - _adjustedLag);\n _lastUpdate += elapsed;\n time = _lastUpdate - _startTime;\n overlap = time - _nextTime;\n\n if (overlap > 0 || manual) {\n frame = ++_self.frame;\n _delta = time - _self.time * 1000;\n _self.time = time = time / 1000;\n _nextTime += overlap + (overlap >= _gap ? 4 : _gap - overlap);\n dispatch = 1;\n }\n\n manual || (_id = _req(_tick)); //make sure the request is made before we dispatch the \"tick\" event so that timing is maintained. Otherwise, if processing the \"tick\" requires a bunch of time (like 15ms) and we're using a setTimeout() that's based on 16.7ms, it'd technically take 31.7ms between frames otherwise.\n\n if (dispatch) {\n for (_i = 0; _i < _listeners.length; _i++) {\n // use _i and check _listeners.length instead of a variable because a listener could get removed during the loop, and if that happens to an element less than the current index, it'd throw things off in the loop.\n _listeners[_i](time, _delta, frame, v);\n }\n }\n };\n\n _self = {\n time: 0,\n frame: 0,\n tick: function tick() {\n _tick(true);\n },\n deltaRatio: function deltaRatio(fps) {\n return _delta / (1000 / (fps || 60));\n },\n wake: function wake() {\n if (_coreReady) {\n if (!_coreInitted && _windowExists()) {\n _win = _coreInitted = window;\n _doc = _win.document || {};\n _globals.gsap = gsap;\n (_win.gsapVersions || (_win.gsapVersions = [])).push(gsap.version);\n\n _install(_installScope || _win.GreenSockGlobals || !_win.gsap && _win || {});\n\n _raf = _win.requestAnimationFrame;\n }\n\n _id && _self.sleep();\n\n _req = _raf || function (f) {\n return setTimeout(f, _nextTime - _self.time * 1000 + 1 | 0);\n };\n\n _tickerActive = 1;\n\n _tick(2);\n }\n },\n sleep: function sleep() {\n (_raf ? _win.cancelAnimationFrame : clearTimeout)(_id);\n _tickerActive = 0;\n _req = _emptyFunc;\n },\n lagSmoothing: function lagSmoothing(threshold, adjustedLag) {\n _lagThreshold = threshold || 1 / _tinyNum; //zero should be interpreted as basically unlimited\n\n _adjustedLag = Math.min(adjustedLag, _lagThreshold, 0);\n },\n fps: function fps(_fps) {\n _gap = 1000 / (_fps || 240);\n _nextTime = _self.time * 1000 + _gap;\n },\n add: function add(callback, once, prioritize) {\n var func = once ? function (t, d, f, v) {\n callback(t, d, f, v);\n\n _self.remove(func);\n } : callback;\n\n _self.remove(callback);\n\n _listeners[prioritize ? \"unshift\" : \"push\"](func);\n\n _wake();\n\n return func;\n },\n remove: function remove(callback, i) {\n ~(i = _listeners.indexOf(callback)) && _listeners.splice(i, 1) && _i >= i && _i--;\n },\n _listeners: _listeners\n };\n return _self;\n}(),\n _wake = function _wake() {\n return !_tickerActive && _ticker.wake();\n},\n //also ensures the core classes are initialized.\n\n/*\n* -------------------------------------------------\n* EASING\n* -------------------------------------------------\n*/\n_easeMap = {},\n _customEaseExp = /^[\\d.\\-M][\\d.\\-,\\s]/,\n _quotesExp = /[\"']/g,\n _parseObjectInString = function _parseObjectInString(value) {\n //takes a string like \"{wiggles:10, type:anticipate})\" and turns it into a real object. Notice it ends in \")\" and includes the {} wrappers. This is because we only use this function for parsing ease configs and prioritized optimization rather than reusability.\n var obj = {},\n split = value.substr(1, value.length - 3).split(\":\"),\n key = split[0],\n i = 1,\n l = split.length,\n index,\n val,\n parsedVal;\n\n for (; i < l; i++) {\n val = split[i];\n index = i !== l - 1 ? val.lastIndexOf(\",\") : val.length;\n parsedVal = val.substr(0, index);\n obj[key] = isNaN(parsedVal) ? parsedVal.replace(_quotesExp, \"\").trim() : +parsedVal;\n key = val.substr(index + 1).trim();\n }\n\n return obj;\n},\n _valueInParentheses = function _valueInParentheses(value) {\n var open = value.indexOf(\"(\") + 1,\n close = value.indexOf(\")\"),\n nested = value.indexOf(\"(\", open);\n return value.substring(open, ~nested && nested < close ? value.indexOf(\")\", close + 1) : close);\n},\n _configEaseFromString = function _configEaseFromString(name) {\n //name can be a string like \"elastic.out(1,0.5)\", and pass in _easeMap as obj and it'll parse it out and call the actual function like _easeMap.Elastic.easeOut.config(1,0.5). It will also parse custom ease strings as long as CustomEase is loaded and registered (internally as _easeMap._CE).\n var split = (name + \"\").split(\"(\"),\n ease = _easeMap[split[0]];\n return ease && split.length > 1 && ease.config ? ease.config.apply(null, ~name.indexOf(\"{\") ? [_parseObjectInString(split[1])] : _valueInParentheses(name).split(\",\").map(_numericIfPossible)) : _easeMap._CE && _customEaseExp.test(name) ? _easeMap._CE(\"\", name) : ease;\n},\n _invertEase = function _invertEase(ease) {\n return function (p) {\n return 1 - ease(1 - p);\n };\n},\n // allow yoyoEase to be set in children and have those affected when the parent/ancestor timeline yoyos.\n_propagateYoyoEase = function _propagateYoyoEase(timeline, isYoyo) {\n var child = timeline._first,\n ease;\n\n while (child) {\n if (child instanceof Timeline) {\n _propagateYoyoEase(child, isYoyo);\n } else if (child.vars.yoyoEase && (!child._yoyo || !child._repeat) && child._yoyo !== isYoyo) {\n if (child.timeline) {\n _propagateYoyoEase(child.timeline, isYoyo);\n } else {\n ease = child._ease;\n child._ease = child._yEase;\n child._yEase = ease;\n child._yoyo = isYoyo;\n }\n }\n\n child = child._next;\n }\n},\n _parseEase = function _parseEase(ease, defaultEase) {\n return !ease ? defaultEase : (_isFunction(ease) ? ease : _easeMap[ease] || _configEaseFromString(ease)) || defaultEase;\n},\n _insertEase = function _insertEase(names, easeIn, easeOut, easeInOut) {\n if (easeOut === void 0) {\n easeOut = function easeOut(p) {\n return 1 - easeIn(1 - p);\n };\n }\n\n if (easeInOut === void 0) {\n easeInOut = function easeInOut(p) {\n return p < .5 ? easeIn(p * 2) / 2 : 1 - easeIn((1 - p) * 2) / 2;\n };\n }\n\n var ease = {\n easeIn: easeIn,\n easeOut: easeOut,\n easeInOut: easeInOut\n },\n lowercaseName;\n\n _forEachName(names, function (name) {\n _easeMap[name] = _globals[name] = ease;\n _easeMap[lowercaseName = name.toLowerCase()] = easeOut;\n\n for (var p in ease) {\n _easeMap[lowercaseName + (p === \"easeIn\" ? \".in\" : p === \"easeOut\" ? \".out\" : \".inOut\")] = _easeMap[name + \".\" + p] = ease[p];\n }\n });\n\n return ease;\n},\n _easeInOutFromOut = function _easeInOutFromOut(easeOut) {\n return function (p) {\n return p < .5 ? (1 - easeOut(1 - p * 2)) / 2 : .5 + easeOut((p - .5) * 2) / 2;\n };\n},\n _configElastic = function _configElastic(type, amplitude, period) {\n var p1 = amplitude >= 1 ? amplitude : 1,\n //note: if amplitude is < 1, we simply adjust the period for a more natural feel. Otherwise the math doesn't work right and the curve starts at 1.\n p2 = (period || (type ? .3 : .45)) / (amplitude < 1 ? amplitude : 1),\n p3 = p2 / _2PI * (Math.asin(1 / p1) || 0),\n easeOut = function easeOut(p) {\n return p === 1 ? 1 : p1 * Math.pow(2, -10 * p) * _sin((p - p3) * p2) + 1;\n },\n ease = type === \"out\" ? easeOut : type === \"in\" ? function (p) {\n return 1 - easeOut(1 - p);\n } : _easeInOutFromOut(easeOut);\n\n p2 = _2PI / p2; //precalculate to optimize\n\n ease.config = function (amplitude, period) {\n return _configElastic(type, amplitude, period);\n };\n\n return ease;\n},\n _configBack = function _configBack(type, overshoot) {\n if (overshoot === void 0) {\n overshoot = 1.70158;\n }\n\n var easeOut = function easeOut(p) {\n return p ? --p * p * ((overshoot + 1) * p + overshoot) + 1 : 0;\n },\n ease = type === \"out\" ? easeOut : type === \"in\" ? function (p) {\n return 1 - easeOut(1 - p);\n } : _easeInOutFromOut(easeOut);\n\n ease.config = function (overshoot) {\n return _configBack(type, overshoot);\n };\n\n return ease;\n}; // a cheaper (kb and cpu) but more mild way to get a parameterized weighted ease by feeding in a value between -1 (easeIn) and 1 (easeOut) where 0 is linear.\n// _weightedEase = ratio => {\n// \tlet y = 0.5 + ratio / 2;\n// \treturn p => (2 * (1 - p) * p * y + p * p);\n// },\n// a stronger (but more expensive kb/cpu) parameterized weighted ease that lets you feed in a value between -1 (easeIn) and 1 (easeOut) where 0 is linear.\n// _weightedEaseStrong = ratio => {\n// \tratio = .5 + ratio / 2;\n// \tlet o = 1 / 3 * (ratio < .5 ? ratio : 1 - ratio),\n// \t\tb = ratio - o,\n// \t\tc = ratio + o;\n// \treturn p => p === 1 ? p : 3 * b * (1 - p) * (1 - p) * p + 3 * c * (1 - p) * p * p + p * p * p;\n// };\n\n\n_forEachName(\"Linear,Quad,Cubic,Quart,Quint,Strong\", function (name, i) {\n var power = i < 5 ? i + 1 : i;\n\n _insertEase(name + \",Power\" + (power - 1), i ? function (p) {\n return Math.pow(p, power);\n } : function (p) {\n return p;\n }, function (p) {\n return 1 - Math.pow(1 - p, power);\n }, function (p) {\n return p < .5 ? Math.pow(p * 2, power) / 2 : 1 - Math.pow((1 - p) * 2, power) / 2;\n });\n});\n\n_easeMap.Linear.easeNone = _easeMap.none = _easeMap.Linear.easeIn;\n\n_insertEase(\"Elastic\", _configElastic(\"in\"), _configElastic(\"out\"), _configElastic());\n\n(function (n, c) {\n var n1 = 1 / c,\n n2 = 2 * n1,\n n3 = 2.5 * n1,\n easeOut = function easeOut(p) {\n return p < n1 ? n * p * p : p < n2 ? n * Math.pow(p - 1.5 / c, 2) + .75 : p < n3 ? n * (p -= 2.25 / c) * p + .9375 : n * Math.pow(p - 2.625 / c, 2) + .984375;\n };\n\n _insertEase(\"Bounce\", function (p) {\n return 1 - easeOut(1 - p);\n }, easeOut);\n})(7.5625, 2.75);\n\n_insertEase(\"Expo\", function (p) {\n return p ? Math.pow(2, 10 * (p - 1)) : 0;\n});\n\n_insertEase(\"Circ\", function (p) {\n return -(_sqrt(1 - p * p) - 1);\n});\n\n_insertEase(\"Sine\", function (p) {\n return p === 1 ? 1 : -_cos(p * _HALF_PI) + 1;\n});\n\n_insertEase(\"Back\", _configBack(\"in\"), _configBack(\"out\"), _configBack());\n\n_easeMap.SteppedEase = _easeMap.steps = _globals.SteppedEase = {\n config: function config(steps, immediateStart) {\n if (steps === void 0) {\n steps = 1;\n }\n\n var p1 = 1 / steps,\n p2 = steps + (immediateStart ? 0 : 1),\n p3 = immediateStart ? 1 : 0,\n max = 1 - _tinyNum;\n return function (p) {\n return ((p2 * _clamp(0, max, p) | 0) + p3) * p1;\n };\n }\n};\n_defaults.ease = _easeMap[\"quad.out\"];\n\n_forEachName(\"onComplete,onUpdate,onStart,onRepeat,onReverseComplete,onInterrupt\", function (name) {\n return _callbackNames += name + \",\" + name + \"Params,\";\n});\n/*\n * --------------------------------------------------------------------------------------\n * CACHE\n * --------------------------------------------------------------------------------------\n */\n\n\nexport var GSCache = function GSCache(target, harness) {\n this.id = _gsID++;\n target._gsap = this;\n this.target = target;\n this.harness = harness;\n this.get = harness ? harness.get : _getProperty;\n this.set = harness ? harness.getSetter : _getSetter;\n};\n/*\n * --------------------------------------------------------------------------------------\n * ANIMATION\n * --------------------------------------------------------------------------------------\n */\n\nexport var Animation = /*#__PURE__*/function () {\n function Animation(vars) {\n this.vars = vars;\n this._delay = +vars.delay || 0;\n\n if (this._repeat = vars.repeat === Infinity ? -2 : vars.repeat || 0) {\n // TODO: repeat: Infinity on a timeline's children must flag that timeline internally and affect its totalDuration, otherwise it'll stop in the negative direction when reaching the start.\n this._rDelay = vars.repeatDelay || 0;\n this._yoyo = !!vars.yoyo || !!vars.yoyoEase;\n }\n\n this._ts = 1;\n\n _setDuration(this, +vars.duration, 1, 1);\n\n this.data = vars.data;\n\n if (_context) {\n this._ctx = _context;\n\n _context.data.push(this);\n }\n\n _tickerActive || _ticker.wake();\n }\n\n var _proto = Animation.prototype;\n\n _proto.delay = function delay(value) {\n if (value || value === 0) {\n this.parent && this.parent.smoothChildTiming && this.startTime(this._start + value - this._delay);\n this._delay = value;\n return this;\n }\n\n return this._delay;\n };\n\n _proto.duration = function duration(value) {\n return arguments.length ? this.totalDuration(this._repeat > 0 ? value + (value + this._rDelay) * this._repeat : value) : this.totalDuration() && this._dur;\n };\n\n _proto.totalDuration = function totalDuration(value) {\n if (!arguments.length) {\n return this._tDur;\n }\n\n this._dirty = 0;\n return _setDuration(this, this._repeat < 0 ? value : (value - this._repeat * this._rDelay) / (this._repeat + 1));\n };\n\n _proto.totalTime = function totalTime(_totalTime, suppressEvents) {\n _wake();\n\n if (!arguments.length) {\n return this._tTime;\n }\n\n var parent = this._dp;\n\n if (parent && parent.smoothChildTiming && this._ts) {\n _alignPlayhead(this, _totalTime);\n\n !parent._dp || parent.parent || _postAddChecks(parent, this); // edge case: if this is a child of a timeline that already completed, for example, we must re-activate the parent.\n //in case any of the ancestor timelines had completed but should now be enabled, we should reset their totalTime() which will also ensure that they're lined up properly and enabled. Skip for animations that are on the root (wasteful). Example: a TimelineLite.exportRoot() is performed when there's a paused tween on the root, the export will not complete until that tween is unpaused, but imagine a child gets restarted later, after all [unpaused] tweens have completed. The start of that child would get pushed out, but one of the ancestors may have completed.\n\n while (parent && parent.parent) {\n if (parent.parent._time !== parent._start + (parent._ts >= 0 ? parent._tTime / parent._ts : (parent.totalDuration() - parent._tTime) / -parent._ts)) {\n parent.totalTime(parent._tTime, true);\n }\n\n parent = parent.parent;\n }\n\n if (!this.parent && this._dp.autoRemoveChildren && (this._ts > 0 && _totalTime < this._tDur || this._ts < 0 && _totalTime > 0 || !this._tDur && !_totalTime)) {\n //if the animation doesn't have a parent, put it back into its last parent (recorded as _dp for exactly cases like this). Limit to parents with autoRemoveChildren (like globalTimeline) so that if the user manually removes an animation from a timeline and then alters its playhead, it doesn't get added back in.\n _addToTimeline(this._dp, this, this._start - this._delay);\n }\n }\n\n if (this._tTime !== _totalTime || !this._dur && !suppressEvents || this._initted && Math.abs(this._zTime) === _tinyNum || !_totalTime && !this._initted && (this.add || this._ptLookup)) {\n // check for _ptLookup on a Tween instance to ensure it has actually finished being instantiated, otherwise if this.reverse() gets called in the Animation constructor, it could trigger a render() here even though the _targets weren't populated, thus when _init() is called there won't be any PropTweens (it'll act like the tween is non-functional)\n this._ts || (this._pTime = _totalTime); // otherwise, if an animation is paused, then the playhead is moved back to zero, then resumed, it'd revert back to the original time at the pause\n //if (!this._lock) { // avoid endless recursion (not sure we need this yet or if it's worth the performance hit)\n // this._lock = 1;\n\n _lazySafeRender(this, _totalTime, suppressEvents); // this._lock = 0;\n //}\n\n }\n\n return this;\n };\n\n _proto.time = function time(value, suppressEvents) {\n return arguments.length ? this.totalTime(Math.min(this.totalDuration(), value + _elapsedCycleDuration(this)) % (this._dur + this._rDelay) || (value ? this._dur : 0), suppressEvents) : this._time; // note: if the modulus results in 0, the playhead could be exactly at the end or the beginning, and we always defer to the END with a non-zero value, otherwise if you set the time() to the very end (duration()), it would render at the START!\n };\n\n _proto.totalProgress = function totalProgress(value, suppressEvents) {\n return arguments.length ? this.totalTime(this.totalDuration() * value, suppressEvents) : this.totalDuration() ? Math.min(1, this._tTime / this._tDur) : this.ratio;\n };\n\n _proto.progress = function progress(value, suppressEvents) {\n return arguments.length ? this.totalTime(this.duration() * (this._yoyo && !(this.iteration() & 1) ? 1 - value : value) + _elapsedCycleDuration(this), suppressEvents) : this.duration() ? Math.min(1, this._time / this._dur) : this.ratio;\n };\n\n _proto.iteration = function iteration(value, suppressEvents) {\n var cycleDuration = this.duration() + this._rDelay;\n\n return arguments.length ? this.totalTime(this._time + (value - 1) * cycleDuration, suppressEvents) : this._repeat ? _animationCycle(this._tTime, cycleDuration) + 1 : 1;\n } // potential future addition:\n // isPlayingBackwards() {\n // \tlet animation = this,\n // \t\torientation = 1; // 1 = forward, -1 = backward\n // \twhile (animation) {\n // \t\torientation *= animation.reversed() || (animation.repeat() && !(animation.iteration() & 1)) ? -1 : 1;\n // \t\tanimation = animation.parent;\n // \t}\n // \treturn orientation < 0;\n // }\n ;\n\n _proto.timeScale = function timeScale(value) {\n if (!arguments.length) {\n return this._rts === -_tinyNum ? 0 : this._rts; // recorded timeScale. Special case: if someone calls reverse() on an animation with timeScale of 0, we assign it -_tinyNum to remember it's reversed.\n }\n\n if (this._rts === value) {\n return this;\n }\n\n var tTime = this.parent && this._ts ? _parentToChildTotalTime(this.parent._time, this) : this._tTime; // make sure to do the parentToChildTotalTime() BEFORE setting the new _ts because the old one must be used in that calculation.\n // future addition? Up side: fast and minimal file size. Down side: only works on this animation; if a timeline is reversed, for example, its childrens' onReverse wouldn't get called.\n //(+value < 0 && this._rts >= 0) && _callback(this, \"onReverse\", true);\n // prioritize rendering where the parent's playhead lines up instead of this._tTime because there could be a tween that's animating another tween's timeScale in the same rendering loop (same parent), thus if the timeScale tween renders first, it would alter _start BEFORE _tTime was set on that tick (in the rendering loop), effectively freezing it until the timeScale tween finishes.\n\n this._rts = +value || 0;\n this._ts = this._ps || value === -_tinyNum ? 0 : this._rts; // _ts is the functional timeScale which would be 0 if the animation is paused.\n\n this.totalTime(_clamp(-this._delay, this._tDur, tTime), true);\n\n _setEnd(this); // if parent.smoothChildTiming was false, the end time didn't get updated in the _alignPlayhead() method, so do it here.\n\n\n return _recacheAncestors(this);\n };\n\n _proto.paused = function paused(value) {\n if (!arguments.length) {\n return this._ps;\n }\n\n if (this._ps !== value) {\n this._ps = value;\n\n if (value) {\n this._pTime = this._tTime || Math.max(-this._delay, this.rawTime()); // if the pause occurs during the delay phase, make sure that's factored in when resuming.\n\n this._ts = this._act = 0; // _ts is the functional timeScale, so a paused tween would effectively have a timeScale of 0. We record the \"real\" timeScale as _rts (recorded time scale)\n } else {\n _wake();\n\n this._ts = this._rts; //only defer to _pTime (pauseTime) if tTime is zero. Remember, someone could pause() an animation, then scrub the playhead and resume(). If the parent doesn't have smoothChildTiming, we render at the rawTime() because the startTime won't get updated.\n\n this.totalTime(this.parent && !this.parent.smoothChildTiming ? this.rawTime() : this._tTime || this._pTime, this.progress() === 1 && Math.abs(this._zTime) !== _tinyNum && (this._tTime -= _tinyNum)); // edge case: animation.progress(1).pause().play() wouldn't render again because the playhead is already at the end, but the call to totalTime() below will add it back to its parent...and not remove it again (since removing only happens upon rendering at a new time). Offsetting the _tTime slightly is done simply to cause the final render in totalTime() that'll pop it off its timeline (if autoRemoveChildren is true, of course). Check to make sure _zTime isn't -_tinyNum to avoid an edge case where the playhead is pushed to the end but INSIDE a tween/callback, the timeline itself is paused thus halting rendering and leaving a few unrendered. When resuming, it wouldn't render those otherwise.\n }\n }\n\n return this;\n };\n\n _proto.startTime = function startTime(value) {\n if (arguments.length) {\n this._start = value;\n var parent = this.parent || this._dp;\n parent && (parent._sort || !this.parent) && _addToTimeline(parent, this, value - this._delay);\n return this;\n }\n\n return this._start;\n };\n\n _proto.endTime = function endTime(includeRepeats) {\n return this._start + (_isNotFalse(includeRepeats) ? this.totalDuration() : this.duration()) / Math.abs(this._ts || 1);\n };\n\n _proto.rawTime = function rawTime(wrapRepeats) {\n var parent = this.parent || this._dp; // _dp = detached parent\n\n return !parent ? this._tTime : wrapRepeats && (!this._ts || this._repeat && this._time && this.totalProgress() < 1) ? this._tTime % (this._dur + this._rDelay) : !this._ts ? this._tTime : _parentToChildTotalTime(parent.rawTime(wrapRepeats), this);\n };\n\n _proto.revert = function revert(config) {\n if (config === void 0) {\n config = _revertConfig;\n }\n\n var prevIsReverting = _reverting;\n _reverting = config;\n this.timeline && this.timeline.revert(config);\n this.totalTime(-0.01, config.suppressEvents);\n this.data !== \"nested\" && _removeFromParent(this);\n _reverting = prevIsReverting;\n return this;\n };\n\n _proto.globalTime = function globalTime(rawTime) {\n var animation = this,\n time = arguments.length ? rawTime : animation.rawTime();\n\n while (animation) {\n time = animation._start + time / (animation._ts || 1);\n animation = animation._dp;\n }\n\n return !this.parent && this.vars.immediateRender ? -1 : time; // the _startAt tweens for .fromTo() and .from() that have immediateRender should always be FIRST in the timeline (important for Recording.revert())\n };\n\n _proto.repeat = function repeat(value) {\n if (arguments.length) {\n this._repeat = value === Infinity ? -2 : value;\n return _onUpdateTotalDuration(this);\n }\n\n return this._repeat === -2 ? Infinity : this._repeat;\n };\n\n _proto.repeatDelay = function repeatDelay(value) {\n if (arguments.length) {\n var time = this._time;\n this._rDelay = value;\n\n _onUpdateTotalDuration(this);\n\n return time ? this.time(time) : this;\n }\n\n return this._rDelay;\n };\n\n _proto.yoyo = function yoyo(value) {\n if (arguments.length) {\n this._yoyo = value;\n return this;\n }\n\n return this._yoyo;\n };\n\n _proto.seek = function seek(position, suppressEvents) {\n return this.totalTime(_parsePosition(this, position), _isNotFalse(suppressEvents));\n };\n\n _proto.restart = function restart(includeDelay, suppressEvents) {\n return this.play().totalTime(includeDelay ? -this._delay : 0, _isNotFalse(suppressEvents));\n };\n\n _proto.play = function play(from, suppressEvents) {\n from != null && this.seek(from, suppressEvents);\n return this.reversed(false).paused(false);\n };\n\n _proto.reverse = function reverse(from, suppressEvents) {\n from != null && this.seek(from || this.totalDuration(), suppressEvents);\n return this.reversed(true).paused(false);\n };\n\n _proto.pause = function pause(atTime, suppressEvents) {\n atTime != null && this.seek(atTime, suppressEvents);\n return this.paused(true);\n };\n\n _proto.resume = function resume() {\n return this.paused(false);\n };\n\n _proto.reversed = function reversed(value) {\n if (arguments.length) {\n !!value !== this.reversed() && this.timeScale(-this._rts || (value ? -_tinyNum : 0)); // in case timeScale is zero, reversing would have no effect so we use _tinyNum.\n\n return this;\n }\n\n return this._rts < 0;\n };\n\n _proto.invalidate = function invalidate() {\n this._initted = this._act = 0;\n this._zTime = -_tinyNum;\n return this;\n };\n\n _proto.isActive = function isActive() {\n var parent = this.parent || this._dp,\n start = this._start,\n rawTime;\n return !!(!parent || this._ts && this._initted && parent.isActive() && (rawTime = parent.rawTime(true)) >= start && rawTime < this.endTime(true) - _tinyNum);\n };\n\n _proto.eventCallback = function eventCallback(type, callback, params) {\n var vars = this.vars;\n\n if (arguments.length > 1) {\n if (!callback) {\n delete vars[type];\n } else {\n vars[type] = callback;\n params && (vars[type + \"Params\"] = params);\n type === \"onUpdate\" && (this._onUpdate = callback);\n }\n\n return this;\n }\n\n return vars[type];\n };\n\n _proto.then = function then(onFulfilled) {\n var self = this;\n return new Promise(function (resolve) {\n var f = _isFunction(onFulfilled) ? onFulfilled : _passThrough,\n _resolve = function _resolve() {\n var _then = self.then;\n self.then = null; // temporarily null the then() method to avoid an infinite loop (see https://github.com/greensock/GSAP/issues/322)\n\n _isFunction(f) && (f = f(self)) && (f.then || f === self) && (self.then = _then);\n resolve(f);\n self.then = _then;\n };\n\n if (self._initted && self.totalProgress() === 1 && self._ts >= 0 || !self._tTime && self._ts < 0) {\n _resolve();\n } else {\n self._prom = _resolve;\n }\n });\n };\n\n _proto.kill = function kill() {\n _interrupt(this);\n };\n\n return Animation;\n}();\n\n_setDefaults(Animation.prototype, {\n _time: 0,\n _start: 0,\n _end: 0,\n _tTime: 0,\n _tDur: 0,\n _dirty: 0,\n _repeat: 0,\n _yoyo: false,\n parent: null,\n _initted: false,\n _rDelay: 0,\n _ts: 1,\n _dp: 0,\n ratio: 0,\n _zTime: -_tinyNum,\n _prom: 0,\n _ps: false,\n _rts: 1\n});\n/*\n * -------------------------------------------------\n * TIMELINE\n * -------------------------------------------------\n */\n\n\nexport var Timeline = /*#__PURE__*/function (_Animation) {\n _inheritsLoose(Timeline, _Animation);\n\n function Timeline(vars, position) {\n var _this;\n\n if (vars === void 0) {\n vars = {};\n }\n\n _this = _Animation.call(this, vars) || this;\n _this.labels = {};\n _this.smoothChildTiming = !!vars.smoothChildTiming;\n _this.autoRemoveChildren = !!vars.autoRemoveChildren;\n _this._sort = _isNotFalse(vars.sortChildren);\n _globalTimeline && _addToTimeline(vars.parent || _globalTimeline, _assertThisInitialized(_this), position);\n vars.reversed && _this.reverse();\n vars.paused && _this.paused(true);\n vars.scrollTrigger && _scrollTrigger(_assertThisInitialized(_this), vars.scrollTrigger);\n return _this;\n }\n\n var _proto2 = Timeline.prototype;\n\n _proto2.to = function to(targets, vars, position) {\n _createTweenType(0, arguments, this);\n\n return this;\n };\n\n _proto2.from = function from(targets, vars, position) {\n _createTweenType(1, arguments, this);\n\n return this;\n };\n\n _proto2.fromTo = function fromTo(targets, fromVars, toVars, position) {\n _createTweenType(2, arguments, this);\n\n return this;\n };\n\n _proto2.set = function set(targets, vars, position) {\n vars.duration = 0;\n vars.parent = this;\n _inheritDefaults(vars).repeatDelay || (vars.repeat = 0);\n vars.immediateRender = !!vars.immediateRender;\n new Tween(targets, vars, _parsePosition(this, position), 1);\n return this;\n };\n\n _proto2.call = function call(callback, params, position) {\n return _addToTimeline(this, Tween.delayedCall(0, callback, params), position);\n } //ONLY for backward compatibility! Maybe delete?\n ;\n\n _proto2.staggerTo = function staggerTo(targets, duration, vars, stagger, position, onCompleteAll, onCompleteAllParams) {\n vars.duration = duration;\n vars.stagger = vars.stagger || stagger;\n vars.onComplete = onCompleteAll;\n vars.onCompleteParams = onCompleteAllParams;\n vars.parent = this;\n new Tween(targets, vars, _parsePosition(this, position));\n return this;\n };\n\n _proto2.staggerFrom = function staggerFrom(targets, duration, vars, stagger, position, onCompleteAll, onCompleteAllParams) {\n vars.runBackwards = 1;\n _inheritDefaults(vars).immediateRender = _isNotFalse(vars.immediateRender);\n return this.staggerTo(targets, duration, vars, stagger, position, onCompleteAll, onCompleteAllParams);\n };\n\n _proto2.staggerFromTo = function staggerFromTo(targets, duration, fromVars, toVars, stagger, position, onCompleteAll, onCompleteAllParams) {\n toVars.startAt = fromVars;\n _inheritDefaults(toVars).immediateRender = _isNotFalse(toVars.immediateRender);\n return this.staggerTo(targets, duration, toVars, stagger, position, onCompleteAll, onCompleteAllParams);\n };\n\n _proto2.render = function render(totalTime, suppressEvents, force) {\n var prevTime = this._time,\n tDur = this._dirty ? this.totalDuration() : this._tDur,\n dur = this._dur,\n tTime = totalTime <= 0 ? 0 : _roundPrecise(totalTime),\n // if a paused timeline is resumed (or its _start is updated for another reason...which rounds it), that could result in the playhead shifting a **tiny** amount and a zero-duration child at that spot may get rendered at a different ratio, like its totalTime in render() may be 1e-17 instead of 0, for example.\n crossingStart = this._zTime < 0 !== totalTime < 0 && (this._initted || !dur),\n time,\n child,\n next,\n iteration,\n cycleDuration,\n prevPaused,\n pauseTween,\n timeScale,\n prevStart,\n prevIteration,\n yoyo,\n isYoyo;\n this !== _globalTimeline && tTime > tDur && totalTime >= 0 && (tTime = tDur);\n\n if (tTime !== this._tTime || force || crossingStart) {\n if (prevTime !== this._time && dur) {\n //if totalDuration() finds a child with a negative startTime and smoothChildTiming is true, things get shifted around internally so we need to adjust the time accordingly. For example, if a tween starts at -30 we must shift EVERYTHING forward 30 seconds and move this timeline's startTime backward by 30 seconds so that things align with the playhead (no jump).\n tTime += this._time - prevTime;\n totalTime += this._time - prevTime;\n }\n\n time = tTime;\n prevStart = this._start;\n timeScale = this._ts;\n prevPaused = !timeScale;\n\n if (crossingStart) {\n dur || (prevTime = this._zTime); //when the playhead arrives at EXACTLY time 0 (right on top) of a zero-duration timeline, we need to discern if events are suppressed so that when the playhead moves again (next time), it'll trigger the callback. If events are NOT suppressed, obviously the callback would be triggered in this render. Basically, the callback should fire either when the playhead ARRIVES or LEAVES this exact spot, not both. Imagine doing a timeline.seek(0) and there's a callback that sits at 0. Since events are suppressed on that seek() by default, nothing will fire, but when the playhead moves off of that position, the callback should fire. This behavior is what people intuitively expect.\n\n (totalTime || !suppressEvents) && (this._zTime = totalTime);\n }\n\n if (this._repeat) {\n //adjust the time for repeats and yoyos\n yoyo = this._yoyo;\n cycleDuration = dur + this._rDelay;\n\n if (this._repeat < -1 && totalTime < 0) {\n return this.totalTime(cycleDuration * 100 + totalTime, suppressEvents, force);\n }\n\n time = _roundPrecise(tTime % cycleDuration); //round to avoid floating point errors. (4 % 0.8 should be 0 but some browsers report it as 0.79999999!)\n\n if (tTime === tDur) {\n // the tDur === tTime is for edge cases where there's a lengthy decimal on the duration and it may reach the very end but the time is rendered as not-quite-there (remember, tDur is rounded to 4 decimals whereas dur isn't)\n iteration = this._repeat;\n time = dur;\n } else {\n iteration = ~~(tTime / cycleDuration);\n\n if (iteration && iteration === tTime / cycleDuration) {\n time = dur;\n iteration--;\n }\n\n time > dur && (time = dur);\n }\n\n prevIteration = _animationCycle(this._tTime, cycleDuration);\n !prevTime && this._tTime && prevIteration !== iteration && (prevIteration = iteration); // edge case - if someone does addPause() at the very beginning of a repeating timeline, that pause is technically at the same spot as the end which causes this._time to get set to 0 when the totalTime would normally place the playhead at the end. See https://greensock.com/forums/topic/23823-closing-nav-animation-not-working-on-ie-and-iphone-6-maybe-other-older-browser/?tab=comments#comment-113005\n\n if (yoyo && iteration & 1) {\n time = dur - time;\n isYoyo = 1;\n }\n /*\n make sure children at the end/beginning of the timeline are rendered properly. If, for example,\n a 3-second long timeline rendered at 2.9 seconds previously, and now renders at 3.2 seconds (which\n would get translated to 2.8 seconds if the timeline yoyos or 0.2 seconds if it just repeats), there\n could be a callback or a short tween that's at 2.95 or 3 seconds in which wouldn't render. So\n we need to push the timeline to the end (and/or beginning depending on its yoyo value). Also we must\n ensure that zero-duration tweens at the very beginning or end of the Timeline work.\n */\n\n\n if (iteration !== prevIteration && !this._lock) {\n var rewinding = yoyo && prevIteration & 1,\n doesWrap = rewinding === (yoyo && iteration & 1);\n iteration < prevIteration && (rewinding = !rewinding);\n prevTime = rewinding ? 0 : dur;\n this._lock = 1;\n this.render(prevTime || (isYoyo ? 0 : _roundPrecise(iteration * cycleDuration)), suppressEvents, !dur)._lock = 0;\n this._tTime = tTime; // if a user gets the iteration() inside the onRepeat, for example, it should be accurate.\n\n !suppressEvents && this.parent && _callback(this, \"onRepeat\");\n this.vars.repeatRefresh && !isYoyo && (this.invalidate()._lock = 1);\n\n if (prevTime && prevTime !== this._time || prevPaused !== !this._ts || this.vars.onRepeat && !this.parent && !this._act) {\n // if prevTime is 0 and we render at the very end, _time will be the end, thus won't match. So in this edge case, prevTime won't match _time but that's okay. If it gets killed in the onRepeat, eject as well.\n return this;\n }\n\n dur = this._dur; // in case the duration changed in the onRepeat\n\n tDur = this._tDur;\n\n if (doesWrap) {\n this._lock = 2;\n prevTime = rewinding ? dur : -0.0001;\n this.render(prevTime, true);\n this.vars.repeatRefresh && !isYoyo && this.invalidate();\n }\n\n this._lock = 0;\n\n if (!this._ts && !prevPaused) {\n return this;\n } //in order for yoyoEase to work properly when there's a stagger, we must swap out the ease in each sub-tween.\n\n\n _propagateYoyoEase(this, isYoyo);\n }\n }\n\n if (this._hasPause && !this._forcing && this._lock < 2) {\n pauseTween = _findNextPauseTween(this, _roundPrecise(prevTime), _roundPrecise(time));\n\n if (pauseTween) {\n tTime -= time - (time = pauseTween._start);\n }\n }\n\n this._tTime = tTime;\n this._time = time;\n this._act = !timeScale; //as long as it's not paused, force it to be active so that if the user renders independent of the parent timeline, it'll be forced to re-render on the next tick.\n\n if (!this._initted) {\n this._onUpdate = this.vars.onUpdate;\n this._initted = 1;\n this._zTime = totalTime;\n prevTime = 0; // upon init, the playhead should always go forward; someone could invalidate() a completed timeline and then if they restart(), that would make child tweens render in reverse order which could lock in the wrong starting values if they build on each other, like tl.to(obj, {x: 100}).to(obj, {x: 0}).\n }\n\n if (!prevTime && time && !suppressEvents) {\n _callback(this, \"onStart\");\n\n if (this._tTime !== tTime) {\n // in case the onStart triggered a render at a different spot, eject. Like if someone did animation.pause(0.5) or something inside the onStart.\n return this;\n }\n }\n\n if (time >= prevTime && totalTime >= 0) {\n child = this._first;\n\n while (child) {\n next = child._next;\n\n if ((child._act || time >= child._start) && child._ts && pauseTween !== child) {\n if (child.parent !== this) {\n // an extreme edge case - the child's render could do something like kill() the \"next\" one in the linked list, or reparent it. In that case we must re-initiate the whole render to be safe.\n return this.render(totalTime, suppressEvents, force);\n }\n\n child.render(child._ts > 0 ? (time - child._start) * child._ts : (child._dirty ? child.totalDuration() : child._tDur) + (time - child._start) * child._ts, suppressEvents, force);\n\n if (time !== this._time || !this._ts && !prevPaused) {\n //in case a tween pauses or seeks the timeline when rendering, like inside of an onUpdate/onComplete\n pauseTween = 0;\n next && (tTime += this._zTime = -_tinyNum); // it didn't finish rendering, so flag zTime as negative so that so that the next time render() is called it'll be forced (to render any remaining children)\n\n break;\n }\n }\n\n child = next;\n }\n } else {\n force = force || _reverting; // if reverting, we should always force renders. If, for example, a .fromTo() tween with a stagger (which creates an internal timeline) gets reverted BEFORE some of its child tweens render for the first time, it may not properly trigger them to revert.\n\n child = this._last;\n var adjustedTime = totalTime < 0 ? totalTime : time; //when the playhead goes backward beyond the start of this timeline, we must pass that information down to the child animations so that zero-duration tweens know whether to render their starting or ending values.\n\n while (child) {\n next = child._prev;\n\n if ((child._act || adjustedTime <= child._end) && child._ts && pauseTween !== child) {\n if (child.parent !== this) {\n // an extreme edge case - the child's render could do something like kill() the \"next\" one in the linked list, or reparent it. In that case we must re-initiate the whole render to be safe.\n return this.render(totalTime, suppressEvents, force);\n }\n\n child.render(child._ts > 0 ? (adjustedTime - child._start) * child._ts : (child._dirty ? child.totalDuration() : child._tDur) + (adjustedTime - child._start) * child._ts, suppressEvents, force);\n\n if (time !== this._time || !this._ts && !prevPaused) {\n //in case a tween pauses or seeks the timeline when rendering, like inside of an onUpdate/onComplete\n pauseTween = 0;\n next && (tTime += this._zTime = adjustedTime ? -_tinyNum : _tinyNum); // it didn't finish rendering, so adjust zTime so that so that the next time render() is called it'll be forced (to render any remaining children)\n\n break;\n }\n }\n\n child = next;\n }\n }\n\n if (pauseTween && !suppressEvents) {\n this.pause();\n pauseTween.render(time >= prevTime ? 0 : -_tinyNum)._zTime = time >= prevTime ? 1 : -1;\n\n if (this._ts) {\n //the callback resumed playback! So since we may have held back the playhead due to where the pause is positioned, go ahead and jump to where it's SUPPOSED to be (if no pause happened).\n this._start = prevStart; //if the pause was at an earlier time and the user resumed in the callback, it could reposition the timeline (changing its startTime), throwing things off slightly, so we make sure the _start doesn't shift.\n\n _setEnd(this);\n\n return this.render(totalTime, suppressEvents, force);\n }\n }\n\n this._onUpdate && !suppressEvents && _callback(this, \"onUpdate\", true);\n if (tTime === tDur && this._tTime >= this.totalDuration() || !tTime && prevTime) if (prevStart === this._start || Math.abs(timeScale) !== Math.abs(this._ts)) if (!this._lock) {\n // remember, a child's callback may alter this timeline's playhead or timeScale which is why we need to add some of these checks.\n (totalTime || !dur) && (tTime === tDur && this._ts > 0 || !tTime && this._ts < 0) && _removeFromParent(this, 1); // don't remove if the timeline is reversed and the playhead isn't at 0, otherwise tl.progress(1).reverse() won't work. Only remove if the playhead is at the end and timeScale is positive, or if the playhead is at 0 and the timeScale is negative.\n\n if (!suppressEvents && !(totalTime < 0 && !prevTime) && (tTime || prevTime || !tDur)) {\n _callback(this, tTime === tDur && totalTime >= 0 ? \"onComplete\" : \"onReverseComplete\", true);\n\n this._prom && !(tTime < tDur && this.timeScale() > 0) && this._prom();\n }\n }\n }\n\n return this;\n };\n\n _proto2.add = function add(child, position) {\n var _this2 = this;\n\n _isNumber(position) || (position = _parsePosition(this, position, child));\n\n if (!(child instanceof Animation)) {\n if (_isArray(child)) {\n child.forEach(function (obj) {\n return _this2.add(obj, position);\n });\n return this;\n }\n\n if (_isString(child)) {\n return this.addLabel(child, position);\n }\n\n if (_isFunction(child)) {\n child = Tween.delayedCall(0, child);\n } else {\n return this;\n }\n }\n\n return this !== child ? _addToTimeline(this, child, position) : this; //don't allow a timeline to be added to itself as a child!\n };\n\n _proto2.getChildren = function getChildren(nested, tweens, timelines, ignoreBeforeTime) {\n if (nested === void 0) {\n nested = true;\n }\n\n if (tweens === void 0) {\n tweens = true;\n }\n\n if (timelines === void 0) {\n timelines = true;\n }\n\n if (ignoreBeforeTime === void 0) {\n ignoreBeforeTime = -_bigNum;\n }\n\n var a = [],\n child = this._first;\n\n while (child) {\n if (child._start >= ignoreBeforeTime) {\n if (child instanceof Tween) {\n tweens && a.push(child);\n } else {\n timelines && a.push(child);\n nested && a.push.apply(a, child.getChildren(true, tweens, timelines));\n }\n }\n\n child = child._next;\n }\n\n return a;\n };\n\n _proto2.getById = function getById(id) {\n var animations = this.getChildren(1, 1, 1),\n i = animations.length;\n\n while (i--) {\n if (animations[i].vars.id === id) {\n return animations[i];\n }\n }\n };\n\n _proto2.remove = function remove(child) {\n if (_isString(child)) {\n return this.removeLabel(child);\n }\n\n if (_isFunction(child)) {\n return this.killTweensOf(child);\n }\n\n _removeLinkedListItem(this, child);\n\n if (child === this._recent) {\n this._recent = this._last;\n }\n\n return _uncache(this);\n };\n\n _proto2.totalTime = function totalTime(_totalTime2, suppressEvents) {\n if (!arguments.length) {\n return this._tTime;\n }\n\n this._forcing = 1;\n\n if (!this._dp && this._ts) {\n //special case for the global timeline (or any other that has no parent or detached parent).\n this._start = _roundPrecise(_ticker.time - (this._ts > 0 ? _totalTime2 / this._ts : (this.totalDuration() - _totalTime2) / -this._ts));\n }\n\n _Animation.prototype.totalTime.call(this, _totalTime2, suppressEvents);\n\n this._forcing = 0;\n return this;\n };\n\n _proto2.addLabel = function addLabel(label, position) {\n this.labels[label] = _parsePosition(this, position);\n return this;\n };\n\n _proto2.removeLabel = function removeLabel(label) {\n delete this.labels[label];\n return this;\n };\n\n _proto2.addPause = function addPause(position, callback, params) {\n var t = Tween.delayedCall(0, callback || _emptyFunc, params);\n t.data = \"isPause\";\n this._hasPause = 1;\n return _addToTimeline(this, t, _parsePosition(this, position));\n };\n\n _proto2.removePause = function removePause(position) {\n var child = this._first;\n position = _parsePosition(this, position);\n\n while (child) {\n if (child._start === position && child.data === \"isPause\") {\n _removeFromParent(child);\n }\n\n child = child._next;\n }\n };\n\n _proto2.killTweensOf = function killTweensOf(targets, props, onlyActive) {\n var tweens = this.getTweensOf(targets, onlyActive),\n i = tweens.length;\n\n while (i--) {\n _overwritingTween !== tweens[i] && tweens[i].kill(targets, props);\n }\n\n return this;\n };\n\n _proto2.getTweensOf = function getTweensOf(targets, onlyActive) {\n var a = [],\n parsedTargets = toArray(targets),\n child = this._first,\n isGlobalTime = _isNumber(onlyActive),\n // a number is interpreted as a global time. If the animation spans\n children;\n\n while (child) {\n if (child instanceof Tween) {\n if (_arrayContainsAny(child._targets, parsedTargets) && (isGlobalTime ? (!_overwritingTween || child._initted && child._ts) && child.globalTime(0) <= onlyActive && child.globalTime(child.totalDuration()) > onlyActive : !onlyActive || child.isActive())) {\n // note: if this is for overwriting, it should only be for tweens that aren't paused and are initted.\n a.push(child);\n }\n } else if ((children = child.getTweensOf(parsedTargets, onlyActive)).length) {\n a.push.apply(a, children);\n }\n\n child = child._next;\n }\n\n return a;\n } // potential future feature - targets() on timelines\n // targets() {\n // \tlet result = [];\n // \tthis.getChildren(true, true, false).forEach(t => result.push(...t.targets()));\n // \treturn result.filter((v, i) => result.indexOf(v) === i);\n // }\n ;\n\n _proto2.tweenTo = function tweenTo(position, vars) {\n vars = vars || {};\n\n var tl = this,\n endTime = _parsePosition(tl, position),\n _vars = vars,\n startAt = _vars.startAt,\n _onStart = _vars.onStart,\n onStartParams = _vars.onStartParams,\n immediateRender = _vars.immediateRender,\n initted,\n tween = Tween.to(tl, _setDefaults({\n ease: vars.ease || \"none\",\n lazy: false,\n immediateRender: false,\n time: endTime,\n overwrite: \"auto\",\n duration: vars.duration || Math.abs((endTime - (startAt && \"time\" in startAt ? startAt.time : tl._time)) / tl.timeScale()) || _tinyNum,\n onStart: function onStart() {\n tl.pause();\n\n if (!initted) {\n var duration = vars.duration || Math.abs((endTime - (startAt && \"time\" in startAt ? startAt.time : tl._time)) / tl.timeScale());\n tween._dur !== duration && _setDuration(tween, duration, 0, 1).render(tween._time, true, true);\n initted = 1;\n }\n\n _onStart && _onStart.apply(tween, onStartParams || []); //in case the user had an onStart in the vars - we don't want to overwrite it.\n }\n }, vars));\n\n return immediateRender ? tween.render(0) : tween;\n };\n\n _proto2.tweenFromTo = function tweenFromTo(fromPosition, toPosition, vars) {\n return this.tweenTo(toPosition, _setDefaults({\n startAt: {\n time: _parsePosition(this, fromPosition)\n }\n }, vars));\n };\n\n _proto2.recent = function recent() {\n return this._recent;\n };\n\n _proto2.nextLabel = function nextLabel(afterTime) {\n if (afterTime === void 0) {\n afterTime = this._time;\n }\n\n return _getLabelInDirection(this, _parsePosition(this, afterTime));\n };\n\n _proto2.previousLabel = function previousLabel(beforeTime) {\n if (beforeTime === void 0) {\n beforeTime = this._time;\n }\n\n return _getLabelInDirection(this, _parsePosition(this, beforeTime), 1);\n };\n\n _proto2.currentLabel = function currentLabel(value) {\n return arguments.length ? this.seek(value, true) : this.previousLabel(this._time + _tinyNum);\n };\n\n _proto2.shiftChildren = function shiftChildren(amount, adjustLabels, ignoreBeforeTime) {\n if (ignoreBeforeTime === void 0) {\n ignoreBeforeTime = 0;\n }\n\n var child = this._first,\n labels = this.labels,\n p;\n\n while (child) {\n if (child._start >= ignoreBeforeTime) {\n child._start += amount;\n child._end += amount;\n }\n\n child = child._next;\n }\n\n if (adjustLabels) {\n for (p in labels) {\n if (labels[p] >= ignoreBeforeTime) {\n labels[p] += amount;\n }\n }\n }\n\n return _uncache(this);\n };\n\n _proto2.invalidate = function invalidate() {\n var child = this._first;\n this._lock = 0;\n\n while (child) {\n child.invalidate();\n child = child._next;\n }\n\n return _Animation.prototype.invalidate.call(this);\n };\n\n _proto2.clear = function clear(includeLabels) {\n if (includeLabels === void 0) {\n includeLabels = true;\n }\n\n var child = this._first,\n next;\n\n while (child) {\n next = child._next;\n this.remove(child);\n child = next;\n }\n\n this._dp && (this._time = this._tTime = this._pTime = 0);\n includeLabels && (this.labels = {});\n return _uncache(this);\n };\n\n _proto2.totalDuration = function totalDuration(value) {\n var max = 0,\n self = this,\n child = self._last,\n prevStart = _bigNum,\n prev,\n start,\n parent;\n\n if (arguments.length) {\n return self.timeScale((self._repeat < 0 ? self.duration() : self.totalDuration()) / (self.reversed() ? -value : value));\n }\n\n if (self._dirty) {\n parent = self.parent;\n\n while (child) {\n prev = child._prev; //record it here in case the tween changes position in the sequence...\n\n child._dirty && child.totalDuration(); //could change the tween._startTime, so make sure the animation's cache is clean before analyzing it.\n\n start = child._start;\n\n if (start > prevStart && self._sort && child._ts && !self._lock) {\n //in case one of the tweens shifted out of order, it needs to be re-inserted into the correct position in the sequence\n self._lock = 1; //prevent endless recursive calls - there are methods that get triggered that check duration/totalDuration when we add().\n\n _addToTimeline(self, child, start - child._delay, 1)._lock = 0;\n } else {\n prevStart = start;\n }\n\n if (start < 0 && child._ts) {\n //children aren't allowed to have negative startTimes unless smoothChildTiming is true, so adjust here if one is found.\n max -= start;\n\n if (!parent && !self._dp || parent && parent.smoothChildTiming) {\n self._start += start / self._ts;\n self._time -= start;\n self._tTime -= start;\n }\n\n self.shiftChildren(-start, false, -1e999);\n prevStart = 0;\n }\n\n child._end > max && child._ts && (max = child._end);\n child = prev;\n }\n\n _setDuration(self, self === _globalTimeline && self._time > max ? self._time : max, 1, 1);\n\n self._dirty = 0;\n }\n\n return self._tDur;\n };\n\n Timeline.updateRoot = function updateRoot(time) {\n if (_globalTimeline._ts) {\n _lazySafeRender(_globalTimeline, _parentToChildTotalTime(time, _globalTimeline));\n\n _lastRenderedFrame = _ticker.frame;\n }\n\n if (_ticker.frame >= _nextGCFrame) {\n _nextGCFrame += _config.autoSleep || 120;\n var child = _globalTimeline._first;\n if (!child || !child._ts) if (_config.autoSleep && _ticker._listeners.length < 2) {\n while (child && !child._ts) {\n child = child._next;\n }\n\n child || _ticker.sleep();\n }\n }\n };\n\n return Timeline;\n}(Animation);\n\n_setDefaults(Timeline.prototype, {\n _lock: 0,\n _hasPause: 0,\n _forcing: 0\n});\n\nvar _addComplexStringPropTween = function _addComplexStringPropTween(target, prop, start, end, setter, stringFilter, funcParam) {\n //note: we call _addComplexStringPropTween.call(tweenInstance...) to ensure that it's scoped properly. We may call it from within a plugin too, thus \"this\" would refer to the plugin.\n var pt = new PropTween(this._pt, target, prop, 0, 1, _renderComplexString, null, setter),\n index = 0,\n matchIndex = 0,\n result,\n startNums,\n color,\n endNum,\n chunk,\n startNum,\n hasRandom,\n a;\n pt.b = start;\n pt.e = end;\n start += \"\"; //ensure values are strings\n\n end += \"\";\n\n if (hasRandom = ~end.indexOf(\"random(\")) {\n end = _replaceRandom(end);\n }\n\n if (stringFilter) {\n a = [start, end];\n stringFilter(a, target, prop); //pass an array with the starting and ending values and let the filter do whatever it needs to the values.\n\n start = a[0];\n end = a[1];\n }\n\n startNums = start.match(_complexStringNumExp) || [];\n\n while (result = _complexStringNumExp.exec(end)) {\n endNum = result[0];\n chunk = end.substring(index, result.index);\n\n if (color) {\n color = (color + 1) % 5;\n } else if (chunk.substr(-5) === \"rgba(\") {\n color = 1;\n }\n\n if (endNum !== startNums[matchIndex++]) {\n startNum = parseFloat(startNums[matchIndex - 1]) || 0; //these nested PropTweens are handled in a special way - we'll never actually call a render or setter method on them. We'll just loop through them in the parent complex string PropTween's render method.\n\n pt._pt = {\n _next: pt._pt,\n p: chunk || matchIndex === 1 ? chunk : \",\",\n //note: SVG spec allows omission of comma/space when a negative sign is wedged between two numbers, like 2.5-5.3 instead of 2.5,-5.3 but when tweening, the negative value may switch to positive, so we insert the comma just in case.\n s: startNum,\n c: endNum.charAt(1) === \"=\" ? _parseRelative(startNum, endNum) - startNum : parseFloat(endNum) - startNum,\n m: color && color < 4 ? Math.round : 0\n };\n index = _complexStringNumExp.lastIndex;\n }\n }\n\n pt.c = index < end.length ? end.substring(index, end.length) : \"\"; //we use the \"c\" of the PropTween to store the final part of the string (after the last number)\n\n pt.fp = funcParam;\n\n if (_relExp.test(end) || hasRandom) {\n pt.e = 0; //if the end string contains relative values or dynamic random(...) values, delete the end it so that on the final render we don't actually set it to the string with += or -= characters (forces it to use the calculated value).\n }\n\n this._pt = pt; //start the linked list with this new PropTween. Remember, we call _addComplexStringPropTween.call(tweenInstance...) to ensure that it's scoped properly. We may call it from within a plugin too, thus \"this\" would refer to the plugin.\n\n return pt;\n},\n _addPropTween = function _addPropTween(target, prop, start, end, index, targets, modifier, stringFilter, funcParam, optional) {\n _isFunction(end) && (end = end(index || 0, target, targets));\n var currentValue = target[prop],\n parsedStart = start !== \"get\" ? start : !_isFunction(currentValue) ? currentValue : funcParam ? target[prop.indexOf(\"set\") || !_isFunction(target[\"get\" + prop.substr(3)]) ? prop : \"get\" + prop.substr(3)](funcParam) : target[prop](),\n setter = !_isFunction(currentValue) ? _setterPlain : funcParam ? _setterFuncWithParam : _setterFunc,\n pt;\n\n if (_isString(end)) {\n if (~end.indexOf(\"random(\")) {\n end = _replaceRandom(end);\n }\n\n if (end.charAt(1) === \"=\") {\n pt = _parseRelative(parsedStart, end) + (getUnit(parsedStart) || 0);\n\n if (pt || pt === 0) {\n // to avoid isNaN, like if someone passes in a value like \"!= whatever\"\n end = pt;\n }\n }\n }\n\n if (!optional || parsedStart !== end || _forceAllPropTweens) {\n if (!isNaN(parsedStart * end) && end !== \"\") {\n // fun fact: any number multiplied by \"\" is evaluated as the number 0!\n pt = new PropTween(this._pt, target, prop, +parsedStart || 0, end - (parsedStart || 0), typeof currentValue === \"boolean\" ? _renderBoolean : _renderPlain, 0, setter);\n funcParam && (pt.fp = funcParam);\n modifier && pt.modifier(modifier, this, target);\n return this._pt = pt;\n }\n\n !currentValue && !(prop in target) && _missingPlugin(prop, end);\n return _addComplexStringPropTween.call(this, target, prop, parsedStart, end, setter, stringFilter || _config.stringFilter, funcParam);\n }\n},\n //creates a copy of the vars object and processes any function-based values (putting the resulting values directly into the copy) as well as strings with \"random()\" in them. It does NOT process relative values.\n_processVars = function _processVars(vars, index, target, targets, tween) {\n _isFunction(vars) && (vars = _parseFuncOrString(vars, tween, index, target, targets));\n\n if (!_isObject(vars) || vars.style && vars.nodeType || _isArray(vars) || _isTypedArray(vars)) {\n return _isString(vars) ? _parseFuncOrString(vars, tween, index, target, targets) : vars;\n }\n\n var copy = {},\n p;\n\n for (p in vars) {\n copy[p] = _parseFuncOrString(vars[p], tween, index, target, targets);\n }\n\n return copy;\n},\n _checkPlugin = function _checkPlugin(property, vars, tween, index, target, targets) {\n var plugin, pt, ptLookup, i;\n\n if (_plugins[property] && (plugin = new _plugins[property]()).init(target, plugin.rawVars ? vars[property] : _processVars(vars[property], index, target, targets, tween), tween, index, targets) !== false) {\n tween._pt = pt = new PropTween(tween._pt, target, property, 0, 1, plugin.render, plugin, 0, plugin.priority);\n\n if (tween !== _quickTween) {\n ptLookup = tween._ptLookup[tween._targets.indexOf(target)]; //note: we can't use tween._ptLookup[index] because for staggered tweens, the index from the fullTargets array won't match what it is in each individual tween that spawns from the stagger.\n\n i = plugin._props.length;\n\n while (i--) {\n ptLookup[plugin._props[i]] = pt;\n }\n }\n }\n\n return plugin;\n},\n _overwritingTween,\n //store a reference temporarily so we can avoid overwriting itself.\n_forceAllPropTweens,\n _initTween = function _initTween(tween, time) {\n var vars = tween.vars,\n ease = vars.ease,\n startAt = vars.startAt,\n immediateRender = vars.immediateRender,\n lazy = vars.lazy,\n onUpdate = vars.onUpdate,\n onUpdateParams = vars.onUpdateParams,\n callbackScope = vars.callbackScope,\n runBackwards = vars.runBackwards,\n yoyoEase = vars.yoyoEase,\n keyframes = vars.keyframes,\n autoRevert = vars.autoRevert,\n dur = tween._dur,\n prevStartAt = tween._startAt,\n targets = tween._targets,\n parent = tween.parent,\n fullTargets = parent && parent.data === \"nested\" ? parent.vars.targets : targets,\n autoOverwrite = tween._overwrite === \"auto\" && !_suppressOverwrites,\n tl = tween.timeline,\n cleanVars,\n i,\n p,\n pt,\n target,\n hasPriority,\n gsData,\n harness,\n plugin,\n ptLookup,\n index,\n harnessVars,\n overwritten;\n tl && (!keyframes || !ease) && (ease = \"none\");\n tween._ease = _parseEase(ease, _defaults.ease);\n tween._yEase = yoyoEase ? _invertEase(_parseEase(yoyoEase === true ? ease : yoyoEase, _defaults.ease)) : 0;\n\n if (yoyoEase && tween._yoyo && !tween._repeat) {\n //there must have been a parent timeline with yoyo:true that is currently in its yoyo phase, so flip the eases.\n yoyoEase = tween._yEase;\n tween._yEase = tween._ease;\n tween._ease = yoyoEase;\n }\n\n tween._from = !tl && !!vars.runBackwards; //nested timelines should never run backwards - the backwards-ness is in the child tweens.\n\n if (!tl || keyframes && !vars.stagger) {\n //if there's an internal timeline, skip all the parsing because we passed that task down the chain.\n harness = targets[0] ? _getCache(targets[0]).harness : 0;\n harnessVars = harness && vars[harness.prop]; //someone may need to specify CSS-specific values AND non-CSS values, like if the element has an \"x\" property plus it's a standard DOM element. We allow people to distinguish by wrapping plugin-specific stuff in a css:{} object for example.\n\n cleanVars = _copyExcluding(vars, _reservedProps);\n\n if (prevStartAt) {\n time < 0 && runBackwards && immediateRender && !autoRevert ? prevStartAt.render(-1, true) : prevStartAt.revert(runBackwards && dur ? _revertConfig : _startAtRevertConfig); // if it's a \"startAt\" (not \"from()\" or runBackwards: true), we only need to do a shallow revert (keep transforms cached in CSSPlugin)\n // don't just _removeFromParent(prevStartAt.render(-1, true)) because that'll leave inline styles. We're creating a new _startAt for \"startAt\" tweens that re-capture things to ensure that if the pre-tween values changed since the tween was created, they're recorded.\n\n prevStartAt._lazy = 0;\n }\n\n if (startAt) {\n _removeFromParent(tween._startAt = Tween.set(targets, _setDefaults({\n data: \"isStart\",\n overwrite: false,\n parent: parent,\n immediateRender: true,\n lazy: _isNotFalse(lazy),\n startAt: null,\n delay: 0,\n onUpdate: onUpdate,\n onUpdateParams: onUpdateParams,\n callbackScope: callbackScope,\n stagger: 0\n }, startAt))); //copy the properties/values into a new object to avoid collisions, like var to = {x:0}, from = {x:500}; timeline.fromTo(e, from, to).fromTo(e, to, from);\n\n\n time < 0 && (_reverting || !immediateRender && !autoRevert) && tween._startAt.revert(_revertConfig); // rare edge case, like if a render is forced in the negative direction of a non-initted tween.\n\n if (immediateRender) {\n if (dur && time <= 0) {\n time && (tween._zTime = time);\n return; //we skip initialization here so that overwriting doesn't occur until the tween actually begins. Otherwise, if you create several immediateRender:true tweens of the same target/properties to drop into a Timeline, the last one created would overwrite the first ones because they didn't get placed into the timeline yet before the first render occurs and kicks in overwriting.\n }\n }\n } else if (runBackwards && dur) {\n //from() tweens must be handled uniquely: their beginning values must be rendered but we don't want overwriting to occur yet (when time is still 0). Wait until the tween actually begins before doing all the routines like overwriting. At that time, we should render at the END of the tween to ensure that things initialize correctly (remember, from() tweens go backwards)\n if (!prevStartAt) {\n time && (immediateRender = false); //in rare cases (like if a from() tween runs and then is invalidate()-ed), immediateRender could be true but the initial forced-render gets skipped, so there's no need to force the render in this context when the _time is greater than 0\n\n p = _setDefaults({\n overwrite: false,\n data: \"isFromStart\",\n //we tag the tween with as \"isFromStart\" so that if [inside a plugin] we need to only do something at the very END of a tween, we have a way of identifying this tween as merely the one that's setting the beginning values for a \"from()\" tween. For example, clearProps in CSSPlugin should only get applied at the very END of a tween and without this tag, from(...{height:100, clearProps:\"height\", delay:1}) would wipe the height at the beginning of the tween and after 1 second, it'd kick back in.\n lazy: immediateRender && _isNotFalse(lazy),\n immediateRender: immediateRender,\n //zero-duration tweens render immediately by default, but if we're not specifically instructed to render this tween immediately, we should skip this and merely _init() to record the starting values (rendering them immediately would push them to completion which is wasteful in that case - we'd have to render(-1) immediately after)\n stagger: 0,\n parent: parent //ensures that nested tweens that had a stagger are handled properly, like gsap.from(\".class\", {y:gsap.utils.wrap([-100,100])})\n\n }, cleanVars);\n harnessVars && (p[harness.prop] = harnessVars); // in case someone does something like .from(..., {css:{}})\n\n _removeFromParent(tween._startAt = Tween.set(targets, p));\n\n time < 0 && (_reverting ? tween._startAt.revert(_revertConfig) : tween._startAt.render(-1, true));\n tween._zTime = time;\n\n if (!immediateRender) {\n _initTween(tween._startAt, _tinyNum); //ensures that the initial values are recorded\n\n } else if (!time) {\n return;\n }\n }\n }\n\n tween._pt = tween._ptCache = 0;\n lazy = dur && _isNotFalse(lazy) || lazy && !dur;\n\n for (i = 0; i < targets.length; i++) {\n target = targets[i];\n gsData = target._gsap || _harness(targets)[i]._gsap;\n tween._ptLookup[i] = ptLookup = {};\n _lazyLookup[gsData.id] && _lazyTweens.length && _lazyRender(); //if other tweens of the same target have recently initted but haven't rendered yet, we've got to force the render so that the starting values are correct (imagine populating a timeline with a bunch of sequential tweens and then jumping to the end)\n\n index = fullTargets === targets ? i : fullTargets.indexOf(target);\n\n if (harness && (plugin = new harness()).init(target, harnessVars || cleanVars, tween, index, fullTargets) !== false) {\n tween._pt = pt = new PropTween(tween._pt, target, plugin.name, 0, 1, plugin.render, plugin, 0, plugin.priority);\n\n plugin._props.forEach(function (name) {\n ptLookup[name] = pt;\n });\n\n plugin.priority && (hasPriority = 1);\n }\n\n if (!harness || harnessVars) {\n for (p in cleanVars) {\n if (_plugins[p] && (plugin = _checkPlugin(p, cleanVars, tween, index, target, fullTargets))) {\n plugin.priority && (hasPriority = 1);\n } else {\n ptLookup[p] = pt = _addPropTween.call(tween, target, p, \"get\", cleanVars[p], index, fullTargets, 0, vars.stringFilter);\n }\n }\n }\n\n tween._op && tween._op[i] && tween.kill(target, tween._op[i]);\n\n if (autoOverwrite && tween._pt) {\n _overwritingTween = tween;\n\n _globalTimeline.killTweensOf(target, ptLookup, tween.globalTime(time)); // make sure the overwriting doesn't overwrite THIS tween!!!\n\n\n overwritten = !tween.parent;\n _overwritingTween = 0;\n }\n\n tween._pt && lazy && (_lazyLookup[gsData.id] = 1);\n }\n\n hasPriority && _sortPropTweensByPriority(tween);\n tween._onInit && tween._onInit(tween); //plugins like RoundProps must wait until ALL of the PropTweens are instantiated. In the plugin's init() function, it sets the _onInit on the tween instance. May not be pretty/intuitive, but it's fast and keeps file size down.\n }\n\n tween._onUpdate = onUpdate;\n tween._initted = (!tween._op || tween._pt) && !overwritten; // if overwrittenProps resulted in the entire tween being killed, do NOT flag it as initted or else it may render for one tick.\n\n keyframes && time <= 0 && tl.render(_bigNum, true, true); // if there's a 0% keyframe, it'll render in the \"before\" state for any staggered/delayed animations thus when the following tween initializes, it'll use the \"before\" state instead of the \"after\" state as the initial values.\n},\n _updatePropTweens = function _updatePropTweens(tween, property, value, start, startIsRelative, ratio, time) {\n var ptCache = (tween._pt && tween._ptCache || (tween._ptCache = {}))[property],\n pt,\n rootPT,\n lookup,\n i;\n\n if (!ptCache) {\n ptCache = tween._ptCache[property] = [];\n lookup = tween._ptLookup;\n i = tween._targets.length;\n\n while (i--) {\n pt = lookup[i][property];\n\n if (pt && pt.d && pt.d._pt) {\n // it's a plugin, so find the nested PropTween\n pt = pt.d._pt;\n\n while (pt && pt.p !== property && pt.fp !== property) {\n // \"fp\" is functionParam for things like setting CSS variables which require .setProperty(\"--var-name\", value)\n pt = pt._next;\n }\n }\n\n if (!pt) {\n // there is no PropTween associated with that property, so we must FORCE one to be created and ditch out of this\n // if the tween has other properties that already rendered at new positions, we'd normally have to rewind to put them back like tween.render(0, true) before forcing an _initTween(), but that can create another edge case like tweening a timeline's progress would trigger onUpdates to fire which could move other things around. It's better to just inform users that .resetTo() should ONLY be used for tweens that already have that property. For example, you can't gsap.to(...{ y: 0 }) and then tween.restTo(\"x\", 200) for example.\n _forceAllPropTweens = 1; // otherwise, when we _addPropTween() and it finds no change between the start and end values, it skips creating a PropTween (for efficiency...why tween when there's no difference?) but in this case we NEED that PropTween created so we can edit it.\n\n tween.vars[property] = \"+=0\";\n\n _initTween(tween, time);\n\n _forceAllPropTweens = 0;\n return 1;\n }\n\n ptCache.push(pt);\n }\n }\n\n i = ptCache.length;\n\n while (i--) {\n rootPT = ptCache[i];\n pt = rootPT._pt || rootPT; // complex values may have nested PropTweens. We only accommodate the FIRST value.\n\n pt.s = (start || start === 0) && !startIsRelative ? start : pt.s + (start || 0) + ratio * pt.c;\n pt.c = value - pt.s;\n rootPT.e && (rootPT.e = _round(value) + getUnit(rootPT.e)); // mainly for CSSPlugin (end value)\n\n rootPT.b && (rootPT.b = pt.s + getUnit(rootPT.b)); // (beginning value)\n }\n},\n _addAliasesToVars = function _addAliasesToVars(targets, vars) {\n var harness = targets[0] ? _getCache(targets[0]).harness : 0,\n propertyAliases = harness && harness.aliases,\n copy,\n p,\n i,\n aliases;\n\n if (!propertyAliases) {\n return vars;\n }\n\n copy = _merge({}, vars);\n\n for (p in propertyAliases) {\n if (p in copy) {\n aliases = propertyAliases[p].split(\",\");\n i = aliases.length;\n\n while (i--) {\n copy[aliases[i]] = copy[p];\n }\n }\n }\n\n return copy;\n},\n // parses multiple formats, like {\"0%\": {x: 100}, {\"50%\": {x: -20}} and { x: {\"0%\": 100, \"50%\": -20} }, and an \"ease\" can be set on any object. We populate an \"allProps\" object with an Array for each property, like {x: [{}, {}], y:[{}, {}]} with data for each property tween. The objects have a \"t\" (time), \"v\", (value), and \"e\" (ease) property. This allows us to piece together a timeline later.\n_parseKeyframe = function _parseKeyframe(prop, obj, allProps, easeEach) {\n var ease = obj.ease || easeEach || \"power1.inOut\",\n p,\n a;\n\n if (_isArray(obj)) {\n a = allProps[prop] || (allProps[prop] = []); // t = time (out of 100), v = value, e = ease\n\n obj.forEach(function (value, i) {\n return a.push({\n t: i / (obj.length - 1) * 100,\n v: value,\n e: ease\n });\n });\n } else {\n for (p in obj) {\n a = allProps[p] || (allProps[p] = []);\n p === \"ease\" || a.push({\n t: parseFloat(prop),\n v: obj[p],\n e: ease\n });\n }\n }\n},\n _parseFuncOrString = function _parseFuncOrString(value, tween, i, target, targets) {\n return _isFunction(value) ? value.call(tween, i, target, targets) : _isString(value) && ~value.indexOf(\"random(\") ? _replaceRandom(value) : value;\n},\n _staggerTweenProps = _callbackNames + \"repeat,repeatDelay,yoyo,repeatRefresh,yoyoEase,autoRevert\",\n _staggerPropsToSkip = {};\n\n_forEachName(_staggerTweenProps + \",id,stagger,delay,duration,paused,scrollTrigger\", function (name) {\n return _staggerPropsToSkip[name] = 1;\n});\n/*\n * --------------------------------------------------------------------------------------\n * TWEEN\n * --------------------------------------------------------------------------------------\n */\n\n\nexport var Tween = /*#__PURE__*/function (_Animation2) {\n _inheritsLoose(Tween, _Animation2);\n\n function Tween(targets, vars, position, skipInherit) {\n var _this3;\n\n if (typeof vars === \"number\") {\n position.duration = vars;\n vars = position;\n position = null;\n }\n\n _this3 = _Animation2.call(this, skipInherit ? vars : _inheritDefaults(vars)) || this;\n var _this3$vars = _this3.vars,\n duration = _this3$vars.duration,\n delay = _this3$vars.delay,\n immediateRender = _this3$vars.immediateRender,\n stagger = _this3$vars.stagger,\n overwrite = _this3$vars.overwrite,\n keyframes = _this3$vars.keyframes,\n defaults = _this3$vars.defaults,\n scrollTrigger = _this3$vars.scrollTrigger,\n yoyoEase = _this3$vars.yoyoEase,\n parent = vars.parent || _globalTimeline,\n parsedTargets = (_isArray(targets) || _isTypedArray(targets) ? _isNumber(targets[0]) : \"length\" in vars) ? [targets] : toArray(targets),\n tl,\n i,\n copy,\n l,\n p,\n curTarget,\n staggerFunc,\n staggerVarsToMerge;\n _this3._targets = parsedTargets.length ? _harness(parsedTargets) : _warn(\"GSAP target \" + targets + \" not found. https://greensock.com\", !_config.nullTargetWarn) || [];\n _this3._ptLookup = []; //PropTween lookup. An array containing an object for each target, having keys for each tweening property\n\n _this3._overwrite = overwrite;\n\n if (keyframes || stagger || _isFuncOrString(duration) || _isFuncOrString(delay)) {\n vars = _this3.vars;\n tl = _this3.timeline = new Timeline({\n data: \"nested\",\n defaults: defaults || {},\n targets: parent && parent.data === \"nested\" ? parent.vars.targets : parsedTargets\n }); // we need to store the targets because for staggers and keyframes, we end up creating an individual tween for each but function-based values need to know the index and the whole Array of targets.\n\n tl.kill();\n tl.parent = tl._dp = _assertThisInitialized(_this3);\n tl._start = 0;\n\n if (stagger || _isFuncOrString(duration) || _isFuncOrString(delay)) {\n l = parsedTargets.length;\n staggerFunc = stagger && distribute(stagger);\n\n if (_isObject(stagger)) {\n //users can pass in callbacks like onStart/onComplete in the stagger object. These should fire with each individual tween.\n for (p in stagger) {\n if (~_staggerTweenProps.indexOf(p)) {\n staggerVarsToMerge || (staggerVarsToMerge = {});\n staggerVarsToMerge[p] = stagger[p];\n }\n }\n }\n\n for (i = 0; i < l; i++) {\n copy = _copyExcluding(vars, _staggerPropsToSkip);\n copy.stagger = 0;\n yoyoEase && (copy.yoyoEase = yoyoEase);\n staggerVarsToMerge && _merge(copy, staggerVarsToMerge);\n curTarget = parsedTargets[i]; //don't just copy duration or delay because if they're a string or function, we'd end up in an infinite loop because _isFuncOrString() would evaluate as true in the child tweens, entering this loop, etc. So we parse the value straight from vars and default to 0.\n\n copy.duration = +_parseFuncOrString(duration, _assertThisInitialized(_this3), i, curTarget, parsedTargets);\n copy.delay = (+_parseFuncOrString(delay, _assertThisInitialized(_this3), i, curTarget, parsedTargets) || 0) - _this3._delay;\n\n if (!stagger && l === 1 && copy.delay) {\n // if someone does delay:\"random(1, 5)\", repeat:-1, for example, the delay shouldn't be inside the repeat.\n _this3._delay = delay = copy.delay;\n _this3._start += delay;\n copy.delay = 0;\n }\n\n tl.to(curTarget, copy, staggerFunc ? staggerFunc(i, curTarget, parsedTargets) : 0);\n tl._ease = _easeMap.none;\n }\n\n tl.duration() ? duration = delay = 0 : _this3.timeline = 0; // if the timeline's duration is 0, we don't need a timeline internally!\n } else if (keyframes) {\n _inheritDefaults(_setDefaults(tl.vars.defaults, {\n ease: \"none\"\n }));\n\n tl._ease = _parseEase(keyframes.ease || vars.ease || \"none\");\n var time = 0,\n a,\n kf,\n v;\n\n if (_isArray(keyframes)) {\n keyframes.forEach(function (frame) {\n return tl.to(parsedTargets, frame, \">\");\n });\n tl.duration(); // to ensure tl._dur is cached because we tap into it for performance purposes in the render() method.\n } else {\n copy = {};\n\n for (p in keyframes) {\n p === \"ease\" || p === \"easeEach\" || _parseKeyframe(p, keyframes[p], copy, keyframes.easeEach);\n }\n\n for (p in copy) {\n a = copy[p].sort(function (a, b) {\n return a.t - b.t;\n });\n time = 0;\n\n for (i = 0; i < a.length; i++) {\n kf = a[i];\n v = {\n ease: kf.e,\n duration: (kf.t - (i ? a[i - 1].t : 0)) / 100 * duration\n };\n v[p] = kf.v;\n tl.to(parsedTargets, v, time);\n time += v.duration;\n }\n }\n\n tl.duration() < duration && tl.to({}, {\n duration: duration - tl.duration()\n }); // in case keyframes didn't go to 100%\n }\n }\n\n duration || _this3.duration(duration = tl.duration());\n } else {\n _this3.timeline = 0; //speed optimization, faster lookups (no going up the prototype chain)\n }\n\n if (overwrite === true && !_suppressOverwrites) {\n _overwritingTween = _assertThisInitialized(_this3);\n\n _globalTimeline.killTweensOf(parsedTargets);\n\n _overwritingTween = 0;\n }\n\n _addToTimeline(parent, _assertThisInitialized(_this3), position);\n\n vars.reversed && _this3.reverse();\n vars.paused && _this3.paused(true);\n\n if (immediateRender || !duration && !keyframes && _this3._start === _roundPrecise(parent._time) && _isNotFalse(immediateRender) && _hasNoPausedAncestors(_assertThisInitialized(_this3)) && parent.data !== \"nested\") {\n _this3._tTime = -_tinyNum; //forces a render without having to set the render() \"force\" parameter to true because we want to allow lazying by default (using the \"force\" parameter always forces an immediate full render)\n\n _this3.render(Math.max(0, -delay)); //in case delay is negative\n\n }\n\n scrollTrigger && _scrollTrigger(_assertThisInitialized(_this3), scrollTrigger);\n return _this3;\n }\n\n var _proto3 = Tween.prototype;\n\n _proto3.render = function render(totalTime, suppressEvents, force) {\n var prevTime = this._time,\n tDur = this._tDur,\n dur = this._dur,\n isNegative = totalTime < 0,\n tTime = totalTime > tDur - _tinyNum && !isNegative ? tDur : totalTime < _tinyNum ? 0 : totalTime,\n time,\n pt,\n iteration,\n cycleDuration,\n prevIteration,\n isYoyo,\n ratio,\n timeline,\n yoyoEase;\n\n if (!dur) {\n _renderZeroDurationTween(this, totalTime, suppressEvents, force);\n } else if (tTime !== this._tTime || !totalTime || force || !this._initted && this._tTime || this._startAt && this._zTime < 0 !== isNegative) {\n //this senses if we're crossing over the start time, in which case we must record _zTime and force the render, but we do it in this lengthy conditional way for performance reasons (usually we can skip the calculations): this._initted && (this._zTime < 0) !== (totalTime < 0)\n time = tTime;\n timeline = this.timeline;\n\n if (this._repeat) {\n //adjust the time for repeats and yoyos\n cycleDuration = dur + this._rDelay;\n\n if (this._repeat < -1 && isNegative) {\n return this.totalTime(cycleDuration * 100 + totalTime, suppressEvents, force);\n }\n\n time = _roundPrecise(tTime % cycleDuration); //round to avoid floating point errors. (4 % 0.8 should be 0 but some browsers report it as 0.79999999!)\n\n if (tTime === tDur) {\n // the tDur === tTime is for edge cases where there's a lengthy decimal on the duration and it may reach the very end but the time is rendered as not-quite-there (remember, tDur is rounded to 4 decimals whereas dur isn't)\n iteration = this._repeat;\n time = dur;\n } else {\n iteration = ~~(tTime / cycleDuration);\n\n if (iteration && iteration === tTime / cycleDuration) {\n time = dur;\n iteration--;\n }\n\n time > dur && (time = dur);\n }\n\n isYoyo = this._yoyo && iteration & 1;\n\n if (isYoyo) {\n yoyoEase = this._yEase;\n time = dur - time;\n }\n\n prevIteration = _animationCycle(this._tTime, cycleDuration);\n\n if (time === prevTime && !force && this._initted) {\n //could be during the repeatDelay part. No need to render and fire callbacks.\n this._tTime = tTime;\n return this;\n }\n\n if (iteration !== prevIteration) {\n timeline && this._yEase && _propagateYoyoEase(timeline, isYoyo); //repeatRefresh functionality\n\n if (this.vars.repeatRefresh && !isYoyo && !this._lock) {\n this._lock = force = 1; //force, otherwise if lazy is true, the _attemptInitTween() will return and we'll jump out and get caught bouncing on each tick.\n\n this.render(_roundPrecise(cycleDuration * iteration), true).invalidate()._lock = 0;\n }\n }\n }\n\n if (!this._initted) {\n if (_attemptInitTween(this, isNegative ? totalTime : time, force, suppressEvents)) {\n this._tTime = 0; // in constructor if immediateRender is true, we set _tTime to -_tinyNum to have the playhead cross the starting point but we can't leave _tTime as a negative number.\n\n return this;\n }\n\n if (prevTime !== this._time) {\n // rare edge case - during initialization, an onUpdate in the _startAt (.fromTo()) might force this tween to render at a different spot in which case we should ditch this render() call so that it doesn't revert the values.\n return this;\n }\n\n if (dur !== this._dur) {\n // while initting, a plugin like InertiaPlugin might alter the duration, so rerun from the start to ensure everything renders as it should.\n return this.render(totalTime, suppressEvents, force);\n }\n }\n\n this._tTime = tTime;\n this._time = time;\n\n if (!this._act && this._ts) {\n this._act = 1; //as long as it's not paused, force it to be active so that if the user renders independent of the parent timeline, it'll be forced to re-render on the next tick.\n\n this._lazy = 0;\n }\n\n this.ratio = ratio = (yoyoEase || this._ease)(time / dur);\n\n if (this._from) {\n this.ratio = ratio = 1 - ratio;\n }\n\n if (time && !prevTime && !suppressEvents) {\n _callback(this, \"onStart\");\n\n if (this._tTime !== tTime) {\n // in case the onStart triggered a render at a different spot, eject. Like if someone did animation.pause(0.5) or something inside the onStart.\n return this;\n }\n }\n\n pt = this._pt;\n\n while (pt) {\n pt.r(ratio, pt.d);\n pt = pt._next;\n }\n\n timeline && timeline.render(totalTime < 0 ? totalTime : !time && isYoyo ? -_tinyNum : timeline._dur * timeline._ease(time / this._dur), suppressEvents, force) || this._startAt && (this._zTime = totalTime);\n\n if (this._onUpdate && !suppressEvents) {\n isNegative && _rewindStartAt(this, totalTime, suppressEvents, force); //note: for performance reasons, we tuck this conditional logic inside less traveled areas (most tweens don't have an onUpdate). We'd just have it at the end before the onComplete, but the values should be updated before any onUpdate is called, so we ALSO put it here and then if it's not called, we do so later near the onComplete.\n\n _callback(this, \"onUpdate\");\n }\n\n this._repeat && iteration !== prevIteration && this.vars.onRepeat && !suppressEvents && this.parent && _callback(this, \"onRepeat\");\n\n if ((tTime === this._tDur || !tTime) && this._tTime === tTime) {\n isNegative && !this._onUpdate && _rewindStartAt(this, totalTime, true, true);\n (totalTime || !dur) && (tTime === this._tDur && this._ts > 0 || !tTime && this._ts < 0) && _removeFromParent(this, 1); // don't remove if we're rendering at exactly a time of 0, as there could be autoRevert values that should get set on the next tick (if the playhead goes backward beyond the startTime, negative totalTime). Don't remove if the timeline is reversed and the playhead isn't at 0, otherwise tl.progress(1).reverse() won't work. Only remove if the playhead is at the end and timeScale is positive, or if the playhead is at 0 and the timeScale is negative.\n\n if (!suppressEvents && !(isNegative && !prevTime) && (tTime || prevTime)) {\n // if prevTime and tTime are zero, we shouldn't fire the onReverseComplete. This could happen if you gsap.to(... {paused:true}).play();\n _callback(this, tTime === tDur ? \"onComplete\" : \"onReverseComplete\", true);\n\n this._prom && !(tTime < tDur && this.timeScale() > 0) && this._prom();\n }\n }\n }\n\n return this;\n };\n\n _proto3.targets = function targets() {\n return this._targets;\n };\n\n _proto3.invalidate = function invalidate() {\n this._pt = this._op = this._startAt = this._onUpdate = this._lazy = this.ratio = 0;\n this._ptLookup = [];\n this.timeline && this.timeline.invalidate();\n return _Animation2.prototype.invalidate.call(this);\n };\n\n _proto3.resetTo = function resetTo(property, value, start, startIsRelative) {\n _tickerActive || _ticker.wake();\n this._ts || this.play();\n var time = Math.min(this._dur, (this._dp._time - this._start) * this._ts),\n ratio;\n this._initted || _initTween(this, time);\n ratio = this._ease(time / this._dur); // don't just get tween.ratio because it may not have rendered yet.\n // possible future addition to allow an object with multiple values to update, like tween.resetTo({x: 100, y: 200}); At this point, it doesn't seem worth the added kb given the fact that most users will likely opt for the convenient gsap.quickTo() way of interacting with this method.\n // if (_isObject(property)) { // performance optimization\n // \tfor (p in property) {\n // \t\tif (_updatePropTweens(this, p, property[p], value ? value[p] : null, start, ratio, time)) {\n // \t\t\treturn this.resetTo(property, value, start, startIsRelative); // if a PropTween wasn't found for the property, it'll get forced with a re-initialization so we need to jump out and start over again.\n // \t\t}\n // \t}\n // } else {\n\n if (_updatePropTweens(this, property, value, start, startIsRelative, ratio, time)) {\n return this.resetTo(property, value, start, startIsRelative); // if a PropTween wasn't found for the property, it'll get forced with a re-initialization so we need to jump out and start over again.\n } //}\n\n\n _alignPlayhead(this, 0);\n\n this.parent || _addLinkedListItem(this._dp, this, \"_first\", \"_last\", this._dp._sort ? \"_start\" : 0);\n return this.render(0);\n };\n\n _proto3.kill = function kill(targets, vars) {\n if (vars === void 0) {\n vars = \"all\";\n }\n\n if (!targets && (!vars || vars === \"all\")) {\n this._lazy = this._pt = 0;\n return this.parent ? _interrupt(this) : this;\n }\n\n if (this.timeline) {\n var tDur = this.timeline.totalDuration();\n this.timeline.killTweensOf(targets, vars, _overwritingTween && _overwritingTween.vars.overwrite !== true)._first || _interrupt(this); // if nothing is left tweening, interrupt.\n\n this.parent && tDur !== this.timeline.totalDuration() && _setDuration(this, this._dur * this.timeline._tDur / tDur, 0, 1); // if a nested tween is killed that changes the duration, it should affect this tween's duration. We must use the ratio, though, because sometimes the internal timeline is stretched like for keyframes where they don't all add up to whatever the parent tween's duration was set to.\n\n return this;\n }\n\n var parsedTargets = this._targets,\n killingTargets = targets ? toArray(targets) : parsedTargets,\n propTweenLookup = this._ptLookup,\n firstPT = this._pt,\n overwrittenProps,\n curLookup,\n curOverwriteProps,\n props,\n p,\n pt,\n i;\n\n if ((!vars || vars === \"all\") && _arraysMatch(parsedTargets, killingTargets)) {\n vars === \"all\" && (this._pt = 0);\n return _interrupt(this);\n }\n\n overwrittenProps = this._op = this._op || [];\n\n if (vars !== \"all\") {\n //so people can pass in a comma-delimited list of property names\n if (_isString(vars)) {\n p = {};\n\n _forEachName(vars, function (name) {\n return p[name] = 1;\n });\n\n vars = p;\n }\n\n vars = _addAliasesToVars(parsedTargets, vars);\n }\n\n i = parsedTargets.length;\n\n while (i--) {\n if (~killingTargets.indexOf(parsedTargets[i])) {\n curLookup = propTweenLookup[i];\n\n if (vars === \"all\") {\n overwrittenProps[i] = vars;\n props = curLookup;\n curOverwriteProps = {};\n } else {\n curOverwriteProps = overwrittenProps[i] = overwrittenProps[i] || {};\n props = vars;\n }\n\n for (p in props) {\n pt = curLookup && curLookup[p];\n\n if (pt) {\n if (!(\"kill\" in pt.d) || pt.d.kill(p) === true) {\n _removeLinkedListItem(this, pt, \"_pt\");\n }\n\n delete curLookup[p];\n }\n\n if (curOverwriteProps !== \"all\") {\n curOverwriteProps[p] = 1;\n }\n }\n }\n }\n\n this._initted && !this._pt && firstPT && _interrupt(this); //if all tweening properties are killed, kill the tween. Without this line, if there's a tween with multiple targets and then you killTweensOf() each target individually, the tween would technically still remain active and fire its onComplete even though there aren't any more properties tweening.\n\n return this;\n };\n\n Tween.to = function to(targets, vars) {\n return new Tween(targets, vars, arguments[2]);\n };\n\n Tween.from = function from(targets, vars) {\n return _createTweenType(1, arguments);\n };\n\n Tween.delayedCall = function delayedCall(delay, callback, params, scope) {\n return new Tween(callback, 0, {\n immediateRender: false,\n lazy: false,\n overwrite: false,\n delay: delay,\n onComplete: callback,\n onReverseComplete: callback,\n onCompleteParams: params,\n onReverseCompleteParams: params,\n callbackScope: scope\n }); // we must use onReverseComplete too for things like timeline.add(() => {...}) which should be triggered in BOTH directions (forward and reverse)\n };\n\n Tween.fromTo = function fromTo(targets, fromVars, toVars) {\n return _createTweenType(2, arguments);\n };\n\n Tween.set = function set(targets, vars) {\n vars.duration = 0;\n vars.repeatDelay || (vars.repeat = 0);\n return new Tween(targets, vars);\n };\n\n Tween.killTweensOf = function killTweensOf(targets, props, onlyActive) {\n return _globalTimeline.killTweensOf(targets, props, onlyActive);\n };\n\n return Tween;\n}(Animation);\n\n_setDefaults(Tween.prototype, {\n _targets: [],\n _lazy: 0,\n _startAt: 0,\n _op: 0,\n _onInit: 0\n}); //add the pertinent timeline methods to Tween instances so that users can chain conveniently and create a timeline automatically. (removed due to concerns that it'd ultimately add to more confusion especially for beginners)\n// _forEachName(\"to,from,fromTo,set,call,add,addLabel,addPause\", name => {\n// \tTween.prototype[name] = function() {\n// \t\tlet tl = new Timeline();\n// \t\treturn _addToTimeline(tl, this)[name].apply(tl, toArray(arguments));\n// \t}\n// });\n//for backward compatibility. Leverage the timeline calls.\n\n\n_forEachName(\"staggerTo,staggerFrom,staggerFromTo\", function (name) {\n Tween[name] = function () {\n var tl = new Timeline(),\n params = _slice.call(arguments, 0);\n\n params.splice(name === \"staggerFromTo\" ? 5 : 4, 0, 0);\n return tl[name].apply(tl, params);\n };\n});\n/*\n * --------------------------------------------------------------------------------------\n * PROPTWEEN\n * --------------------------------------------------------------------------------------\n */\n\n\nvar _setterPlain = function _setterPlain(target, property, value) {\n return target[property] = value;\n},\n _setterFunc = function _setterFunc(target, property, value) {\n return target[property](value);\n},\n _setterFuncWithParam = function _setterFuncWithParam(target, property, value, data) {\n return target[property](data.fp, value);\n},\n _setterAttribute = function _setterAttribute(target, property, value) {\n return target.setAttribute(property, value);\n},\n _getSetter = function _getSetter(target, property) {\n return _isFunction(target[property]) ? _setterFunc : _isUndefined(target[property]) && target.setAttribute ? _setterAttribute : _setterPlain;\n},\n _renderPlain = function _renderPlain(ratio, data) {\n return data.set(data.t, data.p, Math.round((data.s + data.c * ratio) * 1000000) / 1000000, data);\n},\n _renderBoolean = function _renderBoolean(ratio, data) {\n return data.set(data.t, data.p, !!(data.s + data.c * ratio), data);\n},\n _renderComplexString = function _renderComplexString(ratio, data) {\n var pt = data._pt,\n s = \"\";\n\n if (!ratio && data.b) {\n //b = beginning string\n s = data.b;\n } else if (ratio === 1 && data.e) {\n //e = ending string\n s = data.e;\n } else {\n while (pt) {\n s = pt.p + (pt.m ? pt.m(pt.s + pt.c * ratio) : Math.round((pt.s + pt.c * ratio) * 10000) / 10000) + s; //we use the \"p\" property for the text inbetween (like a suffix). And in the context of a complex string, the modifier (m) is typically just Math.round(), like for RGB colors.\n\n pt = pt._next;\n }\n\n s += data.c; //we use the \"c\" of the PropTween to store the final chunk of non-numeric text.\n }\n\n data.set(data.t, data.p, s, data);\n},\n _renderPropTweens = function _renderPropTweens(ratio, data) {\n var pt = data._pt;\n\n while (pt) {\n pt.r(ratio, pt.d);\n pt = pt._next;\n }\n},\n _addPluginModifier = function _addPluginModifier(modifier, tween, target, property) {\n var pt = this._pt,\n next;\n\n while (pt) {\n next = pt._next;\n pt.p === property && pt.modifier(modifier, tween, target);\n pt = next;\n }\n},\n _killPropTweensOf = function _killPropTweensOf(property) {\n var pt = this._pt,\n hasNonDependentRemaining,\n next;\n\n while (pt) {\n next = pt._next;\n\n if (pt.p === property && !pt.op || pt.op === property) {\n _removeLinkedListItem(this, pt, \"_pt\");\n } else if (!pt.dep) {\n hasNonDependentRemaining = 1;\n }\n\n pt = next;\n }\n\n return !hasNonDependentRemaining;\n},\n _setterWithModifier = function _setterWithModifier(target, property, value, data) {\n data.mSet(target, property, data.m.call(data.tween, value, data.mt), data);\n},\n _sortPropTweensByPriority = function _sortPropTweensByPriority(parent) {\n var pt = parent._pt,\n next,\n pt2,\n first,\n last; //sorts the PropTween linked list in order of priority because some plugins need to do their work after ALL of the PropTweens were created (like RoundPropsPlugin and ModifiersPlugin)\n\n while (pt) {\n next = pt._next;\n pt2 = first;\n\n while (pt2 && pt2.pr > pt.pr) {\n pt2 = pt2._next;\n }\n\n if (pt._prev = pt2 ? pt2._prev : last) {\n pt._prev._next = pt;\n } else {\n first = pt;\n }\n\n if (pt._next = pt2) {\n pt2._prev = pt;\n } else {\n last = pt;\n }\n\n pt = next;\n }\n\n parent._pt = first;\n}; //PropTween key: t = target, p = prop, r = renderer, d = data, s = start, c = change, op = overwriteProperty (ONLY populated when it's different than p), pr = priority, _next/_prev for the linked list siblings, set = setter, m = modifier, mSet = modifierSetter (the original setter, before a modifier was added)\n\n\nexport var PropTween = /*#__PURE__*/function () {\n function PropTween(next, target, prop, start, change, renderer, data, setter, priority) {\n this.t = target;\n this.s = start;\n this.c = change;\n this.p = prop;\n this.r = renderer || _renderPlain;\n this.d = data || this;\n this.set = setter || _setterPlain;\n this.pr = priority || 0;\n this._next = next;\n\n if (next) {\n next._prev = this;\n }\n }\n\n var _proto4 = PropTween.prototype;\n\n _proto4.modifier = function modifier(func, tween, target) {\n this.mSet = this.mSet || this.set; //in case it was already set (a PropTween can only have one modifier)\n\n this.set = _setterWithModifier;\n this.m = func;\n this.mt = target; //modifier target\n\n this.tween = tween;\n };\n\n return PropTween;\n}(); //Initialization tasks\n\n_forEachName(_callbackNames + \"parent,duration,ease,delay,overwrite,runBackwards,startAt,yoyo,immediateRender,repeat,repeatDelay,data,paused,reversed,lazy,callbackScope,stringFilter,id,yoyoEase,stagger,inherit,repeatRefresh,keyframes,autoRevert,scrollTrigger\", function (name) {\n return _reservedProps[name] = 1;\n});\n\n_globals.TweenMax = _globals.TweenLite = Tween;\n_globals.TimelineLite = _globals.TimelineMax = Timeline;\n_globalTimeline = new Timeline({\n sortChildren: false,\n defaults: _defaults,\n autoRemoveChildren: true,\n id: \"root\",\n smoothChildTiming: true\n});\n_config.stringFilter = _colorStringFilter;\n\nvar _media = [],\n _listeners = {},\n _emptyArray = [],\n _lastMediaTime = 0,\n _dispatch = function _dispatch(type) {\n return (_listeners[type] || _emptyArray).map(function (f) {\n return f();\n });\n},\n _onMediaChange = function _onMediaChange() {\n var time = Date.now(),\n matches = [];\n\n if (time - _lastMediaTime > 2) {\n _dispatch(\"matchMediaInit\");\n\n _media.forEach(function (c) {\n var queries = c.queries,\n conditions = c.conditions,\n match,\n p,\n anyMatch,\n toggled;\n\n for (p in queries) {\n match = _win.matchMedia(queries[p]).matches; // Firefox doesn't update the \"matches\" property of the MediaQueryList object correctly - it only does so as it calls its change handler - so we must re-create a media query here to ensure it's accurate.\n\n match && (anyMatch = 1);\n\n if (match !== conditions[p]) {\n conditions[p] = match;\n toggled = 1;\n }\n }\n\n if (toggled) {\n c.revert();\n anyMatch && matches.push(c);\n }\n });\n\n _dispatch(\"matchMediaRevert\");\n\n matches.forEach(function (c) {\n return c.onMatch(c);\n });\n _lastMediaTime = time;\n\n _dispatch(\"matchMedia\");\n }\n};\n\nvar Context = /*#__PURE__*/function () {\n function Context(func, scope) {\n this.selector = scope && selector(scope);\n this.data = [];\n this._r = []; // returned/cleanup functions\n\n this.isReverted = false;\n func && this.add(func);\n }\n\n var _proto5 = Context.prototype;\n\n _proto5.add = function add(name, func, scope) {\n if (_isFunction(name)) {\n scope = func;\n func = name;\n name = _isFunction;\n }\n\n var self = this,\n f = function f() {\n var prev = _context,\n prevSelector = self.selector,\n result;\n prev && prev !== self && prev.data.push(self);\n scope && (self.selector = selector(scope));\n _context = self;\n result = func.apply(self, arguments);\n _isFunction(result) && self._r.push(result);\n _context = prev;\n self.selector = prevSelector;\n self.isReverted = false;\n return result;\n };\n\n self.last = f;\n return name === _isFunction ? f(self) : name ? self[name] = f : f;\n };\n\n _proto5.ignore = function ignore(func) {\n var prev = _context;\n _context = null;\n func(this);\n _context = prev;\n };\n\n _proto5.getTweens = function getTweens() {\n var a = [];\n this.data.forEach(function (e) {\n return e instanceof Context ? a.push.apply(a, e.getTweens()) : e instanceof Tween && a.push(e);\n });\n return a;\n };\n\n _proto5.clear = function clear() {\n this._r.length = this.data.length = 0;\n };\n\n _proto5.kill = function kill(revert, matchMedia) {\n var _this4 = this;\n\n if (revert) {\n // save as an object so that we can cache the globalTime for each tween to optimize performance during the sort\n this.getTweens().map(function (t) {\n return {\n g: t.globalTime(0),\n t: t\n };\n }).sort(function (a, b) {\n return b.g - a.g || -1;\n }).forEach(function (o) {\n return o.t.revert(revert);\n }); // note: all of the _startAt tweens should be reverted in reverse order that thy were created, and they'll all have the same globalTime (-1) so the \" || -1\" in the sort keeps the order properly.\n\n this.data.forEach(function (e) {\n return !(e instanceof Animation) && e.revert && e.revert(revert);\n });\n\n this._r.forEach(function (f) {\n return f(revert, _this4);\n });\n\n this.isReverted = true;\n } else {\n this.data.forEach(function (e) {\n return e.kill && e.kill();\n });\n }\n\n this.clear();\n\n if (matchMedia) {\n var i = _media.indexOf(this);\n\n !!~i && _media.splice(i, 1);\n }\n };\n\n _proto5.revert = function revert(config) {\n this.kill(config || {});\n };\n\n return Context;\n}();\n\nvar MatchMedia = /*#__PURE__*/function () {\n function MatchMedia(scope) {\n this.contexts = [];\n this.scope = scope;\n }\n\n var _proto6 = MatchMedia.prototype;\n\n _proto6.add = function add(conditions, func, scope) {\n _isObject(conditions) || (conditions = {\n matches: conditions\n });\n var context = new Context(0, scope || this.scope),\n cond = context.conditions = {},\n mq,\n p,\n active;\n this.contexts.push(context);\n func = context.add(\"onMatch\", func);\n context.queries = conditions;\n\n for (p in conditions) {\n if (p === \"all\") {\n active = 1;\n } else {\n mq = _win.matchMedia(conditions[p]);\n\n if (mq) {\n _media.indexOf(context) < 0 && _media.push(context);\n (cond[p] = mq.matches) && (active = 1);\n mq.addListener ? mq.addListener(_onMediaChange) : mq.addEventListener(\"change\", _onMediaChange);\n }\n }\n }\n\n active && func(context);\n return this;\n } // refresh() {\n // \tlet time = _lastMediaTime,\n // \t\tmedia = _media;\n // \t_lastMediaTime = -1;\n // \t_media = this.contexts;\n // \t_onMediaChange();\n // \t_lastMediaTime = time;\n // \t_media = media;\n // }\n ;\n\n _proto6.revert = function revert(config) {\n this.kill(config || {});\n };\n\n _proto6.kill = function kill(revert) {\n this.contexts.forEach(function (c) {\n return c.kill(revert, true);\n });\n };\n\n return MatchMedia;\n}();\n/*\n * --------------------------------------------------------------------------------------\n * GSAP\n * --------------------------------------------------------------------------------------\n */\n\n\nvar _gsap = {\n registerPlugin: function registerPlugin() {\n for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {\n args[_key2] = arguments[_key2];\n }\n\n args.forEach(function (config) {\n return _createPlugin(config);\n });\n },\n timeline: function timeline(vars) {\n return new Timeline(vars);\n },\n getTweensOf: function getTweensOf(targets, onlyActive) {\n return _globalTimeline.getTweensOf(targets, onlyActive);\n },\n getProperty: function getProperty(target, property, unit, uncache) {\n _isString(target) && (target = toArray(target)[0]); //in case selector text or an array is passed in\n\n var getter = _getCache(target || {}).get,\n format = unit ? _passThrough : _numericIfPossible;\n\n unit === \"native\" && (unit = \"\");\n return !target ? target : !property ? function (property, unit, uncache) {\n return format((_plugins[property] && _plugins[property].get || getter)(target, property, unit, uncache));\n } : format((_plugins[property] && _plugins[property].get || getter)(target, property, unit, uncache));\n },\n quickSetter: function quickSetter(target, property, unit) {\n target = toArray(target);\n\n if (target.length > 1) {\n var setters = target.map(function (t) {\n return gsap.quickSetter(t, property, unit);\n }),\n l = setters.length;\n return function (value) {\n var i = l;\n\n while (i--) {\n setters[i](value);\n }\n };\n }\n\n target = target[0] || {};\n\n var Plugin = _plugins[property],\n cache = _getCache(target),\n p = cache.harness && (cache.harness.aliases || {})[property] || property,\n // in case it's an alias, like \"rotate\" for \"rotation\".\n setter = Plugin ? function (value) {\n var p = new Plugin();\n _quickTween._pt = 0;\n p.init(target, unit ? value + unit : value, _quickTween, 0, [target]);\n p.render(1, p);\n _quickTween._pt && _renderPropTweens(1, _quickTween);\n } : cache.set(target, p);\n\n return Plugin ? setter : function (value) {\n return setter(target, p, unit ? value + unit : value, cache, 1);\n };\n },\n quickTo: function quickTo(target, property, vars) {\n var _merge2;\n\n var tween = gsap.to(target, _merge((_merge2 = {}, _merge2[property] = \"+=0.1\", _merge2.paused = true, _merge2), vars || {})),\n func = function func(value, start, startIsRelative) {\n return tween.resetTo(property, value, start, startIsRelative);\n };\n\n func.tween = tween;\n return func;\n },\n isTweening: function isTweening(targets) {\n return _globalTimeline.getTweensOf(targets, true).length > 0;\n },\n defaults: function defaults(value) {\n value && value.ease && (value.ease = _parseEase(value.ease, _defaults.ease));\n return _mergeDeep(_defaults, value || {});\n },\n config: function config(value) {\n return _mergeDeep(_config, value || {});\n },\n registerEffect: function registerEffect(_ref3) {\n var name = _ref3.name,\n effect = _ref3.effect,\n plugins = _ref3.plugins,\n defaults = _ref3.defaults,\n extendTimeline = _ref3.extendTimeline;\n (plugins || \"\").split(\",\").forEach(function (pluginName) {\n return pluginName && !_plugins[pluginName] && !_globals[pluginName] && _warn(name + \" effect requires \" + pluginName + \" plugin.\");\n });\n\n _effects[name] = function (targets, vars, tl) {\n return effect(toArray(targets), _setDefaults(vars || {}, defaults), tl);\n };\n\n if (extendTimeline) {\n Timeline.prototype[name] = function (targets, vars, position) {\n return this.add(_effects[name](targets, _isObject(vars) ? vars : (position = vars) && {}, this), position);\n };\n }\n },\n registerEase: function registerEase(name, ease) {\n _easeMap[name] = _parseEase(ease);\n },\n parseEase: function parseEase(ease, defaultEase) {\n return arguments.length ? _parseEase(ease, defaultEase) : _easeMap;\n },\n getById: function getById(id) {\n return _globalTimeline.getById(id);\n },\n exportRoot: function exportRoot(vars, includeDelayedCalls) {\n if (vars === void 0) {\n vars = {};\n }\n\n var tl = new Timeline(vars),\n child,\n next;\n tl.smoothChildTiming = _isNotFalse(vars.smoothChildTiming);\n\n _globalTimeline.remove(tl);\n\n tl._dp = 0; //otherwise it'll get re-activated when adding children and be re-introduced into _globalTimeline's linked list (then added to itself).\n\n tl._time = tl._tTime = _globalTimeline._time;\n child = _globalTimeline._first;\n\n while (child) {\n next = child._next;\n\n if (includeDelayedCalls || !(!child._dur && child instanceof Tween && child.vars.onComplete === child._targets[0])) {\n _addToTimeline(tl, child, child._start - child._delay);\n }\n\n child = next;\n }\n\n _addToTimeline(_globalTimeline, tl, 0);\n\n return tl;\n },\n context: function context(func, scope) {\n return func ? new Context(func, scope) : _context;\n },\n matchMedia: function matchMedia(scope) {\n return new MatchMedia(scope);\n },\n matchMediaRefresh: function matchMediaRefresh() {\n return _media.forEach(function (c) {\n var cond = c.conditions,\n found,\n p;\n\n for (p in cond) {\n if (cond[p]) {\n cond[p] = false;\n found = 1;\n }\n }\n\n found && c.revert();\n }) || _onMediaChange();\n },\n addEventListener: function addEventListener(type, callback) {\n var a = _listeners[type] || (_listeners[type] = []);\n ~a.indexOf(callback) || a.push(callback);\n },\n removeEventListener: function removeEventListener(type, callback) {\n var a = _listeners[type],\n i = a && a.indexOf(callback);\n i >= 0 && a.splice(i, 1);\n },\n utils: {\n wrap: wrap,\n wrapYoyo: wrapYoyo,\n distribute: distribute,\n random: random,\n snap: snap,\n normalize: normalize,\n getUnit: getUnit,\n clamp: clamp,\n splitColor: splitColor,\n toArray: toArray,\n selector: selector,\n mapRange: mapRange,\n pipe: pipe,\n unitize: unitize,\n interpolate: interpolate,\n shuffle: shuffle\n },\n install: _install,\n effects: _effects,\n ticker: _ticker,\n updateRoot: Timeline.updateRoot,\n plugins: _plugins,\n globalTimeline: _globalTimeline,\n core: {\n PropTween: PropTween,\n globals: _addGlobal,\n Tween: Tween,\n Timeline: Timeline,\n Animation: Animation,\n getCache: _getCache,\n _removeLinkedListItem: _removeLinkedListItem,\n reverting: function reverting() {\n return _reverting;\n },\n context: function context(toAdd) {\n if (toAdd && _context) {\n _context.data.push(toAdd);\n\n toAdd._ctx = _context;\n }\n\n return _context;\n },\n suppressOverwrites: function suppressOverwrites(value) {\n return _suppressOverwrites = value;\n }\n }\n};\n\n_forEachName(\"to,from,fromTo,delayedCall,set,killTweensOf\", function (name) {\n return _gsap[name] = Tween[name];\n});\n\n_ticker.add(Timeline.updateRoot);\n\n_quickTween = _gsap.to({}, {\n duration: 0\n}); // ---- EXTRA PLUGINS --------------------------------------------------------\n\nvar _getPluginPropTween = function _getPluginPropTween(plugin, prop) {\n var pt = plugin._pt;\n\n while (pt && pt.p !== prop && pt.op !== prop && pt.fp !== prop) {\n pt = pt._next;\n }\n\n return pt;\n},\n _addModifiers = function _addModifiers(tween, modifiers) {\n var targets = tween._targets,\n p,\n i,\n pt;\n\n for (p in modifiers) {\n i = targets.length;\n\n while (i--) {\n pt = tween._ptLookup[i][p];\n\n if (pt && (pt = pt.d)) {\n if (pt._pt) {\n // is a plugin\n pt = _getPluginPropTween(pt, p);\n }\n\n pt && pt.modifier && pt.modifier(modifiers[p], tween, targets[i], p);\n }\n }\n }\n},\n _buildModifierPlugin = function _buildModifierPlugin(name, modifier) {\n return {\n name: name,\n rawVars: 1,\n //don't pre-process function-based values or \"random()\" strings.\n init: function init(target, vars, tween) {\n tween._onInit = function (tween) {\n var temp, p;\n\n if (_isString(vars)) {\n temp = {};\n\n _forEachName(vars, function (name) {\n return temp[name] = 1;\n }); //if the user passes in a comma-delimited list of property names to roundProps, like \"x,y\", we round to whole numbers.\n\n\n vars = temp;\n }\n\n if (modifier) {\n temp = {};\n\n for (p in vars) {\n temp[p] = modifier(vars[p]);\n }\n\n vars = temp;\n }\n\n _addModifiers(tween, vars);\n };\n }\n };\n}; //register core plugins\n\n\nexport var gsap = _gsap.registerPlugin({\n name: \"attr\",\n init: function init(target, vars, tween, index, targets) {\n var p, pt, v;\n this.tween = tween;\n\n for (p in vars) {\n v = target.getAttribute(p) || \"\";\n pt = this.add(target, \"setAttribute\", (v || 0) + \"\", vars[p], index, targets, 0, 0, p);\n pt.op = p;\n pt.b = v; // record the beginning value so we can revert()\n\n this._props.push(p);\n }\n },\n render: function render(ratio, data) {\n var pt = data._pt;\n\n while (pt) {\n _reverting ? pt.set(pt.t, pt.p, pt.b, pt) : pt.r(ratio, pt.d); // if reverting, go back to the original (pt.b)\n\n pt = pt._next;\n }\n }\n}, {\n name: \"endArray\",\n init: function init(target, value) {\n var i = value.length;\n\n while (i--) {\n this.add(target, i, target[i] || 0, value[i], 0, 0, 0, 0, 0, 1);\n }\n }\n}, _buildModifierPlugin(\"roundProps\", _roundModifier), _buildModifierPlugin(\"modifiers\"), _buildModifierPlugin(\"snap\", snap)) || _gsap; //to prevent the core plugins from being dropped via aggressive tree shaking, we must include them in the variable declaration in this way.\n\nTween.version = Timeline.version = gsap.version = \"3.11.1\";\n_coreReady = 1;\n_windowExists() && _wake();\nvar Power0 = _easeMap.Power0,\n Power1 = _easeMap.Power1,\n Power2 = _easeMap.Power2,\n Power3 = _easeMap.Power3,\n Power4 = _easeMap.Power4,\n Linear = _easeMap.Linear,\n Quad = _easeMap.Quad,\n Cubic = _easeMap.Cubic,\n Quart = _easeMap.Quart,\n Quint = _easeMap.Quint,\n Strong = _easeMap.Strong,\n Elastic = _easeMap.Elastic,\n Back = _easeMap.Back,\n SteppedEase = _easeMap.SteppedEase,\n Bounce = _easeMap.Bounce,\n Sine = _easeMap.Sine,\n Expo = _easeMap.Expo,\n Circ = _easeMap.Circ;\nexport { Power0, Power1, Power2, Power3, Power4, Linear, Quad, Cubic, Quart, Quint, Strong, Elastic, Back, SteppedEase, Bounce, Sine, Expo, Circ };\nexport { Tween as TweenMax, Tween as TweenLite, Timeline as TimelineMax, Timeline as TimelineLite, gsap as default, wrap, wrapYoyo, distribute, random, snap, normalize, getUnit, clamp, splitColor, toArray, selector, mapRange, pipe, unitize, interpolate, shuffle }; //export some internal methods/orojects for use in CSSPlugin so that we can externalize that file and allow custom builds that exclude it.\n\nexport { _getProperty, _numExp, _numWithUnitExp, _isString, _isUndefined, _renderComplexString, _relExp, _setDefaults, _removeLinkedListItem, _forEachName, _sortPropTweensByPriority, _colorStringFilter, _replaceRandom, _checkPlugin, _plugins, _ticker, _config, _roundModifier, _round, _missingPlugin, _getSetter, _getCache, _colorExp, _parseRelative };","/*!\n * CSSPlugin 3.11.1\n * https://greensock.com\n *\n * Copyright 2008-2022, GreenSock. All rights reserved.\n * Subject to the terms at https://greensock.com/standard-license or for\n * Club GreenSock members, the agreement issued with that membership.\n * @author: Jack Doyle, jack@greensock.com\n*/\n\n/* eslint-disable */\nimport { gsap, _getProperty, _numExp, _numWithUnitExp, getUnit, _isString, _isUndefined, _renderComplexString, _relExp, _forEachName, _sortPropTweensByPriority, _colorStringFilter, _checkPlugin, _replaceRandom, _plugins, GSCache, PropTween, _config, _ticker, _round, _missingPlugin, _getSetter, _getCache, _colorExp, _parseRelative, _setDefaults, _removeLinkedListItem //for the commented-out className feature.\n} from \"./gsap-core.js\";\n\nvar _win,\n _doc,\n _docElement,\n _pluginInitted,\n _tempDiv,\n _tempDivStyler,\n _recentSetterPlugin,\n _reverting,\n _windowExists = function _windowExists() {\n return typeof window !== \"undefined\";\n},\n _transformProps = {},\n _RAD2DEG = 180 / Math.PI,\n _DEG2RAD = Math.PI / 180,\n _atan2 = Math.atan2,\n _bigNum = 1e8,\n _capsExp = /([A-Z])/g,\n _horizontalExp = /(left|right|width|margin|padding|x)/i,\n _complexExp = /[\\s,\\(]\\S/,\n _propertyAliases = {\n autoAlpha: \"opacity,visibility\",\n scale: \"scaleX,scaleY\",\n alpha: \"opacity\"\n},\n _renderCSSProp = function _renderCSSProp(ratio, data) {\n return data.set(data.t, data.p, Math.round((data.s + data.c * ratio) * 10000) / 10000 + data.u, data);\n},\n _renderPropWithEnd = function _renderPropWithEnd(ratio, data) {\n return data.set(data.t, data.p, ratio === 1 ? data.e : Math.round((data.s + data.c * ratio) * 10000) / 10000 + data.u, data);\n},\n _renderCSSPropWithBeginning = function _renderCSSPropWithBeginning(ratio, data) {\n return data.set(data.t, data.p, ratio ? Math.round((data.s + data.c * ratio) * 10000) / 10000 + data.u : data.b, data);\n},\n //if units change, we need a way to render the original unit/value when the tween goes all the way back to the beginning (ratio:0)\n_renderRoundedCSSProp = function _renderRoundedCSSProp(ratio, data) {\n var value = data.s + data.c * ratio;\n data.set(data.t, data.p, ~~(value + (value < 0 ? -.5 : .5)) + data.u, data);\n},\n _renderNonTweeningValue = function _renderNonTweeningValue(ratio, data) {\n return data.set(data.t, data.p, ratio ? data.e : data.b, data);\n},\n _renderNonTweeningValueOnlyAtEnd = function _renderNonTweeningValueOnlyAtEnd(ratio, data) {\n return data.set(data.t, data.p, ratio !== 1 ? data.b : data.e, data);\n},\n _setterCSSStyle = function _setterCSSStyle(target, property, value) {\n return target.style[property] = value;\n},\n _setterCSSProp = function _setterCSSProp(target, property, value) {\n return target.style.setProperty(property, value);\n},\n _setterTransform = function _setterTransform(target, property, value) {\n return target._gsap[property] = value;\n},\n _setterScale = function _setterScale(target, property, value) {\n return target._gsap.scaleX = target._gsap.scaleY = value;\n},\n _setterScaleWithRender = function _setterScaleWithRender(target, property, value, data, ratio) {\n var cache = target._gsap;\n cache.scaleX = cache.scaleY = value;\n cache.renderTransform(ratio, cache);\n},\n _setterTransformWithRender = function _setterTransformWithRender(target, property, value, data, ratio) {\n var cache = target._gsap;\n cache[property] = value;\n cache.renderTransform(ratio, cache);\n},\n _transformProp = \"transform\",\n _transformOriginProp = _transformProp + \"Origin\",\n _saveStyle = function _saveStyle(property) {\n var _this = this;\n\n var target = this.target,\n style = target.style;\n\n if (property in _transformProps) {\n this.tfm = this.tfm || {};\n\n if (property !== \"transform\") {\n property = _propertyAliases[property] || property;\n ~property.indexOf(\",\") ? property.split(\",\").forEach(function (a) {\n return _this.tfm[a] = _get(target, a);\n }) : this.tfm[property] = target._gsap.x ? target._gsap[property] : _get(target, property); // note: scale would map to \"scaleX,scaleY\", thus we loop and apply them both.\n }\n\n if (this.props.indexOf(_transformProp) >= 0) {\n return;\n }\n\n if (target._gsap.svg) {\n this.svgo = target.getAttribute(\"data-svg-origin\");\n this.props.push(_transformOriginProp, \"\");\n }\n\n property = _transformProp;\n }\n\n style && this.props.push(property, style[property]);\n},\n _removeIndependentTransforms = function _removeIndependentTransforms(style) {\n if (style.translate) {\n style.removeProperty(\"translate\");\n style.removeProperty(\"scale\");\n style.removeProperty(\"rotate\");\n }\n},\n _revertStyle = function _revertStyle() {\n var props = this.props,\n target = this.target,\n style = target.style,\n cache = target._gsap,\n i,\n p;\n\n for (i = 0; i < props.length; i += 2) {\n props[i + 1] ? style[props[i]] = props[i + 1] : style.removeProperty(props[i].replace(_capsExp, \"-$1\").toLowerCase());\n }\n\n if (this.tfm) {\n for (p in this.tfm) {\n cache[p] = this.tfm[p];\n }\n\n if (cache.svg) {\n cache.renderTransform();\n target.setAttribute(\"data-svg-origin\", this.svgo || \"\");\n }\n\n i = _reverting();\n\n if (i && !i.isStart && !style[_transformProp]) {\n _removeIndependentTransforms(style);\n\n cache.uncache = 1; // if it's a startAt that's being reverted in the _initTween() of the core, we don't need to uncache transforms. This is purely a performance optimization.\n }\n }\n},\n _getStyleSaver = function _getStyleSaver(target, properties) {\n var saver = {\n target: target,\n props: [],\n revert: _revertStyle,\n save: _saveStyle\n };\n properties && properties.split(\",\").forEach(function (p) {\n return saver.save(p);\n });\n return saver;\n},\n _supports3D,\n _createElement = function _createElement(type, ns) {\n var e = _doc.createElementNS ? _doc.createElementNS((ns || \"http://www.w3.org/1999/xhtml\").replace(/^https/, \"http\"), type) : _doc.createElement(type); //some servers swap in https for http in the namespace which can break things, making \"style\" inaccessible.\n\n return e.style ? e : _doc.createElement(type); //some environments won't allow access to the element's style when created with a namespace in which case we default to the standard createElement() to work around the issue. Also note that when GSAP is embedded directly inside an SVG file, createElement() won't allow access to the style object in Firefox (see https://greensock.com/forums/topic/20215-problem-using-tweenmax-in-standalone-self-containing-svg-file-err-cannot-set-property-csstext-of-undefined/).\n},\n _getComputedProperty = function _getComputedProperty(target, property, skipPrefixFallback) {\n var cs = getComputedStyle(target);\n return cs[property] || cs.getPropertyValue(property.replace(_capsExp, \"-$1\").toLowerCase()) || cs.getPropertyValue(property) || !skipPrefixFallback && _getComputedProperty(target, _checkPropPrefix(property) || property, 1) || \"\"; //css variables may not need caps swapped out for dashes and lowercase.\n},\n _prefixes = \"O,Moz,ms,Ms,Webkit\".split(\",\"),\n _checkPropPrefix = function _checkPropPrefix(property, element, preferPrefix) {\n var e = element || _tempDiv,\n s = e.style,\n i = 5;\n\n if (property in s && !preferPrefix) {\n return property;\n }\n\n property = property.charAt(0).toUpperCase() + property.substr(1);\n\n while (i-- && !(_prefixes[i] + property in s)) {}\n\n return i < 0 ? null : (i === 3 ? \"ms\" : i >= 0 ? _prefixes[i] : \"\") + property;\n},\n _initCore = function _initCore() {\n if (_windowExists() && window.document) {\n _win = window;\n _doc = _win.document;\n _docElement = _doc.documentElement;\n _tempDiv = _createElement(\"div\") || {\n style: {}\n };\n _tempDivStyler = _createElement(\"div\");\n _transformProp = _checkPropPrefix(_transformProp);\n _transformOriginProp = _transformProp + \"Origin\";\n _tempDiv.style.cssText = \"border-width:0;line-height:0;position:absolute;padding:0\"; //make sure to override certain properties that may contaminate measurements, in case the user has overreaching style sheets.\n\n _supports3D = !!_checkPropPrefix(\"perspective\");\n _reverting = gsap.core.reverting;\n _pluginInitted = 1;\n }\n},\n _getBBoxHack = function _getBBoxHack(swapIfPossible) {\n //works around issues in some browsers (like Firefox) that don't correctly report getBBox() on SVG elements inside a element and/or . We try creating an SVG, adding it to the documentElement and toss the element in there so that it's definitely part of the rendering tree, then grab the bbox and if it works, we actually swap out the original getBBox() method for our own that does these extra steps whenever getBBox is needed. This helps ensure that performance is optimal (only do all these extra steps when absolutely necessary...most elements don't need it).\n var svg = _createElement(\"svg\", this.ownerSVGElement && this.ownerSVGElement.getAttribute(\"xmlns\") || \"http://www.w3.org/2000/svg\"),\n oldParent = this.parentNode,\n oldSibling = this.nextSibling,\n oldCSS = this.style.cssText,\n bbox;\n\n _docElement.appendChild(svg);\n\n svg.appendChild(this);\n this.style.display = \"block\";\n\n if (swapIfPossible) {\n try {\n bbox = this.getBBox();\n this._gsapBBox = this.getBBox; //store the original\n\n this.getBBox = _getBBoxHack;\n } catch (e) {}\n } else if (this._gsapBBox) {\n bbox = this._gsapBBox();\n }\n\n if (oldParent) {\n if (oldSibling) {\n oldParent.insertBefore(this, oldSibling);\n } else {\n oldParent.appendChild(this);\n }\n }\n\n _docElement.removeChild(svg);\n\n this.style.cssText = oldCSS;\n return bbox;\n},\n _getAttributeFallbacks = function _getAttributeFallbacks(target, attributesArray) {\n var i = attributesArray.length;\n\n while (i--) {\n if (target.hasAttribute(attributesArray[i])) {\n return target.getAttribute(attributesArray[i]);\n }\n }\n},\n _getBBox = function _getBBox(target) {\n var bounds;\n\n try {\n bounds = target.getBBox(); //Firefox throws errors if you try calling getBBox() on an SVG element that's not rendered (like in a or ). https://bugzilla.mozilla.org/show_bug.cgi?id=612118\n } catch (error) {\n bounds = _getBBoxHack.call(target, true);\n }\n\n bounds && (bounds.width || bounds.height) || target.getBBox === _getBBoxHack || (bounds = _getBBoxHack.call(target, true)); //some browsers (like Firefox) misreport the bounds if the element has zero width and height (it just assumes it's at x:0, y:0), thus we need to manually grab the position in that case.\n\n return bounds && !bounds.width && !bounds.x && !bounds.y ? {\n x: +_getAttributeFallbacks(target, [\"x\", \"cx\", \"x1\"]) || 0,\n y: +_getAttributeFallbacks(target, [\"y\", \"cy\", \"y1\"]) || 0,\n width: 0,\n height: 0\n } : bounds;\n},\n _isSVG = function _isSVG(e) {\n return !!(e.getCTM && (!e.parentNode || e.ownerSVGElement) && _getBBox(e));\n},\n //reports if the element is an SVG on which getBBox() actually works\n_removeProperty = function _removeProperty(target, property) {\n if (property) {\n var style = target.style;\n\n if (property in _transformProps && property !== _transformOriginProp) {\n property = _transformProp;\n }\n\n if (style.removeProperty) {\n if (property.substr(0, 2) === \"ms\" || property.substr(0, 6) === \"webkit\") {\n //Microsoft and some Webkit browsers don't conform to the standard of capitalizing the first prefix character, so we adjust so that when we prefix the caps with a dash, it's correct (otherwise it'd be \"ms-transform\" instead of \"-ms-transform\" for IE9, for example)\n property = \"-\" + property;\n }\n\n style.removeProperty(property.replace(_capsExp, \"-$1\").toLowerCase());\n } else {\n //note: old versions of IE use \"removeAttribute()\" instead of \"removeProperty()\"\n style.removeAttribute(property);\n }\n }\n},\n _addNonTweeningPT = function _addNonTweeningPT(plugin, target, property, beginning, end, onlySetAtEnd) {\n var pt = new PropTween(plugin._pt, target, property, 0, 1, onlySetAtEnd ? _renderNonTweeningValueOnlyAtEnd : _renderNonTweeningValue);\n plugin._pt = pt;\n pt.b = beginning;\n pt.e = end;\n\n plugin._props.push(property);\n\n return pt;\n},\n _nonConvertibleUnits = {\n deg: 1,\n rad: 1,\n turn: 1\n},\n _nonStandardLayouts = {\n grid: 1,\n flex: 1\n},\n //takes a single value like 20px and converts it to the unit specified, like \"%\", returning only the numeric amount.\n_convertToUnit = function _convertToUnit(target, property, value, unit) {\n var curValue = parseFloat(value) || 0,\n curUnit = (value + \"\").trim().substr((curValue + \"\").length) || \"px\",\n // some browsers leave extra whitespace at the beginning of CSS variables, hence the need to trim()\n style = _tempDiv.style,\n horizontal = _horizontalExp.test(property),\n isRootSVG = target.tagName.toLowerCase() === \"svg\",\n measureProperty = (isRootSVG ? \"client\" : \"offset\") + (horizontal ? \"Width\" : \"Height\"),\n amount = 100,\n toPixels = unit === \"px\",\n toPercent = unit === \"%\",\n px,\n parent,\n cache,\n isSVG;\n\n if (unit === curUnit || !curValue || _nonConvertibleUnits[unit] || _nonConvertibleUnits[curUnit]) {\n return curValue;\n }\n\n curUnit !== \"px\" && !toPixels && (curValue = _convertToUnit(target, property, value, \"px\"));\n isSVG = target.getCTM && _isSVG(target);\n\n if ((toPercent || curUnit === \"%\") && (_transformProps[property] || ~property.indexOf(\"adius\"))) {\n px = isSVG ? target.getBBox()[horizontal ? \"width\" : \"height\"] : target[measureProperty];\n return _round(toPercent ? curValue / px * amount : curValue / 100 * px);\n }\n\n style[horizontal ? \"width\" : \"height\"] = amount + (toPixels ? curUnit : unit);\n parent = ~property.indexOf(\"adius\") || unit === \"em\" && target.appendChild && !isRootSVG ? target : target.parentNode;\n\n if (isSVG) {\n parent = (target.ownerSVGElement || {}).parentNode;\n }\n\n if (!parent || parent === _doc || !parent.appendChild) {\n parent = _doc.body;\n }\n\n cache = parent._gsap;\n\n if (cache && toPercent && cache.width && horizontal && cache.time === _ticker.time && !cache.uncache) {\n return _round(curValue / cache.width * amount);\n } else {\n (toPercent || curUnit === \"%\") && !_nonStandardLayouts[_getComputedProperty(parent, \"display\")] && (style.position = _getComputedProperty(target, \"position\"));\n parent === target && (style.position = \"static\"); // like for borderRadius, if it's a % we must have it relative to the target itself but that may not have position: relative or position: absolute in which case it'd go up the chain until it finds its offsetParent (bad). position: static protects against that.\n\n parent.appendChild(_tempDiv);\n px = _tempDiv[measureProperty];\n parent.removeChild(_tempDiv);\n style.position = \"absolute\";\n\n if (horizontal && toPercent) {\n cache = _getCache(parent);\n cache.time = _ticker.time;\n cache.width = parent[measureProperty];\n }\n }\n\n return _round(toPixels ? px * curValue / amount : px && curValue ? amount / px * curValue : 0);\n},\n _get = function _get(target, property, unit, uncache) {\n var value;\n _pluginInitted || _initCore();\n\n if (property in _propertyAliases && property !== \"transform\") {\n property = _propertyAliases[property];\n\n if (~property.indexOf(\",\")) {\n property = property.split(\",\")[0];\n }\n }\n\n if (_transformProps[property] && property !== \"transform\") {\n value = _parseTransform(target, uncache);\n value = property !== \"transformOrigin\" ? value[property] : value.svg ? value.origin : _firstTwoOnly(_getComputedProperty(target, _transformOriginProp)) + \" \" + value.zOrigin + \"px\";\n } else {\n value = target.style[property];\n\n if (!value || value === \"auto\" || uncache || ~(value + \"\").indexOf(\"calc(\")) {\n value = _specialProps[property] && _specialProps[property](target, property, unit) || _getComputedProperty(target, property) || _getProperty(target, property) || (property === \"opacity\" ? 1 : 0); // note: some browsers, like Firefox, don't report borderRadius correctly! Instead, it only reports every corner like borderTopLeftRadius\n }\n }\n\n return unit && !~(value + \"\").trim().indexOf(\" \") ? _convertToUnit(target, property, value, unit) + unit : value;\n},\n _tweenComplexCSSString = function _tweenComplexCSSString(target, prop, start, end) {\n // note: we call _tweenComplexCSSString.call(pluginInstance...) to ensure that it's scoped properly. We may call it from within a plugin too, thus \"this\" would refer to the plugin.\n if (!start || start === \"none\") {\n // some browsers like Safari actually PREFER the prefixed property and mis-report the unprefixed value like clipPath (BUG). In other words, even though clipPath exists in the style (\"clipPath\" in target.style) and it's set in the CSS properly (along with -webkit-clip-path), Safari reports clipPath as \"none\" whereas WebkitClipPath reports accurately like \"ellipse(100% 0% at 50% 0%)\", so in this case we must SWITCH to using the prefixed property instead. See https://greensock.com/forums/topic/18310-clippath-doesnt-work-on-ios/\n var p = _checkPropPrefix(prop, target, 1),\n s = p && _getComputedProperty(target, p, 1);\n\n if (s && s !== start) {\n prop = p;\n start = s;\n } else if (prop === \"borderColor\") {\n start = _getComputedProperty(target, \"borderTopColor\"); // Firefox bug: always reports \"borderColor\" as \"\", so we must fall back to borderTopColor. See https://greensock.com/forums/topic/24583-how-to-return-colors-that-i-had-after-reverse/\n }\n }\n\n var pt = new PropTween(this._pt, target.style, prop, 0, 1, _renderComplexString),\n index = 0,\n matchIndex = 0,\n a,\n result,\n startValues,\n startNum,\n color,\n startValue,\n endValue,\n endNum,\n chunk,\n endUnit,\n startUnit,\n endValues;\n pt.b = start;\n pt.e = end;\n start += \"\"; // ensure values are strings\n\n end += \"\";\n\n if (end === \"auto\") {\n target.style[prop] = end;\n end = _getComputedProperty(target, prop) || end;\n target.style[prop] = start;\n }\n\n a = [start, end];\n\n _colorStringFilter(a); // pass an array with the starting and ending values and let the filter do whatever it needs to the values. If colors are found, it returns true and then we must match where the color shows up order-wise because for things like boxShadow, sometimes the browser provides the computed values with the color FIRST, but the user provides it with the color LAST, so flip them if necessary. Same for drop-shadow().\n\n\n start = a[0];\n end = a[1];\n startValues = start.match(_numWithUnitExp) || [];\n endValues = end.match(_numWithUnitExp) || [];\n\n if (endValues.length) {\n while (result = _numWithUnitExp.exec(end)) {\n endValue = result[0];\n chunk = end.substring(index, result.index);\n\n if (color) {\n color = (color + 1) % 5;\n } else if (chunk.substr(-5) === \"rgba(\" || chunk.substr(-5) === \"hsla(\") {\n color = 1;\n }\n\n if (endValue !== (startValue = startValues[matchIndex++] || \"\")) {\n startNum = parseFloat(startValue) || 0;\n startUnit = startValue.substr((startNum + \"\").length);\n endValue.charAt(1) === \"=\" && (endValue = _parseRelative(startNum, endValue) + startUnit);\n endNum = parseFloat(endValue);\n endUnit = endValue.substr((endNum + \"\").length);\n index = _numWithUnitExp.lastIndex - endUnit.length;\n\n if (!endUnit) {\n //if something like \"perspective:300\" is passed in and we must add a unit to the end\n endUnit = endUnit || _config.units[prop] || startUnit;\n\n if (index === end.length) {\n end += endUnit;\n pt.e += endUnit;\n }\n }\n\n if (startUnit !== endUnit) {\n startNum = _convertToUnit(target, prop, startValue, endUnit) || 0;\n } // these nested PropTweens are handled in a special way - we'll never actually call a render or setter method on them. We'll just loop through them in the parent complex string PropTween's render method.\n\n\n pt._pt = {\n _next: pt._pt,\n p: chunk || matchIndex === 1 ? chunk : \",\",\n //note: SVG spec allows omission of comma/space when a negative sign is wedged between two numbers, like 2.5-5.3 instead of 2.5,-5.3 but when tweening, the negative value may switch to positive, so we insert the comma just in case.\n s: startNum,\n c: endNum - startNum,\n m: color && color < 4 || prop === \"zIndex\" ? Math.round : 0\n };\n }\n }\n\n pt.c = index < end.length ? end.substring(index, end.length) : \"\"; //we use the \"c\" of the PropTween to store the final part of the string (after the last number)\n } else {\n pt.r = prop === \"display\" && end === \"none\" ? _renderNonTweeningValueOnlyAtEnd : _renderNonTweeningValue;\n }\n\n _relExp.test(end) && (pt.e = 0); //if the end string contains relative values or dynamic random(...) values, delete the end it so that on the final render we don't actually set it to the string with += or -= characters (forces it to use the calculated value).\n\n this._pt = pt; //start the linked list with this new PropTween. Remember, we call _tweenComplexCSSString.call(pluginInstance...) to ensure that it's scoped properly. We may call it from within another plugin too, thus \"this\" would refer to the plugin.\n\n return pt;\n},\n _keywordToPercent = {\n top: \"0%\",\n bottom: \"100%\",\n left: \"0%\",\n right: \"100%\",\n center: \"50%\"\n},\n _convertKeywordsToPercentages = function _convertKeywordsToPercentages(value) {\n var split = value.split(\" \"),\n x = split[0],\n y = split[1] || \"50%\";\n\n if (x === \"top\" || x === \"bottom\" || y === \"left\" || y === \"right\") {\n //the user provided them in the wrong order, so flip them\n value = x;\n x = y;\n y = value;\n }\n\n split[0] = _keywordToPercent[x] || x;\n split[1] = _keywordToPercent[y] || y;\n return split.join(\" \");\n},\n _renderClearProps = function _renderClearProps(ratio, data) {\n if (data.tween && data.tween._time === data.tween._dur) {\n var target = data.t,\n style = target.style,\n props = data.u,\n cache = target._gsap,\n prop,\n clearTransforms,\n i;\n\n if (props === \"all\" || props === true) {\n style.cssText = \"\";\n clearTransforms = 1;\n } else {\n props = props.split(\",\");\n i = props.length;\n\n while (--i > -1) {\n prop = props[i];\n\n if (_transformProps[prop]) {\n clearTransforms = 1;\n prop = prop === \"transformOrigin\" ? _transformOriginProp : _transformProp;\n }\n\n _removeProperty(target, prop);\n }\n }\n\n if (clearTransforms) {\n _removeProperty(target, _transformProp);\n\n if (cache) {\n cache.svg && target.removeAttribute(\"transform\");\n\n _parseTransform(target, 1); // force all the cached values back to \"normal\"/identity, otherwise if there's another tween that's already set to render transforms on this element, it could display the wrong values.\n\n\n cache.uncache = 1;\n\n _removeIndependentTransforms(style);\n }\n }\n }\n},\n // note: specialProps should return 1 if (and only if) they have a non-zero priority. It indicates we need to sort the linked list.\n_specialProps = {\n clearProps: function clearProps(plugin, target, property, endValue, tween) {\n if (tween.data !== \"isFromStart\") {\n var pt = plugin._pt = new PropTween(plugin._pt, target, property, 0, 0, _renderClearProps);\n pt.u = endValue;\n pt.pr = -10;\n pt.tween = tween;\n\n plugin._props.push(property);\n\n return 1;\n }\n }\n /* className feature (about 0.4kb gzipped).\n , className(plugin, target, property, endValue, tween) {\n \tlet _renderClassName = (ratio, data) => {\n \t\t\tdata.css.render(ratio, data.css);\n \t\t\tif (!ratio || ratio === 1) {\n \t\t\t\tlet inline = data.rmv,\n \t\t\t\t\ttarget = data.t,\n \t\t\t\t\tp;\n \t\t\t\ttarget.setAttribute(\"class\", ratio ? data.e : data.b);\n \t\t\t\tfor (p in inline) {\n \t\t\t\t\t_removeProperty(target, p);\n \t\t\t\t}\n \t\t\t}\n \t\t},\n \t\t_getAllStyles = (target) => {\n \t\t\tlet styles = {},\n \t\t\t\tcomputed = getComputedStyle(target),\n \t\t\t\tp;\n \t\t\tfor (p in computed) {\n \t\t\t\tif (isNaN(p) && p !== \"cssText\" && p !== \"length\") {\n \t\t\t\t\tstyles[p] = computed[p];\n \t\t\t\t}\n \t\t\t}\n \t\t\t_setDefaults(styles, _parseTransform(target, 1));\n \t\t\treturn styles;\n \t\t},\n \t\tstartClassList = target.getAttribute(\"class\"),\n \t\tstyle = target.style,\n \t\tcssText = style.cssText,\n \t\tcache = target._gsap,\n \t\tclassPT = cache.classPT,\n \t\tinlineToRemoveAtEnd = {},\n \t\tdata = {t:target, plugin:plugin, rmv:inlineToRemoveAtEnd, b:startClassList, e:(endValue.charAt(1) !== \"=\") ? endValue : startClassList.replace(new RegExp(\"(?:\\\\s|^)\" + endValue.substr(2) + \"(?![\\\\w-])\"), \"\") + ((endValue.charAt(0) === \"+\") ? \" \" + endValue.substr(2) : \"\")},\n \t\tchangingVars = {},\n \t\tstartVars = _getAllStyles(target),\n \t\ttransformRelated = /(transform|perspective)/i,\n \t\tendVars, p;\n \tif (classPT) {\n \t\tclassPT.r(1, classPT.d);\n \t\t_removeLinkedListItem(classPT.d.plugin, classPT, \"_pt\");\n \t}\n \ttarget.setAttribute(\"class\", data.e);\n \tendVars = _getAllStyles(target, true);\n \ttarget.setAttribute(\"class\", startClassList);\n \tfor (p in endVars) {\n \t\tif (endVars[p] !== startVars[p] && !transformRelated.test(p)) {\n \t\t\tchangingVars[p] = endVars[p];\n \t\t\tif (!style[p] && style[p] !== \"0\") {\n \t\t\t\tinlineToRemoveAtEnd[p] = 1;\n \t\t\t}\n \t\t}\n \t}\n \tcache.classPT = plugin._pt = new PropTween(plugin._pt, target, \"className\", 0, 0, _renderClassName, data, 0, -11);\n \tif (style.cssText !== cssText) { //only apply if things change. Otherwise, in cases like a background-image that's pulled dynamically, it could cause a refresh. See https://greensock.com/forums/topic/20368-possible-gsap-bug-switching-classnames-in-chrome/.\n \t\tstyle.cssText = cssText; //we recorded cssText before we swapped classes and ran _getAllStyles() because in cases when a className tween is overwritten, we remove all the related tweening properties from that class change (otherwise class-specific stuff can't override properties we've directly set on the target's style object due to specificity).\n \t}\n \t_parseTransform(target, true); //to clear the caching of transforms\n \tdata.css = new gsap.plugins.css();\n \tdata.css.init(target, changingVars, tween);\n \tplugin._props.push(...data.css._props);\n \treturn 1;\n }\n */\n\n},\n\n/*\n * --------------------------------------------------------------------------------------\n * TRANSFORMS\n * --------------------------------------------------------------------------------------\n */\n_identity2DMatrix = [1, 0, 0, 1, 0, 0],\n _rotationalProperties = {},\n _isNullTransform = function _isNullTransform(value) {\n return value === \"matrix(1, 0, 0, 1, 0, 0)\" || value === \"none\" || !value;\n},\n _getComputedTransformMatrixAsArray = function _getComputedTransformMatrixAsArray(target) {\n var matrixString = _getComputedProperty(target, _transformProp);\n\n return _isNullTransform(matrixString) ? _identity2DMatrix : matrixString.substr(7).match(_numExp).map(_round);\n},\n _getMatrix = function _getMatrix(target, force2D) {\n var cache = target._gsap || _getCache(target),\n style = target.style,\n matrix = _getComputedTransformMatrixAsArray(target),\n parent,\n nextSibling,\n temp,\n addedToDOM;\n\n if (cache.svg && target.getAttribute(\"transform\")) {\n temp = target.transform.baseVal.consolidate().matrix; //ensures that even complex values like \"translate(50,60) rotate(135,0,0)\" are parsed because it mashes it into a matrix.\n\n matrix = [temp.a, temp.b, temp.c, temp.d, temp.e, temp.f];\n return matrix.join(\",\") === \"1,0,0,1,0,0\" ? _identity2DMatrix : matrix;\n } else if (matrix === _identity2DMatrix && !target.offsetParent && target !== _docElement && !cache.svg) {\n //note: if offsetParent is null, that means the element isn't in the normal document flow, like if it has display:none or one of its ancestors has display:none). Firefox returns null for getComputedStyle() if the element is in an iframe that has display:none. https://bugzilla.mozilla.org/show_bug.cgi?id=548397\n //browsers don't report transforms accurately unless the element is in the DOM and has a display value that's not \"none\". Firefox and Microsoft browsers have a partial bug where they'll report transforms even if display:none BUT not any percentage-based values like translate(-50%, 8px) will be reported as if it's translate(0, 8px).\n temp = style.display;\n style.display = \"block\";\n parent = target.parentNode;\n\n if (!parent || !target.offsetParent) {\n // note: in 3.3.0 we switched target.offsetParent to _doc.body.contains(target) to avoid [sometimes unnecessary] MutationObserver calls but that wasn't adequate because there are edge cases where nested position: fixed elements need to get reparented to accurately sense transforms. See https://github.com/greensock/GSAP/issues/388 and https://github.com/greensock/GSAP/issues/375\n addedToDOM = 1; //flag\n\n nextSibling = target.nextElementSibling;\n\n _docElement.appendChild(target); //we must add it to the DOM in order to get values properly\n\n }\n\n matrix = _getComputedTransformMatrixAsArray(target);\n temp ? style.display = temp : _removeProperty(target, \"display\");\n\n if (addedToDOM) {\n nextSibling ? parent.insertBefore(target, nextSibling) : parent ? parent.appendChild(target) : _docElement.removeChild(target);\n }\n }\n\n return force2D && matrix.length > 6 ? [matrix[0], matrix[1], matrix[4], matrix[5], matrix[12], matrix[13]] : matrix;\n},\n _applySVGOrigin = function _applySVGOrigin(target, origin, originIsAbsolute, smooth, matrixArray, pluginToAddPropTweensTo) {\n var cache = target._gsap,\n matrix = matrixArray || _getMatrix(target, true),\n xOriginOld = cache.xOrigin || 0,\n yOriginOld = cache.yOrigin || 0,\n xOffsetOld = cache.xOffset || 0,\n yOffsetOld = cache.yOffset || 0,\n a = matrix[0],\n b = matrix[1],\n c = matrix[2],\n d = matrix[3],\n tx = matrix[4],\n ty = matrix[5],\n originSplit = origin.split(\" \"),\n xOrigin = parseFloat(originSplit[0]) || 0,\n yOrigin = parseFloat(originSplit[1]) || 0,\n bounds,\n determinant,\n x,\n y;\n\n if (!originIsAbsolute) {\n bounds = _getBBox(target);\n xOrigin = bounds.x + (~originSplit[0].indexOf(\"%\") ? xOrigin / 100 * bounds.width : xOrigin);\n yOrigin = bounds.y + (~(originSplit[1] || originSplit[0]).indexOf(\"%\") ? yOrigin / 100 * bounds.height : yOrigin);\n } else if (matrix !== _identity2DMatrix && (determinant = a * d - b * c)) {\n //if it's zero (like if scaleX and scaleY are zero), skip it to avoid errors with dividing by zero.\n x = xOrigin * (d / determinant) + yOrigin * (-c / determinant) + (c * ty - d * tx) / determinant;\n y = xOrigin * (-b / determinant) + yOrigin * (a / determinant) - (a * ty - b * tx) / determinant;\n xOrigin = x;\n yOrigin = y;\n }\n\n if (smooth || smooth !== false && cache.smooth) {\n tx = xOrigin - xOriginOld;\n ty = yOrigin - yOriginOld;\n cache.xOffset = xOffsetOld + (tx * a + ty * c) - tx;\n cache.yOffset = yOffsetOld + (tx * b + ty * d) - ty;\n } else {\n cache.xOffset = cache.yOffset = 0;\n }\n\n cache.xOrigin = xOrigin;\n cache.yOrigin = yOrigin;\n cache.smooth = !!smooth;\n cache.origin = origin;\n cache.originIsAbsolute = !!originIsAbsolute;\n target.style[_transformOriginProp] = \"0px 0px\"; //otherwise, if someone sets an origin via CSS, it will likely interfere with the SVG transform attribute ones (because remember, we're baking the origin into the matrix() value).\n\n if (pluginToAddPropTweensTo) {\n _addNonTweeningPT(pluginToAddPropTweensTo, cache, \"xOrigin\", xOriginOld, xOrigin);\n\n _addNonTweeningPT(pluginToAddPropTweensTo, cache, \"yOrigin\", yOriginOld, yOrigin);\n\n _addNonTweeningPT(pluginToAddPropTweensTo, cache, \"xOffset\", xOffsetOld, cache.xOffset);\n\n _addNonTweeningPT(pluginToAddPropTweensTo, cache, \"yOffset\", yOffsetOld, cache.yOffset);\n }\n\n target.setAttribute(\"data-svg-origin\", xOrigin + \" \" + yOrigin);\n},\n _parseTransform = function _parseTransform(target, uncache) {\n var cache = target._gsap || new GSCache(target);\n\n if (\"x\" in cache && !uncache && !cache.uncache) {\n return cache;\n }\n\n var style = target.style,\n invertedScaleX = cache.scaleX < 0,\n px = \"px\",\n deg = \"deg\",\n cs = getComputedStyle(target),\n origin = _getComputedProperty(target, _transformOriginProp) || \"0\",\n x,\n y,\n z,\n scaleX,\n scaleY,\n rotation,\n rotationX,\n rotationY,\n skewX,\n skewY,\n perspective,\n xOrigin,\n yOrigin,\n matrix,\n angle,\n cos,\n sin,\n a,\n b,\n c,\n d,\n a12,\n a22,\n t1,\n t2,\n t3,\n a13,\n a23,\n a33,\n a42,\n a43,\n a32;\n x = y = z = rotation = rotationX = rotationY = skewX = skewY = perspective = 0;\n scaleX = scaleY = 1;\n cache.svg = !!(target.getCTM && _isSVG(target));\n\n if (cs.translate) {\n // accommodate independent transforms by combining them into normal ones.\n if (cs.translate !== \"none\" || cs.scale !== \"none\" || cs.rotate !== \"none\") {\n style[_transformProp] = (cs.translate !== \"none\" ? \"translate3d(\" + (cs.translate + \" 0 0\").split(\" \").slice(0, 3).join(\", \") + \") \" : \"\") + (cs.rotate !== \"none\" ? \"rotate(\" + cs.rotate + \") \" : \"\") + (cs.scale !== \"none\" ? \"scale(\" + cs.scale.split(\" \").join(\",\") + \") \" : \"\") + cs[_transformProp];\n }\n\n style.scale = style.rotate = style.translate = \"none\";\n }\n\n matrix = _getMatrix(target, cache.svg);\n\n if (cache.svg) {\n if (cache.uncache) {\n // if cache.uncache is true (and maybe if origin is 0,0), we need to set element.style.transformOrigin = (cache.xOrigin - bbox.x) + \"px \" + (cache.yOrigin - bbox.y) + \"px\". Previously we let the data-svg-origin stay instead, but when introducing revert(), it complicated things.\n t2 = target.getBBox();\n origin = cache.xOrigin - t2.x + \"px \" + (cache.yOrigin - t2.y) + \"px\";\n t1 = \"\";\n } else {\n t1 = !uncache && target.getAttribute(\"data-svg-origin\"); // Remember, to work around browser inconsistencies we always force SVG elements' transformOrigin to 0,0 and offset the translation accordingly.\n }\n\n _applySVGOrigin(target, t1 || origin, !!t1 || cache.originIsAbsolute, cache.smooth !== false, matrix);\n }\n\n xOrigin = cache.xOrigin || 0;\n yOrigin = cache.yOrigin || 0;\n\n if (matrix !== _identity2DMatrix) {\n a = matrix[0]; //a11\n\n b = matrix[1]; //a21\n\n c = matrix[2]; //a31\n\n d = matrix[3]; //a41\n\n x = a12 = matrix[4];\n y = a22 = matrix[5]; //2D matrix\n\n if (matrix.length === 6) {\n scaleX = Math.sqrt(a * a + b * b);\n scaleY = Math.sqrt(d * d + c * c);\n rotation = a || b ? _atan2(b, a) * _RAD2DEG : 0; //note: if scaleX is 0, we cannot accurately measure rotation. Same for skewX with a scaleY of 0. Therefore, we default to the previously recorded value (or zero if that doesn't exist).\n\n skewX = c || d ? _atan2(c, d) * _RAD2DEG + rotation : 0;\n skewX && (scaleY *= Math.abs(Math.cos(skewX * _DEG2RAD)));\n\n if (cache.svg) {\n x -= xOrigin - (xOrigin * a + yOrigin * c);\n y -= yOrigin - (xOrigin * b + yOrigin * d);\n } //3D matrix\n\n } else {\n a32 = matrix[6];\n a42 = matrix[7];\n a13 = matrix[8];\n a23 = matrix[9];\n a33 = matrix[10];\n a43 = matrix[11];\n x = matrix[12];\n y = matrix[13];\n z = matrix[14];\n angle = _atan2(a32, a33);\n rotationX = angle * _RAD2DEG; //rotationX\n\n if (angle) {\n cos = Math.cos(-angle);\n sin = Math.sin(-angle);\n t1 = a12 * cos + a13 * sin;\n t2 = a22 * cos + a23 * sin;\n t3 = a32 * cos + a33 * sin;\n a13 = a12 * -sin + a13 * cos;\n a23 = a22 * -sin + a23 * cos;\n a33 = a32 * -sin + a33 * cos;\n a43 = a42 * -sin + a43 * cos;\n a12 = t1;\n a22 = t2;\n a32 = t3;\n } //rotationY\n\n\n angle = _atan2(-c, a33);\n rotationY = angle * _RAD2DEG;\n\n if (angle) {\n cos = Math.cos(-angle);\n sin = Math.sin(-angle);\n t1 = a * cos - a13 * sin;\n t2 = b * cos - a23 * sin;\n t3 = c * cos - a33 * sin;\n a43 = d * sin + a43 * cos;\n a = t1;\n b = t2;\n c = t3;\n } //rotationZ\n\n\n angle = _atan2(b, a);\n rotation = angle * _RAD2DEG;\n\n if (angle) {\n cos = Math.cos(angle);\n sin = Math.sin(angle);\n t1 = a * cos + b * sin;\n t2 = a12 * cos + a22 * sin;\n b = b * cos - a * sin;\n a22 = a22 * cos - a12 * sin;\n a = t1;\n a12 = t2;\n }\n\n if (rotationX && Math.abs(rotationX) + Math.abs(rotation) > 359.9) {\n //when rotationY is set, it will often be parsed as 180 degrees different than it should be, and rotationX and rotation both being 180 (it looks the same), so we adjust for that here.\n rotationX = rotation = 0;\n rotationY = 180 - rotationY;\n }\n\n scaleX = _round(Math.sqrt(a * a + b * b + c * c));\n scaleY = _round(Math.sqrt(a22 * a22 + a32 * a32));\n angle = _atan2(a12, a22);\n skewX = Math.abs(angle) > 0.0002 ? angle * _RAD2DEG : 0;\n perspective = a43 ? 1 / (a43 < 0 ? -a43 : a43) : 0;\n }\n\n if (cache.svg) {\n //sense if there are CSS transforms applied on an SVG element in which case we must overwrite them when rendering. The transform attribute is more reliable cross-browser, but we can't just remove the CSS ones because they may be applied in a CSS rule somewhere (not just inline).\n t1 = target.getAttribute(\"transform\");\n cache.forceCSS = target.setAttribute(\"transform\", \"\") || !_isNullTransform(_getComputedProperty(target, _transformProp));\n t1 && target.setAttribute(\"transform\", t1);\n }\n }\n\n if (Math.abs(skewX) > 90 && Math.abs(skewX) < 270) {\n if (invertedScaleX) {\n scaleX *= -1;\n skewX += rotation <= 0 ? 180 : -180;\n rotation += rotation <= 0 ? 180 : -180;\n } else {\n scaleY *= -1;\n skewX += skewX <= 0 ? 180 : -180;\n }\n }\n\n uncache = uncache || cache.uncache;\n cache.x = x - ((cache.xPercent = x && (!uncache && cache.xPercent || (Math.round(target.offsetWidth / 2) === Math.round(-x) ? -50 : 0))) ? target.offsetWidth * cache.xPercent / 100 : 0) + px;\n cache.y = y - ((cache.yPercent = y && (!uncache && cache.yPercent || (Math.round(target.offsetHeight / 2) === Math.round(-y) ? -50 : 0))) ? target.offsetHeight * cache.yPercent / 100 : 0) + px;\n cache.z = z + px;\n cache.scaleX = _round(scaleX);\n cache.scaleY = _round(scaleY);\n cache.rotation = _round(rotation) + deg;\n cache.rotationX = _round(rotationX) + deg;\n cache.rotationY = _round(rotationY) + deg;\n cache.skewX = skewX + deg;\n cache.skewY = skewY + deg;\n cache.transformPerspective = perspective + px;\n\n if (cache.zOrigin = parseFloat(origin.split(\" \")[2]) || 0) {\n style[_transformOriginProp] = _firstTwoOnly(origin);\n }\n\n cache.xOffset = cache.yOffset = 0;\n cache.force3D = _config.force3D;\n cache.renderTransform = cache.svg ? _renderSVGTransforms : _supports3D ? _renderCSSTransforms : _renderNon3DTransforms;\n cache.uncache = 0;\n return cache;\n},\n _firstTwoOnly = function _firstTwoOnly(value) {\n return (value = value.split(\" \"))[0] + \" \" + value[1];\n},\n //for handling transformOrigin values, stripping out the 3rd dimension\n_addPxTranslate = function _addPxTranslate(target, start, value) {\n var unit = getUnit(start);\n return _round(parseFloat(start) + parseFloat(_convertToUnit(target, \"x\", value + \"px\", unit))) + unit;\n},\n _renderNon3DTransforms = function _renderNon3DTransforms(ratio, cache) {\n cache.z = \"0px\";\n cache.rotationY = cache.rotationX = \"0deg\";\n cache.force3D = 0;\n\n _renderCSSTransforms(ratio, cache);\n},\n _zeroDeg = \"0deg\",\n _zeroPx = \"0px\",\n _endParenthesis = \") \",\n _renderCSSTransforms = function _renderCSSTransforms(ratio, cache) {\n var _ref = cache || this,\n xPercent = _ref.xPercent,\n yPercent = _ref.yPercent,\n x = _ref.x,\n y = _ref.y,\n z = _ref.z,\n rotation = _ref.rotation,\n rotationY = _ref.rotationY,\n rotationX = _ref.rotationX,\n skewX = _ref.skewX,\n skewY = _ref.skewY,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n transformPerspective = _ref.transformPerspective,\n force3D = _ref.force3D,\n target = _ref.target,\n zOrigin = _ref.zOrigin,\n transforms = \"\",\n use3D = force3D === \"auto\" && ratio && ratio !== 1 || force3D === true; // Safari has a bug that causes it not to render 3D transform-origin values properly, so we force the z origin to 0, record it in the cache, and then do the math here to offset the translate values accordingly (basically do the 3D transform-origin part manually)\n\n\n if (zOrigin && (rotationX !== _zeroDeg || rotationY !== _zeroDeg)) {\n var angle = parseFloat(rotationY) * _DEG2RAD,\n a13 = Math.sin(angle),\n a33 = Math.cos(angle),\n cos;\n\n angle = parseFloat(rotationX) * _DEG2RAD;\n cos = Math.cos(angle);\n x = _addPxTranslate(target, x, a13 * cos * -zOrigin);\n y = _addPxTranslate(target, y, -Math.sin(angle) * -zOrigin);\n z = _addPxTranslate(target, z, a33 * cos * -zOrigin + zOrigin);\n }\n\n if (transformPerspective !== _zeroPx) {\n transforms += \"perspective(\" + transformPerspective + _endParenthesis;\n }\n\n if (xPercent || yPercent) {\n transforms += \"translate(\" + xPercent + \"%, \" + yPercent + \"%) \";\n }\n\n if (use3D || x !== _zeroPx || y !== _zeroPx || z !== _zeroPx) {\n transforms += z !== _zeroPx || use3D ? \"translate3d(\" + x + \", \" + y + \", \" + z + \") \" : \"translate(\" + x + \", \" + y + _endParenthesis;\n }\n\n if (rotation !== _zeroDeg) {\n transforms += \"rotate(\" + rotation + _endParenthesis;\n }\n\n if (rotationY !== _zeroDeg) {\n transforms += \"rotateY(\" + rotationY + _endParenthesis;\n }\n\n if (rotationX !== _zeroDeg) {\n transforms += \"rotateX(\" + rotationX + _endParenthesis;\n }\n\n if (skewX !== _zeroDeg || skewY !== _zeroDeg) {\n transforms += \"skew(\" + skewX + \", \" + skewY + _endParenthesis;\n }\n\n if (scaleX !== 1 || scaleY !== 1) {\n transforms += \"scale(\" + scaleX + \", \" + scaleY + _endParenthesis;\n }\n\n target.style[_transformProp] = transforms || \"translate(0, 0)\";\n},\n _renderSVGTransforms = function _renderSVGTransforms(ratio, cache) {\n var _ref2 = cache || this,\n xPercent = _ref2.xPercent,\n yPercent = _ref2.yPercent,\n x = _ref2.x,\n y = _ref2.y,\n rotation = _ref2.rotation,\n skewX = _ref2.skewX,\n skewY = _ref2.skewY,\n scaleX = _ref2.scaleX,\n scaleY = _ref2.scaleY,\n target = _ref2.target,\n xOrigin = _ref2.xOrigin,\n yOrigin = _ref2.yOrigin,\n xOffset = _ref2.xOffset,\n yOffset = _ref2.yOffset,\n forceCSS = _ref2.forceCSS,\n tx = parseFloat(x),\n ty = parseFloat(y),\n a11,\n a21,\n a12,\n a22,\n temp;\n\n rotation = parseFloat(rotation);\n skewX = parseFloat(skewX);\n skewY = parseFloat(skewY);\n\n if (skewY) {\n //for performance reasons, we combine all skewing into the skewX and rotation values. Remember, a skewY of 10 degrees looks the same as a rotation of 10 degrees plus a skewX of 10 degrees.\n skewY = parseFloat(skewY);\n skewX += skewY;\n rotation += skewY;\n }\n\n if (rotation || skewX) {\n rotation *= _DEG2RAD;\n skewX *= _DEG2RAD;\n a11 = Math.cos(rotation) * scaleX;\n a21 = Math.sin(rotation) * scaleX;\n a12 = Math.sin(rotation - skewX) * -scaleY;\n a22 = Math.cos(rotation - skewX) * scaleY;\n\n if (skewX) {\n skewY *= _DEG2RAD;\n temp = Math.tan(skewX - skewY);\n temp = Math.sqrt(1 + temp * temp);\n a12 *= temp;\n a22 *= temp;\n\n if (skewY) {\n temp = Math.tan(skewY);\n temp = Math.sqrt(1 + temp * temp);\n a11 *= temp;\n a21 *= temp;\n }\n }\n\n a11 = _round(a11);\n a21 = _round(a21);\n a12 = _round(a12);\n a22 = _round(a22);\n } else {\n a11 = scaleX;\n a22 = scaleY;\n a21 = a12 = 0;\n }\n\n if (tx && !~(x + \"\").indexOf(\"px\") || ty && !~(y + \"\").indexOf(\"px\")) {\n tx = _convertToUnit(target, \"x\", x, \"px\");\n ty = _convertToUnit(target, \"y\", y, \"px\");\n }\n\n if (xOrigin || yOrigin || xOffset || yOffset) {\n tx = _round(tx + xOrigin - (xOrigin * a11 + yOrigin * a12) + xOffset);\n ty = _round(ty + yOrigin - (xOrigin * a21 + yOrigin * a22) + yOffset);\n }\n\n if (xPercent || yPercent) {\n //The SVG spec doesn't support percentage-based translation in the \"transform\" attribute, so we merge it into the translation to simulate it.\n temp = target.getBBox();\n tx = _round(tx + xPercent / 100 * temp.width);\n ty = _round(ty + yPercent / 100 * temp.height);\n }\n\n temp = \"matrix(\" + a11 + \",\" + a21 + \",\" + a12 + \",\" + a22 + \",\" + tx + \",\" + ty + \")\";\n target.setAttribute(\"transform\", temp);\n forceCSS && (target.style[_transformProp] = temp); //some browsers prioritize CSS transforms over the transform attribute. When we sense that the user has CSS transforms applied, we must overwrite them this way (otherwise some browser simply won't render the transform attribute changes!)\n},\n _addRotationalPropTween = function _addRotationalPropTween(plugin, target, property, startNum, endValue) {\n var cap = 360,\n isString = _isString(endValue),\n endNum = parseFloat(endValue) * (isString && ~endValue.indexOf(\"rad\") ? _RAD2DEG : 1),\n change = endNum - startNum,\n finalValue = startNum + change + \"deg\",\n direction,\n pt;\n\n if (isString) {\n direction = endValue.split(\"_\")[1];\n\n if (direction === \"short\") {\n change %= cap;\n\n if (change !== change % (cap / 2)) {\n change += change < 0 ? cap : -cap;\n }\n }\n\n if (direction === \"cw\" && change < 0) {\n change = (change + cap * _bigNum) % cap - ~~(change / cap) * cap;\n } else if (direction === \"ccw\" && change > 0) {\n change = (change - cap * _bigNum) % cap - ~~(change / cap) * cap;\n }\n }\n\n plugin._pt = pt = new PropTween(plugin._pt, target, property, startNum, change, _renderPropWithEnd);\n pt.e = finalValue;\n pt.u = \"deg\";\n\n plugin._props.push(property);\n\n return pt;\n},\n _assign = function _assign(target, source) {\n // Internet Explorer doesn't have Object.assign(), so we recreate it here.\n for (var p in source) {\n target[p] = source[p];\n }\n\n return target;\n},\n _addRawTransformPTs = function _addRawTransformPTs(plugin, transforms, target) {\n //for handling cases where someone passes in a whole transform string, like transform: \"scale(2, 3) rotate(20deg) translateY(30em)\"\n var startCache = _assign({}, target._gsap),\n exclude = \"perspective,force3D,transformOrigin,svgOrigin\",\n style = target.style,\n endCache,\n p,\n startValue,\n endValue,\n startNum,\n endNum,\n startUnit,\n endUnit;\n\n if (startCache.svg) {\n startValue = target.getAttribute(\"transform\");\n target.setAttribute(\"transform\", \"\");\n style[_transformProp] = transforms;\n endCache = _parseTransform(target, 1);\n\n _removeProperty(target, _transformProp);\n\n target.setAttribute(\"transform\", startValue);\n } else {\n startValue = getComputedStyle(target)[_transformProp];\n style[_transformProp] = transforms;\n endCache = _parseTransform(target, 1);\n style[_transformProp] = startValue;\n }\n\n for (p in _transformProps) {\n startValue = startCache[p];\n endValue = endCache[p];\n\n if (startValue !== endValue && exclude.indexOf(p) < 0) {\n //tweening to no perspective gives very unintuitive results - just keep the same perspective in that case.\n startUnit = getUnit(startValue);\n endUnit = getUnit(endValue);\n startNum = startUnit !== endUnit ? _convertToUnit(target, p, startValue, endUnit) : parseFloat(startValue);\n endNum = parseFloat(endValue);\n plugin._pt = new PropTween(plugin._pt, endCache, p, startNum, endNum - startNum, _renderCSSProp);\n plugin._pt.u = endUnit || 0;\n\n plugin._props.push(p);\n }\n }\n\n _assign(endCache, startCache);\n}; // handle splitting apart padding, margin, borderWidth, and borderRadius into their 4 components. Firefox, for example, won't report borderRadius correctly - it will only do borderTopLeftRadius and the other corners. We also want to handle paddingTop, marginLeft, borderRightWidth, etc.\n\n\n_forEachName(\"padding,margin,Width,Radius\", function (name, index) {\n var t = \"Top\",\n r = \"Right\",\n b = \"Bottom\",\n l = \"Left\",\n props = (index < 3 ? [t, r, b, l] : [t + l, t + r, b + r, b + l]).map(function (side) {\n return index < 2 ? name + side : \"border\" + side + name;\n });\n\n _specialProps[index > 1 ? \"border\" + name : name] = function (plugin, target, property, endValue, tween) {\n var a, vars;\n\n if (arguments.length < 4) {\n // getter, passed target, property, and unit (from _get())\n a = props.map(function (prop) {\n return _get(plugin, prop, property);\n });\n vars = a.join(\" \");\n return vars.split(a[0]).length === 5 ? a[0] : vars;\n }\n\n a = (endValue + \"\").split(\" \");\n vars = {};\n props.forEach(function (prop, i) {\n return vars[prop] = a[i] = a[i] || a[(i - 1) / 2 | 0];\n });\n plugin.init(target, vars, tween);\n };\n});\n\nexport var CSSPlugin = {\n name: \"css\",\n register: _initCore,\n targetTest: function targetTest(target) {\n return target.style && target.nodeType;\n },\n init: function init(target, vars, tween, index, targets) {\n var props = this._props,\n style = target.style,\n startAt = tween.vars.startAt,\n startValue,\n endValue,\n endNum,\n startNum,\n type,\n specialProp,\n p,\n startUnit,\n endUnit,\n relative,\n isTransformRelated,\n transformPropTween,\n cache,\n smooth,\n hasPriority,\n inlineProps;\n _pluginInitted || _initCore(); // we may call init() multiple times on the same plugin instance, like when adding special properties, so make sure we don't overwrite the revert data or inlineProps\n\n this.styles = this.styles || _getStyleSaver(target);\n inlineProps = this.styles.props;\n this.tween = tween;\n\n for (p in vars) {\n if (p === \"autoRound\") {\n continue;\n }\n\n endValue = vars[p];\n\n if (_plugins[p] && _checkPlugin(p, vars, tween, index, target, targets)) {\n // plugins\n continue;\n }\n\n type = typeof endValue;\n specialProp = _specialProps[p];\n\n if (type === \"function\") {\n endValue = endValue.call(tween, index, target, targets);\n type = typeof endValue;\n }\n\n if (type === \"string\" && ~endValue.indexOf(\"random(\")) {\n endValue = _replaceRandom(endValue);\n }\n\n if (specialProp) {\n specialProp(this, target, p, endValue, tween) && (hasPriority = 1);\n } else if (p.substr(0, 2) === \"--\") {\n //CSS variable\n startValue = (getComputedStyle(target).getPropertyValue(p) + \"\").trim();\n endValue += \"\";\n _colorExp.lastIndex = 0;\n\n if (!_colorExp.test(startValue)) {\n // colors don't have units\n startUnit = getUnit(startValue);\n endUnit = getUnit(endValue);\n }\n\n endUnit ? startUnit !== endUnit && (startValue = _convertToUnit(target, p, startValue, endUnit) + endUnit) : startUnit && (endValue += startUnit);\n this.add(style, \"setProperty\", startValue, endValue, index, targets, 0, 0, p);\n props.push(p);\n inlineProps.push(p, style[p]);\n } else if (type !== \"undefined\") {\n if (startAt && p in startAt) {\n // in case someone hard-codes a complex value as the start, like top: \"calc(2vh / 2)\". Without this, it'd use the computed value (always in px)\n startValue = typeof startAt[p] === \"function\" ? startAt[p].call(tween, index, target, targets) : startAt[p];\n _isString(startValue) && ~startValue.indexOf(\"random(\") && (startValue = _replaceRandom(startValue));\n getUnit(startValue + \"\") || (startValue += _config.units[p] || getUnit(_get(target, p)) || \"\"); // for cases when someone passes in a unitless value like {x: 100}; if we try setting translate(100, 0px) it won't work.\n\n (startValue + \"\").charAt(1) === \"=\" && (startValue = _get(target, p)); // can't work with relative values\n } else {\n startValue = _get(target, p);\n }\n\n startNum = parseFloat(startValue);\n relative = type === \"string\" && endValue.charAt(1) === \"=\" && endValue.substr(0, 2);\n relative && (endValue = endValue.substr(2));\n endNum = parseFloat(endValue);\n\n if (p in _propertyAliases) {\n if (p === \"autoAlpha\") {\n //special case where we control the visibility along with opacity. We still allow the opacity value to pass through and get tweened.\n if (startNum === 1 && _get(target, \"visibility\") === \"hidden\" && endNum) {\n //if visibility is initially set to \"hidden\", we should interpret that as intent to make opacity 0 (a convenience)\n startNum = 0;\n }\n\n inlineProps.push(\"visibility\", style.visibility);\n\n _addNonTweeningPT(this, style, \"visibility\", startNum ? \"inherit\" : \"hidden\", endNum ? \"inherit\" : \"hidden\", !endNum);\n }\n\n if (p !== \"scale\" && p !== \"transform\") {\n p = _propertyAliases[p];\n ~p.indexOf(\",\") && (p = p.split(\",\")[0]);\n }\n }\n\n isTransformRelated = p in _transformProps; //--- TRANSFORM-RELATED ---\n\n if (isTransformRelated) {\n this.styles.save(p);\n\n if (!transformPropTween) {\n cache = target._gsap;\n cache.renderTransform && !vars.parseTransform || _parseTransform(target, vars.parseTransform); // if, for example, gsap.set(... {transform:\"translateX(50vw)\"}), the _get() call doesn't parse the transform, thus cache.renderTransform won't be set yet so force the parsing of the transform here.\n\n smooth = vars.smoothOrigin !== false && cache.smooth;\n transformPropTween = this._pt = new PropTween(this._pt, style, _transformProp, 0, 1, cache.renderTransform, cache, 0, -1); //the first time through, create the rendering PropTween so that it runs LAST (in the linked list, we keep adding to the beginning)\n\n transformPropTween.dep = 1; //flag it as dependent so that if things get killed/overwritten and this is the only PropTween left, we can safely kill the whole tween.\n }\n\n if (p === \"scale\") {\n this._pt = new PropTween(this._pt, cache, \"scaleY\", cache.scaleY, (relative ? _parseRelative(cache.scaleY, relative + endNum) : endNum) - cache.scaleY || 0, _renderCSSProp);\n this._pt.u = 0;\n props.push(\"scaleY\", p);\n p += \"X\";\n } else if (p === \"transformOrigin\") {\n inlineProps.push(_transformOriginProp, style[_transformOriginProp]);\n endValue = _convertKeywordsToPercentages(endValue); //in case something like \"left top\" or \"bottom right\" is passed in. Convert to percentages.\n\n if (cache.svg) {\n _applySVGOrigin(target, endValue, 0, smooth, 0, this);\n } else {\n endUnit = parseFloat(endValue.split(\" \")[2]) || 0; //handle the zOrigin separately!\n\n endUnit !== cache.zOrigin && _addNonTweeningPT(this, cache, \"zOrigin\", cache.zOrigin, endUnit);\n\n _addNonTweeningPT(this, style, p, _firstTwoOnly(startValue), _firstTwoOnly(endValue));\n }\n\n continue;\n } else if (p === \"svgOrigin\") {\n _applySVGOrigin(target, endValue, 1, smooth, 0, this);\n\n continue;\n } else if (p in _rotationalProperties) {\n _addRotationalPropTween(this, cache, p, startNum, relative ? _parseRelative(startNum, relative + endValue) : endValue);\n\n continue;\n } else if (p === \"smoothOrigin\") {\n _addNonTweeningPT(this, cache, \"smooth\", cache.smooth, endValue);\n\n continue;\n } else if (p === \"force3D\") {\n cache[p] = endValue;\n continue;\n } else if (p === \"transform\") {\n _addRawTransformPTs(this, endValue, target);\n\n continue;\n }\n } else if (!(p in style)) {\n p = _checkPropPrefix(p) || p;\n }\n\n if (isTransformRelated || (endNum || endNum === 0) && (startNum || startNum === 0) && !_complexExp.test(endValue) && p in style) {\n startUnit = (startValue + \"\").substr((startNum + \"\").length);\n endNum || (endNum = 0); // protect against NaN\n\n endUnit = getUnit(endValue) || (p in _config.units ? _config.units[p] : startUnit);\n startUnit !== endUnit && (startNum = _convertToUnit(target, p, startValue, endUnit));\n this._pt = new PropTween(this._pt, isTransformRelated ? cache : style, p, startNum, (relative ? _parseRelative(startNum, relative + endNum) : endNum) - startNum, !isTransformRelated && (endUnit === \"px\" || p === \"zIndex\") && vars.autoRound !== false ? _renderRoundedCSSProp : _renderCSSProp);\n this._pt.u = endUnit || 0;\n\n if (startUnit !== endUnit && endUnit !== \"%\") {\n //when the tween goes all the way back to the beginning, we need to revert it to the OLD/ORIGINAL value (with those units). We record that as a \"b\" (beginning) property and point to a render method that handles that. (performance optimization)\n this._pt.b = startValue;\n this._pt.r = _renderCSSPropWithBeginning;\n }\n } else if (!(p in style)) {\n if (p in target) {\n //maybe it's not a style - it could be a property added directly to an element in which case we'll try to animate that.\n this.add(target, p, startValue || target[p], relative ? relative + endValue : endValue, index, targets);\n } else {\n _missingPlugin(p, endValue);\n\n continue;\n }\n } else {\n _tweenComplexCSSString.call(this, target, p, startValue, relative ? relative + endValue : endValue);\n }\n\n isTransformRelated || inlineProps.push(p, style[p]);\n props.push(p);\n }\n }\n\n hasPriority && _sortPropTweensByPriority(this);\n },\n render: function render(ratio, data) {\n if (data.tween._time || !_reverting()) {\n var pt = data._pt;\n\n while (pt) {\n pt.r(ratio, pt.d);\n pt = pt._next;\n }\n } else {\n data.styles.revert();\n }\n },\n get: _get,\n aliases: _propertyAliases,\n getSetter: function getSetter(target, property, plugin) {\n //returns a setter function that accepts target, property, value and applies it accordingly. Remember, properties like \"x\" aren't as simple as target.style.property = value because they've got to be applied to a proxy object and then merged into a transform string in a renderer.\n var p = _propertyAliases[property];\n p && p.indexOf(\",\") < 0 && (property = p);\n return property in _transformProps && property !== _transformOriginProp && (target._gsap.x || _get(target, \"x\")) ? plugin && _recentSetterPlugin === plugin ? property === \"scale\" ? _setterScale : _setterTransform : (_recentSetterPlugin = plugin || {}) && (property === \"scale\" ? _setterScaleWithRender : _setterTransformWithRender) : target.style && !_isUndefined(target.style[property]) ? _setterCSSStyle : ~property.indexOf(\"-\") ? _setterCSSProp : _getSetter(target, property);\n },\n core: {\n _removeProperty: _removeProperty,\n _getMatrix: _getMatrix\n }\n};\ngsap.utils.checkPrefix = _checkPropPrefix;\ngsap.core.getStyleSaver = _getStyleSaver;\n\n(function (positionAndScale, rotation, others, aliases) {\n var all = _forEachName(positionAndScale + \",\" + rotation + \",\" + others, function (name) {\n _transformProps[name] = 1;\n });\n\n _forEachName(rotation, function (name) {\n _config.units[name] = \"deg\";\n _rotationalProperties[name] = 1;\n });\n\n _propertyAliases[all[13]] = positionAndScale + \",\" + rotation;\n\n _forEachName(aliases, function (name) {\n var split = name.split(\":\");\n _propertyAliases[split[1]] = all[split[0]];\n });\n})(\"x,y,z,scale,scaleX,scaleY,xPercent,yPercent\", \"rotation,rotationX,rotationY,skewX,skewY\", \"transform,transformOrigin,svgOrigin,force3D,smoothOrigin,transformPerspective\", \"0:translateX,1:translateY,2:translateZ,8:rotate,8:rotationZ,8:rotateZ,9:rotateX,10:rotateY\");\n\n_forEachName(\"x,y,z,top,right,bottom,left,width,height,fontSize,padding,margin,perspective\", function (name) {\n _config.units[name] = \"px\";\n});\n\ngsap.registerPlugin(CSSPlugin);\nexport { CSSPlugin as default, _getBBox, _createElement, _checkPropPrefix as checkPrefix };","import { gsap, Power0, Power1, Power2, Power3, Power4, Linear, Quad, Cubic, Quart, Quint, Strong, Elastic, Back, SteppedEase, Bounce, Sine, Expo, Circ, TweenLite, TimelineLite, TimelineMax } from \"./gsap-core.js\";\nimport { CSSPlugin } from \"./CSSPlugin.js\";\nvar gsapWithCSS = gsap.registerPlugin(CSSPlugin) || gsap,\n // to protect from tree shaking\nTweenMaxWithCSS = gsapWithCSS.core.Tween;\nexport { gsapWithCSS as gsap, gsapWithCSS as default, CSSPlugin, TweenMaxWithCSS as TweenMax, TweenLite, TimelineMax, TimelineLite, Power0, Power1, Power2, Power3, Power4, Linear, Quad, Cubic, Quart, Quint, Strong, Elastic, Back, SteppedEase, Bounce, Sine, Expo, Circ };","type LimitedConsole = Pick<\n Console,\n \"error\" | \"info\" | \"log\" | \"debug\" | \"trace\" | \"warn\"\n> & {\n track(...data: any[]): void;\n};\n\nconst createLogger = (): LimitedConsole => {\n return {\n error: (args) => window.console.error(args),\n info: (args) => window.console.info(args),\n log: (args) => window.console.log(args),\n debug: (args) => window.console.debug(args),\n trace: (args) => window.console.trace(args),\n warn: (args) => window.console.warn(args),\n track: (...data: any[]) => {\n const TrackJS = (window as any).TrackJS;\n if (TrackJS && typeof TrackJS.track === \"function\") {\n TrackJS.track(data);\n } else {\n window.console.error(data);\n }\n },\n };\n};\n\nexport const logger = createLogger();\n","/**\n * Returns true if value is a string containing non-whitespace characters.\n */\nexport const isNonEmptyString = (value: unknown): boolean => {\n return typeof value === \"string\" && value.trim().length > 0;\n};\n","export interface TransformOptions {\n // Position properties\n w: number;\n h: number;\n x: number;\n y: number;\n z: number;\n\n // Transform properties\n angle: number;\n scaleX: number;\n scaleY: number;\n\n // Resize Options\n canScaleY: boolean;\n canScaleX: boolean;\n lockAspect: boolean;\n\n opacity: number;\n\n [key: string]: any;\n}\n\nexport const DefaultTransformOptions: TransformOptions = {\n w: 540,\n h: 360,\n x: 0,\n y: 0,\n z: 0,\n opacity: 100,\n angle: 0,\n scaleX: 1,\n scaleY: 1,\n canScaleY: true,\n canScaleX: true,\n lockAspect: false,\n};\n","/* eslint-disable no-extra-boolean-cast */\n/* eslint-disable @typescript-eslint/prefer-as-const */\n/* eslint-disable no-useless-escape */\n\n/**\n-------------------------------------------------------------------------------\nThe MIT License (MIT)\n\nCopyright (c) 2019 Damon Oehlman \n\nPermission is hereby granted, free of charge, to any person obtaining a copy\nof this software and associated documentation files (the \"Software\"), to deal\nin the Software without restriction, including without limitation the rights\nto use, copy, modify, merge, publish, distribute, sublicense, and/or sell\ncopies of the Software, and to permit persons to whom the Software is\nfurnished to do so, subject to the following conditions:\n\nThe above copyright notice and this permission notice shall be included in all\ncopies or substantial portions of the Software.\n\nTHE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\nIMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\nFITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\nAUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\nLIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\nOUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE\nSOFTWARE.\n-------------------------------------------------------------------------------\n\nModified from by Jesse Gavin\nhttps://github.com/DamonOehlman/detect-browser\n\n-------------------------------------------------------------------------------\n*/\n\nexport type DetectedInfoType = \"browser\";\n\ninterface DetectedInfo<\n T extends DetectedInfoType,\n N extends string,\n O,\n V = null\n> {\n readonly type: T;\n readonly name: N;\n readonly version: V;\n readonly os: O;\n}\n\nexport class BrowserInfo\n implements DetectedInfo<\"browser\", Browser, OperatingSystem | null, string>\n{\n public readonly type = \"browser\";\n constructor(\n public readonly name: Browser,\n public readonly version: string,\n public readonly os: OperatingSystem | null\n ) {}\n}\n\nexport type Browser =\n | \"aol\"\n | \"edge\"\n | \"edge-ios\"\n | \"yandexbrowser\"\n | \"kakaotalk\"\n | \"samsung\"\n | \"silk\"\n | \"miui\"\n | \"beaker\"\n | \"edge-chromium\"\n | \"chrome\"\n | \"chromium-webview\"\n | \"phantomjs\"\n | \"crios\"\n | \"firefox\"\n | \"fxios\"\n | \"opera-mini\"\n | \"opera\"\n | \"pie\"\n | \"netfront\"\n | \"ie\"\n | \"bb10\"\n | \"android\"\n | \"ios\"\n | \"safari\"\n | \"facebook\"\n | \"instagram\"\n | \"ios-webview\"\n | \"curl\"\n | \"searchbot\";\nexport type OperatingSystem =\n | \"iOS\"\n | \"Android OS\"\n | \"BlackBerry OS\"\n | \"Windows Mobile\"\n | \"Amazon OS\"\n | \"Windows 3.11\"\n | \"Windows 95\"\n | \"Windows 98\"\n | \"Windows 2000\"\n | \"Windows XP\"\n | \"Windows Server 2003\"\n | \"Windows Vista\"\n | \"Windows 7\"\n | \"Windows 8\"\n | \"Windows 8.1\"\n | \"Windows 10\"\n | \"Windows ME\"\n | \"Windows CE\"\n | \"Open BSD\"\n | \"Sun OS\"\n | \"Linux\"\n | \"Mac OS\"\n | \"QNX\"\n | \"BeOS\"\n | \"OS/2\"\n | \"Chrome OS\";\ntype UserAgentRule = [Browser, RegExp];\ntype UserAgentMatch = [Browser, RegExpExecArray] | false;\ntype OperatingSystemRule = [OperatingSystem, RegExp];\n\nconst REQUIRED_VERSION_PARTS = 3;\n\nconst userAgentRules: UserAgentRule[] = [\n [\"aol\", /AOLShield\\/([0-9\\._]+)/],\n [\"edge\", /Edge\\/([0-9\\._]+)/],\n [\"edge-ios\", /EdgiOS\\/([0-9\\._]+)/],\n [\"yandexbrowser\", /YaBrowser\\/([0-9\\._]+)/],\n [\"kakaotalk\", /KAKAOTALK\\s([0-9\\.]+)/],\n [\"samsung\", /SamsungBrowser\\/([0-9\\.]+)/],\n [\"silk\", /\\bSilk\\/([0-9._-]+)\\b/],\n [\"miui\", /MiuiBrowser\\/([0-9\\.]+)$/],\n [\"beaker\", /BeakerBrowser\\/([0-9\\.]+)/],\n [\"edge-chromium\", /EdgA?\\/([0-9\\.]+)/],\n [\n \"chromium-webview\",\n /(?!Chrom.*OPR)wv\\).*Chrom(?:e|ium)\\/([0-9\\.]+)(:?\\s|$)/,\n ],\n [\"chrome\", /(?!Chrom.*OPR)Chrom(?:e|ium)\\/([0-9\\.]+)(:?\\s|$)/],\n [\"phantomjs\", /PhantomJS\\/([0-9\\.]+)(:?\\s|$)/],\n [\"crios\", /CriOS\\/([0-9\\.]+)(:?\\s|$)/],\n [\"firefox\", /Firefox\\/([0-9\\.]+)(?:\\s|$)/],\n [\"fxios\", /FxiOS\\/([0-9\\.]+)/],\n [\"opera-mini\", /Opera Mini.*Version\\/([0-9\\.]+)/],\n [\"opera\", /Opera\\/([0-9\\.]+)(?:\\s|$)/],\n [\"opera\", /OPR\\/([0-9\\.]+)(:?\\s|$)/],\n [\"pie\", /^Microsoft Pocket Internet Explorer\\/(\\d+\\.\\d+)$/],\n [\n \"pie\",\n /^Mozilla\\/\\d\\.\\d+\\s\\(compatible;\\s(?:MSP?IE|MSInternet Explorer) (\\d+\\.\\d+);.*Windows CE.*\\)$/,\n ],\n [\"netfront\", /^Mozilla\\/\\d\\.\\d+.*NetFront\\/(\\d.\\d)/],\n [\"ie\", /Trident\\/7\\.0.*rv\\:([0-9\\.]+).*\\).*Gecko$/],\n [\"ie\", /MSIE\\s([0-9\\.]+);.*Trident\\/[4-7].0/],\n [\"ie\", /MSIE\\s(7\\.0)/],\n [\"bb10\", /BB10;\\sTouch.*Version\\/([0-9\\.]+)/],\n [\"android\", /Android\\s([0-9\\.]+)/],\n [\"ios\", /Version\\/([0-9\\._]+).*Mobile.*Safari.*/],\n [\"safari\", /Version\\/([0-9\\._]+).*Safari/],\n [\"facebook\", /FB[AS]V\\/([0-9\\.]+)/],\n [\"instagram\", /Instagram\\s([0-9\\.]+)/],\n [\"ios-webview\", /AppleWebKit\\/([0-9\\.]+).*Mobile/],\n [\"ios-webview\", /AppleWebKit\\/([0-9\\.]+).*Gecko\\)$/],\n [\"curl\", /^curl\\/([0-9\\.]+)$/],\n];\nconst operatingSystemRules: OperatingSystemRule[] = [\n [\"iOS\", /iP(hone|od|ad)/],\n [\"Android OS\", /Android/],\n [\"BlackBerry OS\", /BlackBerry|BB10/],\n [\"Windows Mobile\", /IEMobile/],\n [\"Amazon OS\", /Kindle/],\n [\"Windows 3.11\", /Win16/],\n [\"Windows 95\", /(Windows 95)|(Win95)|(Windows_95)/],\n [\"Windows 98\", /(Windows 98)|(Win98)/],\n [\"Windows 2000\", /(Windows NT 5.0)|(Windows 2000)/],\n [\"Windows XP\", /(Windows NT 5.1)|(Windows XP)/],\n [\"Windows Server 2003\", /(Windows NT 5.2)/],\n [\"Windows Vista\", /(Windows NT 6.0)/],\n [\"Windows 7\", /(Windows NT 6.1)/],\n [\"Windows 8\", /(Windows NT 6.2)/],\n [\"Windows 8.1\", /(Windows NT 6.3)/],\n [\"Windows 10\", /(Windows NT 10.0)/],\n [\"Windows ME\", /Windows ME/],\n [\"Windows CE\", /Windows CE|WinCE|Microsoft Pocket Internet Explorer/],\n [\"Open BSD\", /OpenBSD/],\n [\"Sun OS\", /SunOS/],\n [\"Chrome OS\", /CrOS/],\n [\"Linux\", /(Linux)|(X11)/],\n [\"Mac OS\", /(Mac_PowerPC)|(Macintosh)/],\n [\"QNX\", /QNX/],\n [\"BeOS\", /BeOS/],\n [\"OS/2\", /OS\\/2/],\n];\n\nfunction matchUserAgent(ua: string): UserAgentMatch {\n // opted for using reduce here rather than Array#first with a regex.test call\n // this is primarily because using the reduce we only perform the regex\n // execution once rather than once for the test and for the exec again below\n // probably something that needs to be benchmarked though\n return (\n ua !== \"\" &&\n userAgentRules.reduce(\n (matched: UserAgentMatch, [browser, regex]) => {\n if (matched) {\n return matched;\n }\n\n const uaMatch = regex.exec(ua);\n return !!uaMatch && [browser, uaMatch];\n },\n false\n )\n );\n}\n\nfunction parseUserAgent(ua: string): BrowserInfo | null {\n const matchedRule: UserAgentMatch = matchUserAgent(ua);\n\n if (!matchedRule) {\n return null;\n }\n\n const [name, match] = matchedRule;\n if (name === \"searchbot\") {\n return null;\n }\n // Do not use RegExp for split operation as some browser do not support it (See: http://blog.stevenlevithan.com/archives/cross-browser-split)\n let versionParts =\n match[1] && match[1].split(\".\").join(\"_\").split(\"_\").slice(0, 3);\n if (versionParts) {\n if (versionParts.length < REQUIRED_VERSION_PARTS) {\n versionParts = [\n ...versionParts,\n ...createVersionParts(REQUIRED_VERSION_PARTS - versionParts.length),\n ];\n }\n } else {\n versionParts = [];\n }\n\n const version = versionParts.join(\".\");\n const os = detectOS(ua);\n\n return new BrowserInfo(name, version, os);\n}\n\nfunction detectOS(ua: string): OperatingSystem | null {\n for (let ii = 0, count = operatingSystemRules.length; ii < count; ii++) {\n const [os, regex] = operatingSystemRules[ii];\n const match = regex.exec(ua);\n if (match) {\n return os;\n }\n }\n\n return null;\n}\n\nfunction createVersionParts(count: number): string[] {\n const output = [];\n for (let ii = 0; ii < count; ii++) {\n output.push(\"0\");\n }\n\n return output;\n}\n\nfunction requiresAbsoluteSvgUrlFix(browser: BrowserInfo | null) {\n /**\n * https://github.com/airbnb/lottie-web/issues/360#issuecomment-750913563\n * Browsers that require absolute urls for SVG fragment url() values\n * - Chrome 53 and older\n * - Firefox 51 and older\n * - Safari 13.1 and older\n */\n\n if (browser === null) {\n return false;\n }\n const version = parseFloat(browser.version);\n return [\n browser.name === \"chrome\" && version < 54,\n browser.name === \"safari\" && version < 13.1,\n browser.name === \"firefox\" && version < 51,\n ].some((x) => x);\n}\n\nconst browser = parseUserAgent(navigator.userAgent);\n\nconst svgUrlFixRequired = requiresAbsoluteSvgUrlFix(browser);\n\nexport { browser, svgUrlFixRequired };\n","import { Position, Point, Size, Rectangle } from \"@/types\";\nimport { BaseTextStyles, ShadowProps } from \"@/types/properties\";\nimport { logger } from \"@core/logger\";\nimport { isNonEmptyString } from \"@core/utils/isNonEmptyString\";\n\nimport { DateTime } from \"luxon\";\nimport { BackendError } from \"./api/backend\";\nimport {\n DefaultTransformOptions,\n TransformOptions,\n} from \"./components/widgets/TransformOptions\";\nimport { Widget } from \"./components/widgets/Widget\";\nimport { svgUrlFixRequired } from \"./util/browser\";\nimport { RepeaterOptions } from \"@/components/widgets/Repeater/RepeaterOptions\";\n\nexport const cleanColorValue = (color: string) => {\n let value = color;\n let formattedValue = color;\n const hexRegex = /^#?([a-f0-9]{6,8}|[a-f0-9]{3,4})$/i;\n const isHex = value.match(hexRegex);\n // Ensure a \"#\" is prepended to hex-formatted values\n if (isHex && !color.startsWith(\"#\")) {\n value = `#${color}`;\n formattedValue = value;\n }\n // Don't display the final two characters of an 8-digit hex value\n if (color.length === 9 && color.startsWith(\"#\") && color.endsWith(\"FF\")) {\n formattedValue = color.substring(0, 7);\n }\n return { value, formattedValue };\n};\n\nexport const loadImages = (urls: string[]) => {\n return urls.map(\n (url) =>\n new Promise((resolve, reject) => {\n const img = new Image();\n\n img.onload = function () {\n img.onload = null; // for garbage collection\n resolve(\"ok\");\n };\n\n img.onerror = function () {\n img.onerror = null; // for garbage collection\n reject();\n };\n\n img.src = url;\n })\n );\n};\n\n/**\n * These 3 methods are used to determine what query params to append to on-the-fly image requests made by Connect widgets.\n * The two main use cases are: Image widget, and Shape widget with a background image.\n */\n\ninterface ImageQueryParams {\n w: number;\n h: number;\n scaleX: number;\n scaleY: number;\n type: string;\n backgroundImageW?: number;\n backgroundImageH?: number;\n backgroundRepeat?: string;\n backgroundRepeatSize?: number;\n backgroundSize?: string;\n}\n\nconst getImageWidgetQueryParams = (payload: ImageQueryParams) => {\n const { w, h, scaleX, scaleY, backgroundSize } = payload;\n // console.log(\"get image widget query params\", window.devicePixelRatio);\n const width = Math.floor(w * scaleX * window.devicePixelRatio);\n const height = Math.floor(h * scaleY * window.devicePixelRatio);\n const fitMode = backgroundSize === \"contain\" ? \"max\" : \"crop\";\n return `width=${width}&height=${height}&mode=${fitMode}`;\n};\n\nconst getShapeWidgetImageQueryParams = (payload: ImageQueryParams) => {\n const {\n w,\n h,\n scaleX,\n scaleY,\n backgroundImageW,\n backgroundImageH,\n backgroundRepeat,\n backgroundRepeatSize,\n type,\n } = payload;\n\n const width = (type === \"Svg\" ? w : w * scaleX) * window.devicePixelRatio;\n const height = (type === \"Svg\" ? h : h * scaleY) * window.devicePixelRatio;\n const bgRatio = (backgroundImageW || 1) / (backgroundImageH || 1);\n\n const imageW =\n backgroundRepeat === \"no-repeat\" ? width : backgroundRepeatSize;\n\n let scl = 1;\n if (!isNaN(backgroundImageW) && !isNaN(backgroundImageH)) {\n scl = (backgroundImageH || 1) / (backgroundImageW || 1);\n }\n\n let imageH = height;\n if (backgroundRepeat !== \"no-repeat\" && backgroundRepeatSize) {\n imageH = backgroundRepeatSize * scl;\n }\n\n const imageRatio = (imageW || 1) / imageH;\n\n return bgRatio > imageRatio\n ? `height=${Math.floor(height)}`\n : `width=${Math.floor(width)}`;\n};\n\nexport const getImageQueryParams = (payload: ImageQueryParams) => {\n const { type } = payload;\n if (type === \"Image\") {\n return getImageWidgetQueryParams(payload);\n }\n return getShapeWidgetImageQueryParams(payload);\n};\n\nexport const computeRepeaterEditingContext = (\n widget: Widget,\n e: MouseEvent,\n scale: number\n) => {\n const { rows, columns, rowGap, columnGap, borderWidth, flow } =\n widget as unknown as RepeaterOptions;\n\n const { w, h, x: widgetX, y: widgetY } = getApparentDims(widget);\n\n const x = e.offsetX / scale;\n const y = e.offsetY / scale;\n\n const cellWidth = (w - columnGap * (columns + 1)) / columns;\n const cellHeight = (h - rowGap * (rows + 1)) / rows;\n\n let row = -1;\n let column = -1;\n let offsetX = 0;\n let offsetY = 0;\n for (let r = 0; r < rows; r++) {\n for (let c = 0; c < columns; c++) {\n const minX = (c + 1) * columnGap + c * cellWidth;\n const maxX = minX + cellWidth;\n const minY = (r + 1) * rowGap + r * cellHeight;\n const maxY = minY + cellHeight;\n if (x >= minX && x <= maxX && y >= minY && y <= maxY) {\n offsetX = minX;\n offsetY = minY;\n row = r;\n column = c;\n }\n }\n }\n\n offsetX += borderWidth;\n offsetY += borderWidth;\n\n if (row === -1 || column === -1) {\n return undefined;\n }\n\n if (rows === 1 && columns === 1) {\n offsetX = 0;\n offsetY = 0;\n }\n\n const repeaterIndex = flow.includes(\"row\")\n ? row * columns + column\n : column * rows + row;\n\n return {\n parentId: widget.wid,\n widgetX,\n widgetY,\n offsetX,\n offsetY,\n repeaterIndex,\n width: cellWidth,\n height: cellHeight,\n };\n};\n\n/**\n * This is sort of like `Promise.allSettled()` except it just swallows all errors.\n *\n * https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Promise/allSettled\n *\n * See Stack Overflow for this technique\n * https://stackoverflow.com/a/36115549/5651\n */\nexport const allSettled = (promises: Promise[]) => {\n return Promise.all(promises.map((p) => p.catch((e) => logger.track(e))));\n};\n\nexport const capitalize = (str: string) => {\n return `${str[0].toUpperCase() + str.slice(1)}`;\n};\n\n/**\n * Takes a css color string and returns true if the alpha channel is equal to zero.\n * @param value a CSS Color string (#RGB, #RGBA, #RRGGBB, #RRGGBBAA, rgba(0,0,0,0), hsla(0,0%,0%,0))\n * @returns boolean - true if color has zero alpha value\n */\nexport const isTransparent = (value: string): boolean => {\n if (!isNonEmptyString(value)) {\n return false;\n }\n\n // #RGBA\n if (value.length === 5) {\n return value.substring(4) === \"0\";\n }\n\n // #RRGGBBAA\n if (value.length === 9) {\n return value.substring(7) === \"00\";\n }\n\n // rgba(0,0,0,0)\n if (value.startsWith(\"rgba\")) {\n const parts = value.split(\",\");\n return parts[3].trim() === \"0)\";\n }\n\n if (value.startsWith(\"hsla\")) {\n const parts = value.split(\",\");\n return parseInt(parts[2].trim()) === 0;\n }\n\n // TODO: handle hsla\n\n return false;\n};\n\nconst parseHexString = (value: string) => {\n const result = [];\n while (value.length >= 2) {\n result.push(parseInt(value.substring(0, 2), 16));\n value = value.substring(2, value.length);\n }\n\n return result;\n};\n\nconst parseRgbString = (value: string) => {\n return value\n .substring(value.indexOf(\"(\") + 1, value.length - 1)\n .split(\",\")\n .map((x) => parseInt(x)); // not sure why this has different results than just .map(parseInt)..\n};\n\nconst parseHslString = (value: string) => {\n return value\n .substring(value.indexOf(\"(\") + 1, value.length - 1)\n .split(\",\")\n .map((x) => parseInt(x));\n};\n\n/**\n * Takes a css color string and returns true if the color is \"near white\".\n * @param value a CSS Color string (#RGB, #RGBA, #RRGGBB, #RRGGBBAA, rgba(0,0,0,0))\n * @returns boolean - true if color is \"near white\"\n */\nexport const isNearWhite = (value: string): boolean => {\n let parts: number[] = [];\n if (value.startsWith(\"#\")) {\n parts = parseHexString(value.slice(1));\n } else if (value.startsWith(\"rgb\")) {\n parts = parseRgbString(value);\n } else if (value.startsWith(\"hsl\")) {\n parts = parseHslString(value);\n return parts[2] > 90;\n }\n\n // For hex and rgb\n if (parts.length > 0) {\n return parts.slice(0, 3).every((p) => p > 230);\n }\n return false;\n};\n\nexport const clampSize = (size: Size, max: number) => {\n let { w, h } = size;\n if (w === h) return { w: max, h: max };\n if (w > h) {\n h /= w / max;\n w = max;\n } else {\n w /= h / max;\n h = max;\n }\n return { w, h };\n};\n\nexport const EMPTY_STATE = \"emptyState\";\n\nexport const purgeNullValues = (obj: any) => {\n if (!obj) return obj;\n const res: any = {};\n Object.keys(obj).forEach((key) => {\n if (obj[key] !== null) res[key] = obj[key];\n });\n return res;\n};\n\nexport const clamp = (val: number, r1: number, r2: number) => {\n if (val < r1) return r1;\n if (val > r2) return r2;\n return val;\n};\n\nexport const getRange = (start: number, end: number) => {\n const res = [];\n for (let i = start; i <= end; i++) {\n res.push(i);\n }\n return res;\n};\n\nexport const getTextCssFromProps = (style: BaseTextStyles) => {\n const s: Partial = {};\n if (!style) return s;\n\n Object.keys(style).forEach((key: keyof BaseTextStyles) => {\n if (style[key] === \"inherit\") return;\n (s as any)[key] = style[key];\n // Map to correct CSS property name\n if (key === \"textColor\") {\n s.color = style[key];\n }\n if (key === \"fontSize\") {\n s.fontSize = `${style[key]}px`;\n }\n if (key === \"alignHorizontal\") {\n switch (style[key]) {\n case \"left\":\n s.justifyContent = \"flex-start\";\n break;\n case \"center\":\n s.justifyContent = \"center\";\n break;\n case \"right\":\n s.justifyContent = \"flex-end\";\n break;\n }\n s.textAlign = style[key];\n }\n if (key === \"letterSpacing\") {\n s.letterSpacing = `${style[key]}pt`;\n }\n });\n return s;\n};\n\n// export const dummyCalendarData = [\n// {\n// Summary: \"All Day Event Title\",\n// Start: makeHourTodayDate(0).toISO(),\n// End: makeHourTodayDate(0, 1).toISO(),\n// Location: \"The Moon\"\n// },\n// {\n// Summary: \"Another Event Title\",\n// Start: makeHourTodayDate(9).toISO(),\n// End: makeHourTodayDate(11, 0, 45).toISO()\n// // Location: \"The Dark Side of the Moon\"\n// },\n// {\n// Summary: \"This is a Short Event!\",\n// Start: makeHourTodayDate(15, 0, 30).toISO(),\n// End: makeHourTodayDate(17).toISO(),\n// Location: \"Room 237\"\n// },\n// {\n// Summary: \"Multi Day Event Title\",\n// Start: makeHourTodayDate(0, -1).toISO(),\n// End: makeHourTodayDate(0, 2).toISO(),\n// Location: \"The Moon\"\n// },\n// {\n// Summary: \"Earth rotates\",\n// Start: makeHourTodayDate(15, 2).toISO(),\n// End: makeHourTodayDate(17, 2).toISO()\n// // Location: \"The Dark Side of the Moon\"\n// },\n// {\n// Summary: \"Time passes\",\n// Start: makeHourTodayDate(18, 1).toISO(),\n// End: makeHourTodayDate(20, 1).toISO(),\n// Location: \"Room 237\"\n// },\n// {\n// Summary: \"Multi Day Event Title 2\",\n// Start: makeHourTodayDate(0, 1).toISO(),\n// End: makeHourTodayDate(0, 3).toISO(),\n// Location: \"The Moon\"\n// },\n// {\n// Summary: \"Something happens\",\n// Start: makeHourTodayDate(11, -3).toISO(),\n// End: makeHourTodayDate(17, -3, 15).toISO()\n// // Location: \"The Dark Side of the Moon\"\n// },\n// {\n// Summary: \"This is an event!\",\n// Start: makeHourTodayDate(10, -2).toISO(),\n// End: makeHourTodayDate(12, -2, 30).toISO(),\n// Location: \"Room 237\"\n// }\n// ];\n\nexport const getEventXProportions = (\n event: { start: DateTime; end: DateTime },\n interval: { start: DateTime; end: DateTime }\n) => {\n const res = [0, 1];\n const intervalDuration = interval.end.diff(interval.start).milliseconds; // positive if first is *after* second\n\n // If event starts after interval start, compute its left position:\n const startDiff = event.start.diff(interval.start).milliseconds;\n if (startDiff > 0) {\n res[0] = startDiff / intervalDuration;\n }\n // If event ends before interval end, compute its right position:\n const endDiff = interval.end.diff(event.end).milliseconds;\n if (endDiff) {\n res[1] = 1 - endDiff / intervalDuration;\n }\n return res;\n};\n\nexport const isCurrency = (val: string): boolean => {\n const regex = /\\$/g;\n return regex.test(val);\n};\n\n/**\n * For now, assumes currency is U.S.$, but should internationalize in the future\n */\nexport const formatCurrencyString = (val: number): string => {\n const formatter = new Intl.NumberFormat(\"en-US\", {\n style: \"currency\",\n currency: \"USD\",\n\n // These options are needed to round to whole numbers if that's what you want.\n //minimumFractionDigits: 0, // (this suffices for whole numbers, but will print 2500.10 as $2,500.1)\n //maximumFractionDigits: 0, // (causes 2500.99 to be printed as $2,501)\n });\n return formatter.format(val);\n};\n\n/*\nThis should return x,y,w,h of bg image, given its scale, \noriginal dims, and the dims of the component.\nThis will allow us to determine limits on repositioning the bg image,\nand also to position the image within its svg.\n*/\nexport const placeCenteredBackgroundImage = (\n wg: any,\n imgSize: Size,\n imgScl: number\n): Position => {\n const { w, h } = getApparentDims(wg);\n // console.log(\"place...\", wg.w, wg.scaleX, w)\n let x = 0,\n y = 0,\n width = 0,\n height = 0;\n const widgetAR = w / h;\n const imgAR = imgSize.w / imgSize.h;\n if (imgAR > widgetAR) {\n height = h;\n width = height * imgAR;\n x = (w - width) / 2;\n } else {\n width = w;\n height = width / imgAR;\n y = (h - height) / 2;\n }\n const wgCenter = { x: w / 2, y: h / 2 };\n const imgRelWgCenter = subtractVectors({ x, y }, wgCenter);\n const scaledVec = scaleVector(imgRelWgCenter, imgScl);\n const finalPt = addVectors(wgCenter, scaledVec);\n const final = {\n x: finalPt.x,\n y: finalPt.y,\n w: width * imgScl,\n h: height * imgScl,\n z: 1,\n };\n\n const clean = {\n x: isNaN(final.x) ? 0 : final.x,\n y: isNaN(final.y) ? 0 : final.y,\n w: isNaN(final.w) ? 0 : final.w,\n h: isNaN(final.h) ? 0 : final.h,\n z: 1,\n };\n\n return clean;\n};\n\nexport const getPolygonPath = (payload: {\n ctr: Point;\n numSides: number;\n radius: number;\n innerRadius: number;\n starEnabled: boolean;\n borderWidth: number;\n}) => {\n const { ctr, numSides, radius, innerRadius, starEnabled } = payload;\n\n // TODO: How to deal with this?\n // This moves the star down a bit (away from perfect center but makes more visual sense):\n // ctr.y *= 1.1;\n // radius -= borderWidth;\n\n // const percent = 100 - (radius - borderWidth) / borderWidth;\n // console.log(\"draw poly\", radius, innerRadius, borderWidth, percent);\n // radius *= (percent / 100);\n // innerRadius *= (percent / 100);\n\n //Commented out the parts that change the shape based on borderWidth\n //We can mask the shape with a copy of itself with no stroke, double the border width,\n //and it will act the same\n //const innerAngle = ((Math.PI / 2) * (numSides - 2)) / numSides;\n // const radiusRemove = borderWidth / Math.sin(innerAngle / 2);\n\n // Close enough:\n // radius -= radiusRemove;\n // innerRadius -= radiusRemove / 2;\n\n const pts = [];\n const angle = (2 * Math.PI) / numSides;\n for (let i = 0; i < numSides; i++) {\n // outer points\n pts.push({\n x: ctr.x + radius * Math.cos(-Math.PI / 2 + angle * i),\n y: ctr.y + radius * Math.sin(-Math.PI / 2 + angle * i),\n });\n if (starEnabled) {\n // inner points (for star)\n pts.push({\n x: ctr.x + innerRadius * Math.cos(-Math.PI / 2 + angle * (i + 0.5)),\n y: ctr.y + innerRadius * Math.sin(-Math.PI / 2 + angle * (i + 0.5)),\n });\n }\n }\n let res = \"M \";\n pts.forEach((pt, i) => {\n if (i == 0) {\n res += `${pt.x} ${pt.y} `;\n } else {\n res += `L ${pt.x} ${pt.y} `;\n }\n });\n return res + \"Z\";\n};\n\n// const fitInto = (display: Size, boundary: Size): Size => {\n// const wr = display.w / boundary.w;\n// const hr = display.h / boundary.h;\n// const proportion = 1 / Math.max(wr, hr);\n// // if (wr >1 || hr > 1) {\n// // proportion = 1 / Math.max(wr, hr);\n// // } else {\n// // proportion = 1 / Math.max(wr, hr);\n// // }\n// return { w: display.w * proportion, h: display.h * proportion };\n// };\n\n// drop-shadow or box-shadow?\n// hmm...drop is less supported, and it does NOT support spread...also no inset shadows...\n// but dropshadow would work on svg....\n\nexport const capitalizeAll = (str: string) => {\n return str\n .split(\" \")\n .map((s) => `${s[0].toUpperCase()}${s.slice(1)}`)\n .join(\" \");\n};\n// Ah shoot, I forgot about border-radius...\n// Oh funny, drop-shadow also adds shadow to the bounding box and handles...huh...\n\n// example input:\n// \"10px 10px 2px hsla(210, 50%, 50%, 1)\",\n// which will get piped in to \"filter: drop-shadow(${stuff})\"\n// or be added directly as \"text-shadow\"\nexport const parseShadowCss = (str: string): ShadowProps => {\n // console.log(\"shadow css:\", str);\n if (!str) return { x: 0, y: 0, blur: 0, color: \"\" };\n\n let color;\n // NOTE: we can also have hex strings...we could just remove all those?\n // we can also have just \"orange\" etc....Yikes\n // const regex = /(hsla[^)]*\\))/g;\n\n // TODO:\n // This should capture hex as well:\n // Ok still missing some.....Not sure if this is the issue or not....keep looking\n const regex = /(hsla[^)]*\\))|(#[A-Fa-f0-9]+)/g;\n const cleanStr = str.replace(regex, (match: string) => {\n color = match;\n return \"\";\n });\n const parts = cleanStr\n .split(\" \")\n .filter((x) => x)\n .map((x) => parseInt(x));\n const x = parts[0];\n const y = parts[1];\n const blur = parts[2];\n\n return { x, y, color: color || \"\", blur };\n};\n\n// console.log(\"TEST:\", parseShadowCss(\"10px 10px 2px hsla(210, 50%, 50%, 1)\"));\n\n// example inputs:\n// \"hsla(210, 85%, 65%, 1)\"\n// \"linear-gradient(40deg, hsla(210, 85%, 65%, 1) 10%, hsla(310, 85%, 65%, 1) 90%)\"\n// \"radial-gradient(hsla(210, 85%, 65%, 1) 10%, hsla(310, 85%, 65%, 1) 90%)\"\nexport const parseGradientCss = (str: string) => {\n let type = \"Solid\";\n\n if (!str) return { type: \"\", color_1: \"\" };\n\n if (str.includes(\"linear\")) type = \"Linear\";\n if (str.includes(\"radial\")) type = \"Radial\";\n\n const colors: string[] = [];\n\n // -[ ] TODO: we can also have hex strings...we could just remove all those?\n const regex = /(hsla[^)]*\\))/g;\n const regex2 = /(hsl[^)]*\\))/g;\n\n const cleanStr = str\n .replace(regex, (match: string) => {\n colors.push(match);\n return \"\";\n })\n .replace(regex2, (match: string) => {\n colors.push(match);\n return \"\";\n });\n\n // console.log('x:', cleanStr);\n\n if (type === \"Solid\") {\n return {\n type,\n color: str,\n };\n }\n\n const start = cleanStr.indexOf(\"(\") + 1;\n const end = cleanStr.lastIndexOf(\")\");\n const parts = cleanStr\n .slice(start, end)\n .split(\",\")\n .map((x) => parseInt(x));\n\n const angle = type === \"Linear\" ? parts[0] : 0;\n const offset_1 = type === \"Linear\" ? parts[1] : parts[0];\n const offset_2 = type === \"Linear\" ? parts[2] : parts[1];\n const color_1 = colors[0];\n const color_2 = colors[1];\n\n return {\n type,\n color_1,\n color_2,\n offset_1,\n offset_2,\n angle,\n };\n};\n\nexport interface SnapAxis {\n name: string;\n value: number;\n dist?: number;\n wid?: string;\n id?: string; // adding for custom snap axes, which we want to keep track of by id, but without association to a widget\n}\n\n// This assumes for now that zooming transform origin is always 50 50\n// which is maybe fine..perhaps we can just accompany scale with a translation\n\n// Translates from \"canvas space\" to \"screen space\":\nexport const scaledCoords = (\n point: Point,\n canvas: Size | Rectangle,\n scale: number\n): Point => {\n const canvasCenter = { x: canvas.w / 2, y: canvas.h / 2 };\n const coords = subtractVectors(point, canvasCenter);\n //const coords= {\n // x: point.x - canvas.w / 2,\n // y: point.y - canvas.h / 2\n // };\n return {\n x: coords.x * scale + canvas.w / 2,\n y: coords.y * scale + canvas.h / 2,\n };\n};\n\n// Translates from \"screen space\" to \"canvas space\":\n// (I think the same could be accomplished by passing in 1/scale to scaledCoords)\nexport const scaledCoordsInvert = (\n point: Point,\n canvas: Size | Rectangle,\n scale: number\n): Point => {\n return {\n x: (point.x - canvas.w / 2) / scale + canvas.w / 2,\n y: (point.y - canvas.h / 2) / scale + canvas.h / 2,\n };\n};\n\nexport const isSameAxis = (first: SnapAxis, second: SnapAxis): boolean => {\n const isX = first.name.includes(\"x\");\n const tgtX = second.name.includes(\"x\");\n const isY = first.name.includes(\"y\");\n const tgtY = second.name.includes(\"y\");\n return (isX && tgtX) || (isY && tgtY);\n};\n\nexport const getAxes = (\n left: number,\n top: number,\n width: number,\n height: number\n) => {\n return {\n x: left,\n cx: left + width / 2,\n rx: left + width,\n y: top,\n cy: top + height / 2,\n by: top + height,\n };\n};\n\nexport const getAxesAbs = (\n left: number,\n top: number,\n right: number,\n bottom: number\n) => {\n const width = right - left;\n const height = bottom - top;\n return {\n x: left,\n cx: left + width / 2,\n rx: left + width,\n y: top,\n cy: top + height / 2,\n by: top + height,\n };\n};\n\nexport const isWithin = (point: Point, rect: Rectangle) => {\n if (!point || !rect) return false;\n return (\n point.x >= rect.x &&\n point.x <= rect.x + rect.w &&\n point.y >= rect.y &&\n point.y <= rect.y + rect.h\n );\n};\n\n// TODO: Abstract shared from these 2\n// Difference is that this will allow box to sit around a single element,\n// as should happen for nested group single children\n\nexport const addVectors = (v1: Point, v2: Point) => {\n return { x: v1.x + v2.x, y: v1.y + v2.y };\n};\n\nexport const subtractVectors = (v1: Point, v2: Point) => {\n return { x: v1.x - v2.x, y: v1.y - v2.y };\n};\n\n// Takes a mouse click position and maps it into canvas coordinates, accounting for scale:\nexport const screenCoordsToCanvas = (\n pt: Point,\n canvasBox: Rectangle,\n scale: number\n) => {\n const ptRelCanvas = subtractVectors(pt, canvasBox);\n const canvasCtr = {\n x: canvasBox.w / 2,\n y: canvasBox.h / 2,\n };\n const ptRelCtr = subtractVectors(ptRelCanvas, canvasCtr);\n const scaledPtRelCtr = {\n x: ptRelCtr.x / scale,\n y: ptRelCtr.y / scale,\n };\n const scaledPt = addVectors(scaledPtRelCtr, canvasCtr);\n return scaledPt;\n};\n\n// TAKES IN ANGLE IN RADIANS:\nexport const rotateVector = (v1: Point, a: number) => {\n return {\n x: v1.x * Math.cos(a) - v1.y * Math.sin(a),\n y: v1.x * Math.sin(a) + v1.y * Math.cos(a),\n };\n};\n\nexport const scaleVector = (v1: Point, k: number) => {\n return { x: v1.x * k, y: v1.y * k };\n};\n/*\nAs I tug RIGHT, x and scaleX update; same for DOWN and y\nAs i tug LEFT, only scaleX changes...\n\nI think the key is going to be that w, h never change, after wg is created. They are static.\n\nSo if x is 500, w is 200, scaleX is 2....\nWe compute where right side is, wwhich is x+w = 700. And then go back 2 * w, to 300.\n\nI think that's it..\n\nApparent w and h are always going to be scaleX*w, or scaleY*h\nAnd apparent x and y are calculated as above\n\nThis function helps us deal with the weird way that free-transform keeps track of position:\n*/\nexport const getApparentDims = (wg: TransformOptions): TransformOptions => {\n if (!wg) {\n return Object.assign({}, DefaultTransformOptions, { w: 0, h: 0 });\n }\n const { x, y, w, h, scaleX, scaleY } = wg;\n return {\n ...wg,\n //Rounding to one decimal place\n x: Math.round((x + w - scaleX * w) * 10) / 10,\n y: Math.round((y + h - scaleY * h) * 10) / 10,\n w: Math.round(w * scaleX * 10) / 10,\n h: Math.round(h * scaleY * 10) / 10,\n scaleX: 1,\n scaleY: 1,\n };\n};\n\n// so...say real x is 100, but apparentX is 200. User enters 201 as new x....so they want apparentX so be 201...\n// so what must x,y be, given w,h,and scales, to produce certain apparentX and y?\n// Actually, what must x, y, w and h be, given apparent x, y, w, h, and scaleX and scaleY\n// Might be better to alter scaleY than h...Same issue with text component resizing\nexport const getApparentDimsInvert = (data: any): Rectangle => {\n // Ensures that when real w and h are scaled by scaleX and scaleY, they yield the desired apparent w and h:\n const w = data.w / data.scaleX;\n const h = data.h / data.scaleY;\n const x = data.x - (1 - data.scaleX) * w;\n const y = data.y - (1 - data.scaleY) * h;\n return { x, y, w, h };\n};\n\n// If scaleY is not 1, then y must change as well as h to produce a change in apparentH that leaves apparentY fixed\nexport const dimsToUpdateApparentHeight = (\n t: TransformOptions,\n newHeight: number\n) => {\n const h = newHeight / t.scaleY;\n // What must y be changed to, given our new h, if apparentY is to be preserved?\n const y = t.y + (t.h - h) * (1 - t.scaleY);\n\n return { h, y };\n};\n\nexport const dimsToUpdateApparentWidth = (\n t: TransformOptions,\n newWidth: number\n) => {\n const w = newWidth / t.scaleX;\n const x = t.x + (t.w - w) * (1 - t.scaleX);\n return { w, x };\n};\n\n/**\n * Computes the area of a triangle defined by 3 vertices.\n * Thanks https://stackoverflow.com/questions/17136084/checking-if-a-point-is-inside-a-rotated-rectangle\n */\nconst computeArea = (triangle: Point[]): number => {\n const [p1, p2, p3] = triangle;\n const area =\n Math.abs(\n p2.x * p1.y -\n p1.x * p2.y +\n (p3.x * p2.y - p2.x * p3.y) +\n (p1.x * p3.y - p3.x * p1.y)\n ) / 2;\n return area;\n};\n\n/**\n * Computes distance between two vertices.\n */\nconst computeDistance = (p1: Point, p2: Point): number => {\n const xDiff = p1.x - p2.x;\n const yDiff = p1.y - p2.y;\n return Math.sqrt(xDiff * xDiff + yDiff * yDiff);\n};\n\n/**\n * Computes whether a point is within a rotated rectangular region defined by 4 vertices.\n * Used for checking whether a click lies within a rotated widget (used for group move, and deselect).\n * Thanks https://stackoverflow.com/questions/17136084/checking-if-a-point-is-inside-a-rotated-rectangle\n *\n * Rectangles should be provided in this order (clockwise from upper left):\n * P1----P2\n * | |\n * P4----P3\n */\nexport const isWithinRotatedRectangle = (\n pt: Point,\n rectangle: Point[]\n): boolean => {\n const [p1, p2, p3, p4] = rectangle;\n const width = computeDistance(p1, p2);\n const height = computeDistance(p1, p4);\n const triangleAreas = [\n computeArea([p1, pt, p4]),\n computeArea([p3, pt, p4]),\n computeArea([p3, pt, p2]),\n computeArea([p1, pt, p2]),\n ];\n const triangleAreasSum = triangleAreas.reduce((sum, area) => sum + area, 0);\n const rectangleArea = width * height;\n\n if (rectangleArea < 0.01) return false;\n\n /**\n * The point is within the region iff sum of triangle areas is less than or equal to area of rectangle.\n * In some cases, rectangle area ends up SLIGHTLY smaller (0.0000001) due to rounding issues.\n * So add just a touch to it, to yield right answer in that case.\n */\n\n return triangleAreasSum <= rectangleArea + 0.01;\n};\n\n/**\n * Thanks to https://stackoverflow.com/questions/10962379/how-to-check-intersection-between-2-rotated-rectangles\n * Helper function to determine whether there is an intersection between the two polygons described\n * by the lists of vertices. Uses the Separating Axis Theorem\n */\nexport const doPolygonsIntersect = (a: Point[], b: Point[]): boolean => {\n var polygons = [a, b];\n var minA, maxA, projected, i, i1, j, minB, maxB;\n\n for (i = 0; i < polygons.length; i++) {\n // for each polygon, look at each edge of the polygon, and determine if it separates\n // the two shapes\n var polygon = polygons[i];\n for (i1 = 0; i1 < polygon.length; i1++) {\n // grab 2 vertices to create an edge\n var i2 = (i1 + 1) % polygon.length;\n var p1 = polygon[i1];\n var p2 = polygon[i2];\n\n // find the line perpendicular to this edge\n var normal = { x: p2.y - p1.y, y: p1.x - p2.x };\n\n // for each vertex in the first shape, project it onto the line perpendicular to the edge\n // and keep track of the min and max of these values\n minA = Infinity;\n maxA = -Infinity;\n\n for (j = 0; j < a.length; j++) {\n projected = normal.x * a[j].x + normal.y * a[j].y;\n if (projected < minA) {\n minA = projected;\n }\n if (projected > maxA) {\n maxA = projected;\n }\n }\n\n // for each vertex in the second shape, project it onto the line perpendicular to the edge\n // and keep track of the min and max of these values\n minB = Infinity;\n maxB = -Infinity;\n\n for (j = 0; j < b.length; j++) {\n projected = normal.x * b[j].x + normal.y * b[j].y;\n if (projected < minB) {\n minB = projected;\n }\n if (projected > maxB) {\n maxB = projected;\n }\n }\n\n // if there is no overlap between the projects, the edge we are looking at separates the two\n // polygons, and we know there is no overlap\n if (maxA < minB || maxB < minA) {\n // console.log(\"polygons don't intersect!\");\n return false;\n }\n }\n }\n return true;\n};\n\n/**\n * Gets the positions of the corners of a possibly rotated box.\n *\n * Returns points in this order:\n * P1----P2\n * | |\n * P4----P3\n */\nexport const getVertices = (box: TransformOptions): Point[] => {\n const { x, y, w, h, angle } = box;\n\n const ctr = { x: x + w / 2, y: y + h / 2 };\n const ptRelCtr = subtractVectors({ x, y }, ctr);\n const radians = (Math.PI * angle) / 180;\n const rotatedVec = rotateVector(ptRelCtr, radians);\n const rotatedW = rotateVector({ x: w, y: 0 }, radians);\n const rotatedH = rotateVector({ x: 0, y: h }, radians);\n const pt1 = addVectors(ctr, rotatedVec);\n const pt2 = addVectors(pt1, rotatedW);\n const pt3 = addVectors(pt2, rotatedH);\n const pt4 = addVectors(pt1, rotatedH);\n return [pt1, pt2, pt3, pt4];\n};\n\n/*\nGets the smallest bounding rectangle that fits around another rectangle, accounting for possible rotation:\n*/\nexport const getAngledBoundingBox = (values: TransformOptions): Rectangle => {\n const { x, y, w, h, angle } = values;\n if (angle === 0) {\n return { x, y, w, h };\n }\n\n const [pt1, pt2, pt3, pt4] = getVertices(values);\n\n const minX = Math.min(pt1.x, pt2.x, pt3.x, pt4.x);\n const minY = Math.min(pt1.y, pt2.y, pt3.y, pt4.y);\n const maxX = Math.max(pt1.x, pt2.x, pt3.x, pt4.x);\n const maxY = Math.max(pt1.y, pt2.y, pt3.y, pt4.y);\n return { x: minX, y: minY, w: maxX - minX, h: maxY - minY };\n};\n\nexport const getBoundingBox = (values: TransformOptions[]): Rectangle => {\n let x = Infinity;\n let y = Infinity;\n let w = 0;\n let h = 0;\n\n values.forEach((value: TransformOptions) => {\n const box = getAngledBoundingBox(getApparentDims(value));\n\n x = Math.min(box.x, x);\n y = Math.min(box.y, y);\n w = Math.max(box.x + box.w, w);\n h = Math.max(box.y + box.h, h);\n });\n\n x = x === Infinity ? 0 : x;\n y = y === Infinity ? 0 : y;\n\n return { x, y, w: w - x, h: h - y };\n};\n\n// NOTE: If we have to just keep recomputing this stuff...\n// maybe best to keep using absolute position...\n// But then have to manually update all ys after every child change...a pain...\nexport const getBoundingBoxWithVerticalDynamism = (\n values: TransformOptions[],\n verticalMargin: number\n): Rectangle => {\n let x = Infinity;\n let y = Infinity;\n let w = 0;\n // let h = 0;\n\n let totalHeight = 0;\n\n values.forEach((value: TransformOptions) => {\n const box = getAngledBoundingBox(getApparentDims(value));\n\n totalHeight += box.h;\n\n x = Math.min(box.x, x);\n y = Math.min(box.y, y);\n w = Math.max(box.x + box.w, w);\n // h = Math.max(box.y + box.h, h);\n });\n\n totalHeight += (values.length - 1) * verticalMargin;\n\n x = x === Infinity ? 0 : x;\n y = y === Infinity ? 0 : y;\n\n return { x, y, w: w - x, h: totalHeight };\n};\n\n// Only use letters that aren't easily confused with numbers\nconst characters = \"abcdefghjkmnpqrstuvwxyz0123456789\";\nconst charactersLength = characters.length;\nconst lettersLength = characters.indexOf(\"0\");\nexport const makeId = (length = 8): string => {\n let result = characters.charAt(Math.floor(Math.random() * lettersLength));\n for (let i = 0; i < length - 1; i++) {\n result += characters.charAt(Math.floor(Math.random() * charactersLength));\n }\n return result;\n};\n\nexport const isNaN = Number.isNaN || (window).isNaN;\nconst REGEXP_NUMBER = /^-?(?:\\d+|\\d+\\.\\d+|\\.\\d+)(?:[eE][-+]?\\d+)?$/;\nconst REGEXP_DECIMALS = /\\.\\d*(?:0|9){10}\\d*$/;\nexport const normalizeDecimalNumber = (value: any, times = 100000000000) =>\n REGEXP_DECIMALS.test(value) ? Math.round(value * times) / times : value;\n\nexport const isNumber = (value: string) => REGEXP_NUMBER.test(value);\n\nexport const quantizeNumber = (\n value: number,\n quantum: number,\n cover = true\n) => {\n if (!quantum) {\n return 0;\n }\n const remainder = value % quantum;\n const sign = value >= 0 ? 1 : -1;\n const mod = cover && remainder ? quantum : 0;\n return value - remainder + sign * mod;\n};\n\nexport const getImageDimensions = (url: string): Promise => {\n const img = new Image();\n img.style.opacity = \"0.01\";\n document.body.appendChild(img);\n img.src = url;\n return new Promise((resolve, reject) => {\n img.onload = function () {\n img.onload = null;\n setTimeout(() => {\n const size = {\n w: img.naturalWidth,\n h: img.naturalHeight,\n };\n\n document.body.removeChild(img);\n resolve(size);\n }, 20);\n };\n img.onerror = function (e) {\n document.body.removeChild(img);\n img.onerror = null;\n reject(e);\n };\n });\n};\n\nexport const removeReactivity = (value: unknown): T => {\n if (typeof value === \"undefined\") {\n return value as unknown as T;\n }\n return JSON.parse(JSON.stringify(value)) as T;\n};\n\nexport const range = (start: number, stop: number, step = 1) =>\n Array(Math.ceil((stop - start) / step))\n .fill(start)\n .map((x, y) => x + y * step);\n\nexport const removeDuplicateStrings = (values: string[]) => {\n let i: number;\n const len = values.length;\n const out: string[] = [];\n const obj: any = {};\n\n for (i = 0; i < len; i++) {\n obj[values[i]] = 0;\n }\n for (const key in obj) {\n out.push(key);\n }\n return out;\n};\n\n/**\n * Grabs a value from a vue-router query object.\n * @returns The value\n */\nexport const getQueryValue = (value: string | (string | null)[]): T => {\n if (Array.isArray(value) && value.length > 0) {\n return value[0] as unknown as T;\n }\n return value as unknown as T;\n};\n\nexport const imageResize = (\n src: string,\n width?: number,\n height?: number,\n mode: \"max\" | \"crop\" = \"max\"\n) => {\n if (typeof width === \"undefined\" && typeof height === \"undefined\") {\n return src;\n }\n\n if (!isNonEmptyString(src)) {\n return undefined;\n }\n\n try {\n const url = new URL(src);\n url.searchParams.set(\"width\", Math.round(width as number).toString());\n url.searchParams.set(\"height\", Math.round(height as number).toString());\n url.searchParams.set(\"mode\", mode);\n return url.toString();\n } catch (err) {\n logger.track(err);\n return undefined;\n }\n};\n\nexport const isNullOrUndefined = (value: any) => {\n return typeof value === \"undefined\" || value === null;\n};\n\nexport const throttle = (callback: (context: any, args: any) => void) => {\n let lastArgs: any;\n let timer: number | null = null;\n\n const later = (context: any) => () => {\n timer = null;\n callback.apply(context, lastArgs);\n };\n\n const throttled = function (...args: any[]) {\n lastArgs = args;\n if (timer === null) {\n // @ts-ignore\n timer = window.requestAnimationFrame(later(this));\n }\n };\n\n throttled.cancel = () => {\n cancelAnimationFrame(timer as number);\n timer = null;\n };\n\n return throttled;\n};\n\n/**\n * Returns a scale value that when applied to the source will make the source fit within the destination\n * @param destination The target size\n * @param source The starting size\n * @returns\n */\nexport const getScaleFactor = (destination: Size, source: Size): number => {\n const widthRatio = source.w / destination.w;\n const heightRatio = source.h / destination.h;\n const proportion = 1 / Math.max(widthRatio, heightRatio);\n return proportion;\n};\n\n/**\n * Modifies scalable widget properties according to the ratio betwen app size and render size.\n * @param widget The widget to modify\n * @param scaleFactor The amount to scale the widget properties\n * @returns Modified widget\n */\nexport const scaleWidget = (widget: Widget, scaleFactor: number) => {\n const PROPS_TO_SCALE = [\n \"w\",\n \"h\",\n \"x\",\n \"y\",\n \"columnGap\",\n \"rowGap\",\n \"shadowX\",\n \"shadowY\",\n \"shadowBlur\",\n \"borderWidth\",\n ];\n\n const PROPS_SUFFIXES_TO_SCALE = [\"fontSize\", \"letterSpacing\"];\n\n const result = { ...widget };\n PROPS_TO_SCALE.forEach((prop) => {\n (result as any)[prop] *= scaleFactor;\n // Must yield exact same value as we predict in getImageQueryParams\n (result as any)[prop] = Math.floor((result as any)[prop]);\n });\n\n Object.keys(widget).forEach((key) => {\n if (PROPS_SUFFIXES_TO_SCALE.some((p) => key.includes(p))) {\n (result as any)[key] *= scaleFactor;\n // Must yield exact same value as we predict in getImageQueryParams\n (result as any)[key] = Math.floor((result as any)[key]);\n }\n });\n\n return result;\n};\n\nexport const extractErrorMessage = (\n err: BackendError[] | { message: string } | string,\n defaultMessage = \"\"\n) => {\n let msg = \"\";\n if (Array.isArray(err)) {\n msg = err[0].message;\n } else if (typeof err === \"object\" && \"message\" in err) {\n msg = err.message;\n } else if (isNonEmptyString(err)) {\n msg = err;\n }\n if (!isNonEmptyString(msg)) {\n return defaultMessage;\n }\n return msg;\n};\n\n/**\n * Returns the value of a Vue Router query param as a string or undefined.\n * Query params with multiple values will be returned as a comma-delimited string.\n * @param value Parameter key\n */\nexport const getRouteQueryValue = (value: string | (string | null)[]) => {\n if (Array.isArray(value)) {\n return value.filter((v) => v !== null).join(\",\");\n }\n if (isNonEmptyString(value)) {\n return value as string;\n }\n return undefined;\n};\n\n/**\n * Takes a fragment ID and returns a an CSS `url()` value with the absolute URL prepended\n * if the browser requires it.\n * @param fragmentId Fragment ID of the SVG element (e.g. #abc234-bg-mask)\n * @example\n * // returns \"url(https://example.com/#abc234-bg-mask)\"\n */\nexport const svgUrl = (fragmentId: string) => {\n const urlAndPath = window.location.href.split(\"#\");\n const id = `#${fragmentId.replace(\"#\", \"\")}`;\n const url = svgUrlFixRequired ? urlAndPath + id : id;\n return `url(${url})`;\n};\n","\n\n\n","import Vue from 'vue'\n\nvar isVue2 = true\nvar isVue3 = false\nvar Vue2 = Vue\nvar warn = Vue.util.warn\n\nfunction install() {}\n\n// createApp polyfill\nexport function createApp(rootComponent, rootProps) {\n var vm\n var provide = {}\n var app = {\n config: Vue.config,\n use: Vue.use.bind(Vue),\n mixin: Vue.mixin.bind(Vue),\n component: Vue.component.bind(Vue),\n provide: function (key, value) {\n provide[key] = value\n return this\n },\n directive: function (name, dir) {\n if (dir) {\n Vue.directive(name, dir)\n return app\n } else {\n return Vue.directive(name)\n }\n },\n mount: function (el, hydrating) {\n if (!vm) {\n vm = new Vue(Object.assign({ propsData: rootProps }, rootComponent, { provide: Object.assign(provide, rootComponent.provide) }))\n vm.$mount(el, hydrating)\n return vm\n } else {\n return vm\n }\n },\n unmount: function () {\n if (vm) {\n vm.$destroy()\n vm = undefined\n }\n },\n }\n return app\n}\n\nexport { Vue, Vue2, isVue2, isVue3, install, warn }\nexport * from 'vue'\n","export function getDevtoolsGlobalHook() {\n return getTarget().__VUE_DEVTOOLS_GLOBAL_HOOK__;\n}\nexport function getTarget() {\n // @ts-ignore\n return (typeof navigator !== 'undefined' && typeof window !== 'undefined')\n ? window\n : typeof global !== 'undefined'\n ? global\n : {};\n}\nexport const isProxyAvailable = typeof Proxy === 'function';\n","export const HOOK_SETUP = 'devtools-plugin:setup';\nexport const HOOK_PLUGIN_SETTINGS_SET = 'plugin:settings:set';\n","let supported;\nlet perf;\nexport function isPerformanceSupported() {\n var _a;\n if (supported !== undefined) {\n return supported;\n }\n if (typeof window !== 'undefined' && window.performance) {\n supported = true;\n perf = window.performance;\n }\n else if (typeof global !== 'undefined' && ((_a = global.perf_hooks) === null || _a === void 0 ? void 0 : _a.performance)) {\n supported = true;\n perf = global.perf_hooks.performance;\n }\n else {\n supported = false;\n }\n return supported;\n}\nexport function now() {\n return isPerformanceSupported() ? perf.now() : Date.now();\n}\n","import { HOOK_PLUGIN_SETTINGS_SET } from './const.js';\nimport { now } from './time.js';\nexport class ApiProxy {\n constructor(plugin, hook) {\n this.target = null;\n this.targetQueue = [];\n this.onQueue = [];\n this.plugin = plugin;\n this.hook = hook;\n const defaultSettings = {};\n if (plugin.settings) {\n for (const id in plugin.settings) {\n const item = plugin.settings[id];\n defaultSettings[id] = item.defaultValue;\n }\n }\n const localSettingsSaveId = `__vue-devtools-plugin-settings__${plugin.id}`;\n let currentSettings = Object.assign({}, defaultSettings);\n try {\n const raw = localStorage.getItem(localSettingsSaveId);\n const data = JSON.parse(raw);\n Object.assign(currentSettings, data);\n }\n catch (e) {\n // noop\n }\n this.fallbacks = {\n getSettings() {\n return currentSettings;\n },\n setSettings(value) {\n try {\n localStorage.setItem(localSettingsSaveId, JSON.stringify(value));\n }\n catch (e) {\n // noop\n }\n currentSettings = value;\n },\n now() {\n return now();\n },\n };\n if (hook) {\n hook.on(HOOK_PLUGIN_SETTINGS_SET, (pluginId, value) => {\n if (pluginId === this.plugin.id) {\n this.fallbacks.setSettings(value);\n }\n });\n }\n this.proxiedOn = new Proxy({}, {\n get: (_target, prop) => {\n if (this.target) {\n return this.target.on[prop];\n }\n else {\n return (...args) => {\n this.onQueue.push({\n method: prop,\n args,\n });\n };\n }\n },\n });\n this.proxiedTarget = new Proxy({}, {\n get: (_target, prop) => {\n if (this.target) {\n return this.target[prop];\n }\n else if (prop === 'on') {\n return this.proxiedOn;\n }\n else if (Object.keys(this.fallbacks).includes(prop)) {\n return (...args) => {\n this.targetQueue.push({\n method: prop,\n args,\n resolve: () => { },\n });\n return this.fallbacks[prop](...args);\n };\n }\n else {\n return (...args) => {\n return new Promise(resolve => {\n this.targetQueue.push({\n method: prop,\n args,\n resolve,\n });\n });\n };\n }\n },\n });\n }\n async setRealTarget(target) {\n this.target = target;\n for (const item of this.onQueue) {\n this.target.on[item.method](...item.args);\n }\n for (const item of this.targetQueue) {\n item.resolve(await this.target[item.method](...item.args));\n }\n }\n}\n","import { getTarget, getDevtoolsGlobalHook, isProxyAvailable } from './env.js';\nimport { HOOK_SETUP } from './const.js';\nimport { ApiProxy } from './proxy.js';\nexport * from './api/index.js';\nexport * from './plugin.js';\nexport * from './time.js';\nexport function setupDevtoolsPlugin(pluginDescriptor, setupFn) {\n const descriptor = pluginDescriptor;\n const target = getTarget();\n const hook = getDevtoolsGlobalHook();\n const enableProxy = isProxyAvailable && descriptor.enableEarlyProxy;\n if (hook && (target.__VUE_DEVTOOLS_PLUGIN_API_AVAILABLE__ || !enableProxy)) {\n hook.emit(HOOK_SETUP, pluginDescriptor, setupFn);\n }\n else {\n const proxy = enableProxy ? new ApiProxy(descriptor, hook) : null;\n const list = target.__VUE_DEVTOOLS_PLUGINS__ = target.__VUE_DEVTOOLS_PLUGINS__ || [];\n list.push({\n pluginDescriptor: descriptor,\n setupFn,\n proxy,\n });\n if (proxy)\n setupFn(proxy.proxiedTarget);\n }\n}\n","/*!\n * pinia v2.0.22\n * (c) 2022 Eduardo San Martin Morote\n * @license MIT\n */\nimport { getCurrentInstance, inject, toRaw, watch, unref, markRaw, effectScope, ref, isVue2, isRef, isReactive, set, onUnmounted, reactive, toRef, del, nextTick, computed, toRefs } from 'vue-demi';\nimport { setupDevtoolsPlugin } from '@vue/devtools-api';\n\n/**\r\n * setActivePinia must be called to handle SSR at the top of functions like\r\n * `fetch`, `setup`, `serverPrefetch` and others\r\n */\r\nlet activePinia;\r\n/**\r\n * Sets or unsets the active pinia. Used in SSR and internally when calling\r\n * actions and getters\r\n *\r\n * @param pinia - Pinia instance\r\n */\r\nconst setActivePinia = (pinia) => (activePinia = pinia);\r\n/**\r\n * Get the currently active pinia if there is any.\r\n */\r\nconst getActivePinia = () => (getCurrentInstance() && inject(piniaSymbol)) || activePinia;\r\nconst piniaSymbol = ((process.env.NODE_ENV !== 'production') ? Symbol('pinia') : /* istanbul ignore next */ Symbol());\n\nfunction isPlainObject(\r\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\r\no) {\r\n return (o &&\r\n typeof o === 'object' &&\r\n Object.prototype.toString.call(o) === '[object Object]' &&\r\n typeof o.toJSON !== 'function');\r\n}\r\n// type DeepReadonly = { readonly [P in keyof T]: DeepReadonly }\r\n// TODO: can we change these to numbers?\r\n/**\r\n * Possible types for SubscriptionCallback\r\n */\r\nvar MutationType;\r\n(function (MutationType) {\r\n /**\r\n * Direct mutation of the state:\r\n *\r\n * - `store.name = 'new name'`\r\n * - `store.$state.name = 'new name'`\r\n * - `store.list.push('new item')`\r\n */\r\n MutationType[\"direct\"] = \"direct\";\r\n /**\r\n * Mutated the state with `$patch` and an object\r\n *\r\n * - `store.$patch({ name: 'newName' })`\r\n */\r\n MutationType[\"patchObject\"] = \"patch object\";\r\n /**\r\n * Mutated the state with `$patch` and a function\r\n *\r\n * - `store.$patch(state => state.name = 'newName')`\r\n */\r\n MutationType[\"patchFunction\"] = \"patch function\";\r\n // maybe reset? for $state = {} and $reset\r\n})(MutationType || (MutationType = {}));\n\nconst IS_CLIENT = typeof window !== 'undefined';\r\n/**\r\n * Should we add the devtools plugins.\r\n * - only if dev mode or forced through the prod devtools flag\r\n * - not in test\r\n * - only if window exists (could change in the future)\r\n */\r\nconst USE_DEVTOOLS = ((process.env.NODE_ENV !== 'production') || (typeof __VUE_PROD_DEVTOOLS__ !== 'undefined' && __VUE_PROD_DEVTOOLS__)) && !(process.env.NODE_ENV === 'test') && IS_CLIENT;\n\n/*\r\n * FileSaver.js A saveAs() FileSaver implementation.\r\n *\r\n * Originally by Eli Grey, adapted as an ESM module by Eduardo San Martin\r\n * Morote.\r\n *\r\n * License : MIT\r\n */\r\n// The one and only way of getting global scope in all environments\r\n// https://stackoverflow.com/q/3277182/1008999\r\nconst _global = /*#__PURE__*/ (() => typeof window === 'object' && window.window === window\r\n ? window\r\n : typeof self === 'object' && self.self === self\r\n ? self\r\n : typeof global === 'object' && global.global === global\r\n ? global\r\n : typeof globalThis === 'object'\r\n ? globalThis\r\n : { HTMLElement: null })();\r\nfunction bom(blob, { autoBom = false } = {}) {\r\n // prepend BOM for UTF-8 XML and text/* types (including HTML)\r\n // note: your browser will automatically convert UTF-16 U+FEFF to EF BB BF\r\n if (autoBom &&\r\n /^\\s*(?:text\\/\\S*|application\\/xml|\\S*\\/\\S*\\+xml)\\s*;.*charset\\s*=\\s*utf-8/i.test(blob.type)) {\r\n return new Blob([String.fromCharCode(0xfeff), blob], { type: blob.type });\r\n }\r\n return blob;\r\n}\r\nfunction download(url, name, opts) {\r\n const xhr = new XMLHttpRequest();\r\n xhr.open('GET', url);\r\n xhr.responseType = 'blob';\r\n xhr.onload = function () {\r\n saveAs(xhr.response, name, opts);\r\n };\r\n xhr.onerror = function () {\r\n console.error('could not download file');\r\n };\r\n xhr.send();\r\n}\r\nfunction corsEnabled(url) {\r\n const xhr = new XMLHttpRequest();\r\n // use sync to avoid popup blocker\r\n xhr.open('HEAD', url, false);\r\n try {\r\n xhr.send();\r\n }\r\n catch (e) { }\r\n return xhr.status >= 200 && xhr.status <= 299;\r\n}\r\n// `a.click()` doesn't work for all browsers (#465)\r\nfunction click(node) {\r\n try {\r\n node.dispatchEvent(new MouseEvent('click'));\r\n }\r\n catch (e) {\r\n const evt = document.createEvent('MouseEvents');\r\n evt.initMouseEvent('click', true, true, window, 0, 0, 0, 80, 20, false, false, false, false, 0, null);\r\n node.dispatchEvent(evt);\r\n }\r\n}\r\nconst _navigator = \r\n typeof navigator === 'object' ? navigator : { userAgent: '' };\r\n// Detect WebView inside a native macOS app by ruling out all browsers\r\n// We just need to check for 'Safari' because all other browsers (besides Firefox) include that too\r\n// https://www.whatismybrowser.com/guides/the-latest-user-agent/macos\r\nconst isMacOSWebView = /*#__PURE__*/ (() => /Macintosh/.test(_navigator.userAgent) &&\r\n /AppleWebKit/.test(_navigator.userAgent) &&\r\n !/Safari/.test(_navigator.userAgent))();\r\nconst saveAs = !IS_CLIENT\r\n ? () => { } // noop\r\n : // Use download attribute first if possible (#193 Lumia mobile) unless this is a macOS WebView or mini program\r\n typeof HTMLAnchorElement !== 'undefined' &&\r\n 'download' in HTMLAnchorElement.prototype &&\r\n !isMacOSWebView\r\n ? downloadSaveAs\r\n : // Use msSaveOrOpenBlob as a second approach\r\n 'msSaveOrOpenBlob' in _navigator\r\n ? msSaveAs\r\n : // Fallback to using FileReader and a popup\r\n fileSaverSaveAs;\r\nfunction downloadSaveAs(blob, name = 'download', opts) {\r\n const a = document.createElement('a');\r\n a.download = name;\r\n a.rel = 'noopener'; // tabnabbing\r\n // TODO: detect chrome extensions & packaged apps\r\n // a.target = '_blank'\r\n if (typeof blob === 'string') {\r\n // Support regular links\r\n a.href = blob;\r\n if (a.origin !== location.origin) {\r\n if (corsEnabled(a.href)) {\r\n download(blob, name, opts);\r\n }\r\n else {\r\n a.target = '_blank';\r\n click(a);\r\n }\r\n }\r\n else {\r\n click(a);\r\n }\r\n }\r\n else {\r\n // Support blobs\r\n a.href = URL.createObjectURL(blob);\r\n setTimeout(function () {\r\n URL.revokeObjectURL(a.href);\r\n }, 4e4); // 40s\r\n setTimeout(function () {\r\n click(a);\r\n }, 0);\r\n }\r\n}\r\nfunction msSaveAs(blob, name = 'download', opts) {\r\n if (typeof blob === 'string') {\r\n if (corsEnabled(blob)) {\r\n download(blob, name, opts);\r\n }\r\n else {\r\n const a = document.createElement('a');\r\n a.href = blob;\r\n a.target = '_blank';\r\n setTimeout(function () {\r\n click(a);\r\n });\r\n }\r\n }\r\n else {\r\n // @ts-ignore: works on windows\r\n navigator.msSaveOrOpenBlob(bom(blob, opts), name);\r\n }\r\n}\r\nfunction fileSaverSaveAs(blob, name, opts, popup) {\r\n // Open a popup immediately do go around popup blocker\r\n // Mostly only available on user interaction and the fileReader is async so...\r\n popup = popup || open('', '_blank');\r\n if (popup) {\r\n popup.document.title = popup.document.body.innerText = 'downloading...';\r\n }\r\n if (typeof blob === 'string')\r\n return download(blob, name, opts);\r\n const force = blob.type === 'application/octet-stream';\r\n const isSafari = /constructor/i.test(String(_global.HTMLElement)) || 'safari' in _global;\r\n const isChromeIOS = /CriOS\\/[\\d]+/.test(navigator.userAgent);\r\n if ((isChromeIOS || (force && isSafari) || isMacOSWebView) &&\r\n typeof FileReader !== 'undefined') {\r\n // Safari doesn't allow downloading of blob URLs\r\n const reader = new FileReader();\r\n reader.onloadend = function () {\r\n let url = reader.result;\r\n if (typeof url !== 'string') {\r\n popup = null;\r\n throw new Error('Wrong reader.result type');\r\n }\r\n url = isChromeIOS\r\n ? url\r\n : url.replace(/^data:[^;]*;/, 'data:attachment/file;');\r\n if (popup) {\r\n popup.location.href = url;\r\n }\r\n else {\r\n location.assign(url);\r\n }\r\n popup = null; // reverse-tabnabbing #460\r\n };\r\n reader.readAsDataURL(blob);\r\n }\r\n else {\r\n const url = URL.createObjectURL(blob);\r\n if (popup)\r\n popup.location.assign(url);\r\n else\r\n location.href = url;\r\n popup = null; // reverse-tabnabbing #460\r\n setTimeout(function () {\r\n URL.revokeObjectURL(url);\r\n }, 4e4); // 40s\r\n }\r\n}\n\n/**\r\n * Shows a toast or console.log\r\n *\r\n * @param message - message to log\r\n * @param type - different color of the tooltip\r\n */\r\nfunction toastMessage(message, type) {\r\n const piniaMessage = '🍍 ' + message;\r\n if (typeof __VUE_DEVTOOLS_TOAST__ === 'function') {\r\n __VUE_DEVTOOLS_TOAST__(piniaMessage, type);\r\n }\r\n else if (type === 'error') {\r\n console.error(piniaMessage);\r\n }\r\n else if (type === 'warn') {\r\n console.warn(piniaMessage);\r\n }\r\n else {\r\n console.log(piniaMessage);\r\n }\r\n}\r\nfunction isPinia(o) {\r\n return '_a' in o && 'install' in o;\r\n}\n\nfunction checkClipboardAccess() {\r\n if (!('clipboard' in navigator)) {\r\n toastMessage(`Your browser doesn't support the Clipboard API`, 'error');\r\n return true;\r\n }\r\n}\r\nfunction checkNotFocusedError(error) {\r\n if (error instanceof Error &&\r\n error.message.toLowerCase().includes('document is not focused')) {\r\n toastMessage('You need to activate the \"Emulate a focused page\" setting in the \"Rendering\" panel of devtools.', 'warn');\r\n return true;\r\n }\r\n return false;\r\n}\r\nasync function actionGlobalCopyState(pinia) {\r\n if (checkClipboardAccess())\r\n return;\r\n try {\r\n await navigator.clipboard.writeText(JSON.stringify(pinia.state.value));\r\n toastMessage('Global state copied to clipboard.');\r\n }\r\n catch (error) {\r\n if (checkNotFocusedError(error))\r\n return;\r\n toastMessage(`Failed to serialize the state. Check the console for more details.`, 'error');\r\n console.error(error);\r\n }\r\n}\r\nasync function actionGlobalPasteState(pinia) {\r\n if (checkClipboardAccess())\r\n return;\r\n try {\r\n pinia.state.value = JSON.parse(await navigator.clipboard.readText());\r\n toastMessage('Global state pasted from clipboard.');\r\n }\r\n catch (error) {\r\n if (checkNotFocusedError(error))\r\n return;\r\n toastMessage(`Failed to deserialize the state from clipboard. Check the console for more details.`, 'error');\r\n console.error(error);\r\n }\r\n}\r\nasync function actionGlobalSaveState(pinia) {\r\n try {\r\n saveAs(new Blob([JSON.stringify(pinia.state.value)], {\r\n type: 'text/plain;charset=utf-8',\r\n }), 'pinia-state.json');\r\n }\r\n catch (error) {\r\n toastMessage(`Failed to export the state as JSON. Check the console for more details.`, 'error');\r\n console.error(error);\r\n }\r\n}\r\nlet fileInput;\r\nfunction getFileOpener() {\r\n if (!fileInput) {\r\n fileInput = document.createElement('input');\r\n fileInput.type = 'file';\r\n fileInput.accept = '.json';\r\n }\r\n function openFile() {\r\n return new Promise((resolve, reject) => {\r\n fileInput.onchange = async () => {\r\n const files = fileInput.files;\r\n if (!files)\r\n return resolve(null);\r\n const file = files.item(0);\r\n if (!file)\r\n return resolve(null);\r\n return resolve({ text: await file.text(), file });\r\n };\r\n // @ts-ignore: TODO: changed from 4.3 to 4.4\r\n fileInput.oncancel = () => resolve(null);\r\n fileInput.onerror = reject;\r\n fileInput.click();\r\n });\r\n }\r\n return openFile;\r\n}\r\nasync function actionGlobalOpenStateFile(pinia) {\r\n try {\r\n const open = await getFileOpener();\r\n const result = await open();\r\n if (!result)\r\n return;\r\n const { text, file } = result;\r\n pinia.state.value = JSON.parse(text);\r\n toastMessage(`Global state imported from \"${file.name}\".`);\r\n }\r\n catch (error) {\r\n toastMessage(`Failed to export the state as JSON. Check the console for more details.`, 'error');\r\n console.error(error);\r\n }\r\n}\n\nfunction formatDisplay(display) {\r\n return {\r\n _custom: {\r\n display,\r\n },\r\n };\r\n}\r\nconst PINIA_ROOT_LABEL = '🍍 Pinia (root)';\r\nconst PINIA_ROOT_ID = '_root';\r\nfunction formatStoreForInspectorTree(store) {\r\n return isPinia(store)\r\n ? {\r\n id: PINIA_ROOT_ID,\r\n label: PINIA_ROOT_LABEL,\r\n }\r\n : {\r\n id: store.$id,\r\n label: store.$id,\r\n };\r\n}\r\nfunction formatStoreForInspectorState(store) {\r\n if (isPinia(store)) {\r\n const storeNames = Array.from(store._s.keys());\r\n const storeMap = store._s;\r\n const state = {\r\n state: storeNames.map((storeId) => ({\r\n editable: true,\r\n key: storeId,\r\n value: store.state.value[storeId],\r\n })),\r\n getters: storeNames\r\n .filter((id) => storeMap.get(id)._getters)\r\n .map((id) => {\r\n const store = storeMap.get(id);\r\n return {\r\n editable: false,\r\n key: id,\r\n value: store._getters.reduce((getters, key) => {\r\n getters[key] = store[key];\r\n return getters;\r\n }, {}),\r\n };\r\n }),\r\n };\r\n return state;\r\n }\r\n const state = {\r\n state: Object.keys(store.$state).map((key) => ({\r\n editable: true,\r\n key,\r\n value: store.$state[key],\r\n })),\r\n };\r\n // avoid adding empty getters\r\n if (store._getters && store._getters.length) {\r\n state.getters = store._getters.map((getterName) => ({\r\n editable: false,\r\n key: getterName,\r\n value: store[getterName],\r\n }));\r\n }\r\n if (store._customProperties.size) {\r\n state.customProperties = Array.from(store._customProperties).map((key) => ({\r\n editable: true,\r\n key,\r\n value: store[key],\r\n }));\r\n }\r\n return state;\r\n}\r\nfunction formatEventData(events) {\r\n if (!events)\r\n return {};\r\n if (Array.isArray(events)) {\r\n // TODO: handle add and delete for arrays and objects\r\n return events.reduce((data, event) => {\r\n data.keys.push(event.key);\r\n data.operations.push(event.type);\r\n data.oldValue[event.key] = event.oldValue;\r\n data.newValue[event.key] = event.newValue;\r\n return data;\r\n }, {\r\n oldValue: {},\r\n keys: [],\r\n operations: [],\r\n newValue: {},\r\n });\r\n }\r\n else {\r\n return {\r\n operation: formatDisplay(events.type),\r\n key: formatDisplay(events.key),\r\n oldValue: events.oldValue,\r\n newValue: events.newValue,\r\n };\r\n }\r\n}\r\nfunction formatMutationType(type) {\r\n switch (type) {\r\n case MutationType.direct:\r\n return 'mutation';\r\n case MutationType.patchFunction:\r\n return '$patch';\r\n case MutationType.patchObject:\r\n return '$patch';\r\n default:\r\n return 'unknown';\r\n }\r\n}\n\n// timeline can be paused when directly changing the state\r\nlet isTimelineActive = true;\r\nconst componentStateTypes = [];\r\nconst MUTATIONS_LAYER_ID = 'pinia:mutations';\r\nconst INSPECTOR_ID = 'pinia';\r\n/**\r\n * Gets the displayed name of a store in devtools\r\n *\r\n * @param id - id of the store\r\n * @returns a formatted string\r\n */\r\nconst getStoreType = (id) => '🍍 ' + id;\r\n/**\r\n * Add the pinia plugin without any store. Allows displaying a Pinia plugin tab\r\n * as soon as it is added to the application.\r\n *\r\n * @param app - Vue application\r\n * @param pinia - pinia instance\r\n */\r\nfunction registerPiniaDevtools(app, pinia) {\r\n setupDevtoolsPlugin({\r\n id: 'dev.esm.pinia',\r\n label: 'Pinia 🍍',\r\n logo: 'https://pinia.vuejs.org/logo.svg',\r\n packageName: 'pinia',\r\n homepage: 'https://pinia.vuejs.org',\r\n componentStateTypes,\r\n app,\r\n }, (api) => {\r\n if (typeof api.now !== 'function') {\r\n toastMessage('You seem to be using an outdated version of Vue Devtools. Are you still using the Beta release instead of the stable one? You can find the links at https://devtools.vuejs.org/guide/installation.html.');\r\n }\r\n api.addTimelineLayer({\r\n id: MUTATIONS_LAYER_ID,\r\n label: `Pinia 🍍`,\r\n color: 0xe5df88,\r\n });\r\n api.addInspector({\r\n id: INSPECTOR_ID,\r\n label: 'Pinia 🍍',\r\n icon: 'storage',\r\n treeFilterPlaceholder: 'Search stores',\r\n actions: [\r\n {\r\n icon: 'content_copy',\r\n action: () => {\r\n actionGlobalCopyState(pinia);\r\n },\r\n tooltip: 'Serialize and copy the state',\r\n },\r\n {\r\n icon: 'content_paste',\r\n action: async () => {\r\n await actionGlobalPasteState(pinia);\r\n api.sendInspectorTree(INSPECTOR_ID);\r\n api.sendInspectorState(INSPECTOR_ID);\r\n },\r\n tooltip: 'Replace the state with the content of your clipboard',\r\n },\r\n {\r\n icon: 'save',\r\n action: () => {\r\n actionGlobalSaveState(pinia);\r\n },\r\n tooltip: 'Save the state as a JSON file',\r\n },\r\n {\r\n icon: 'folder_open',\r\n action: async () => {\r\n await actionGlobalOpenStateFile(pinia);\r\n api.sendInspectorTree(INSPECTOR_ID);\r\n api.sendInspectorState(INSPECTOR_ID);\r\n },\r\n tooltip: 'Import the state from a JSON file',\r\n },\r\n ],\r\n nodeActions: [\r\n {\r\n icon: 'restore',\r\n tooltip: 'Reset the state (option store only)',\r\n action: (nodeId) => {\r\n const store = pinia._s.get(nodeId);\r\n if (!store) {\r\n toastMessage(`Cannot reset \"${nodeId}\" store because it wasn't found.`, 'warn');\r\n }\r\n else if (!store._isOptionsAPI) {\r\n toastMessage(`Cannot reset \"${nodeId}\" store because it's a setup store.`, 'warn');\r\n }\r\n else {\r\n store.$reset();\r\n toastMessage(`Store \"${nodeId}\" reset.`);\r\n }\r\n },\r\n },\r\n ],\r\n });\r\n api.on.inspectComponent((payload, ctx) => {\r\n const proxy = (payload.componentInstance &&\r\n payload.componentInstance.proxy);\r\n if (proxy && proxy._pStores) {\r\n const piniaStores = payload.componentInstance.proxy._pStores;\r\n Object.values(piniaStores).forEach((store) => {\r\n payload.instanceData.state.push({\r\n type: getStoreType(store.$id),\r\n key: 'state',\r\n editable: true,\r\n value: store._isOptionsAPI\r\n ? {\r\n _custom: {\r\n value: toRaw(store.$state),\r\n actions: [\r\n {\r\n icon: 'restore',\r\n tooltip: 'Reset the state of this store',\r\n action: () => store.$reset(),\r\n },\r\n ],\r\n },\r\n }\r\n : // NOTE: workaround to unwrap transferred refs\r\n Object.keys(store.$state).reduce((state, key) => {\r\n state[key] = store.$state[key];\r\n return state;\r\n }, {}),\r\n });\r\n if (store._getters && store._getters.length) {\r\n payload.instanceData.state.push({\r\n type: getStoreType(store.$id),\r\n key: 'getters',\r\n editable: false,\r\n value: store._getters.reduce((getters, key) => {\r\n try {\r\n getters[key] = store[key];\r\n }\r\n catch (error) {\r\n // @ts-expect-error: we just want to show it in devtools\r\n getters[key] = error;\r\n }\r\n return getters;\r\n }, {}),\r\n });\r\n }\r\n });\r\n }\r\n });\r\n api.on.getInspectorTree((payload) => {\r\n if (payload.app === app && payload.inspectorId === INSPECTOR_ID) {\r\n let stores = [pinia];\r\n stores = stores.concat(Array.from(pinia._s.values()));\r\n payload.rootNodes = (payload.filter\r\n ? stores.filter((store) => '$id' in store\r\n ? store.$id\r\n .toLowerCase()\r\n .includes(payload.filter.toLowerCase())\r\n : PINIA_ROOT_LABEL.toLowerCase().includes(payload.filter.toLowerCase()))\r\n : stores).map(formatStoreForInspectorTree);\r\n }\r\n });\r\n api.on.getInspectorState((payload) => {\r\n if (payload.app === app && payload.inspectorId === INSPECTOR_ID) {\r\n const inspectedStore = payload.nodeId === PINIA_ROOT_ID\r\n ? pinia\r\n : pinia._s.get(payload.nodeId);\r\n if (!inspectedStore) {\r\n // this could be the selected store restored for a different project\r\n // so it's better not to say anything here\r\n return;\r\n }\r\n if (inspectedStore) {\r\n payload.state = formatStoreForInspectorState(inspectedStore);\r\n }\r\n }\r\n });\r\n api.on.editInspectorState((payload, ctx) => {\r\n if (payload.app === app && payload.inspectorId === INSPECTOR_ID) {\r\n const inspectedStore = payload.nodeId === PINIA_ROOT_ID\r\n ? pinia\r\n : pinia._s.get(payload.nodeId);\r\n if (!inspectedStore) {\r\n return toastMessage(`store \"${payload.nodeId}\" not found`, 'error');\r\n }\r\n const { path } = payload;\r\n if (!isPinia(inspectedStore)) {\r\n // access only the state\r\n if (path.length !== 1 ||\r\n !inspectedStore._customProperties.has(path[0]) ||\r\n path[0] in inspectedStore.$state) {\r\n path.unshift('$state');\r\n }\r\n }\r\n else {\r\n // Root access, we can omit the `.value` because the devtools API does it for us\r\n path.unshift('state');\r\n }\r\n isTimelineActive = false;\r\n payload.set(inspectedStore, path, payload.state.value);\r\n isTimelineActive = true;\r\n }\r\n });\r\n api.on.editComponentState((payload) => {\r\n if (payload.type.startsWith('🍍')) {\r\n const storeId = payload.type.replace(/^🍍\\s*/, '');\r\n const store = pinia._s.get(storeId);\r\n if (!store) {\r\n return toastMessage(`store \"${storeId}\" not found`, 'error');\r\n }\r\n const { path } = payload;\r\n if (path[0] !== 'state') {\r\n return toastMessage(`Invalid path for store \"${storeId}\":\\n${path}\\nOnly state can be modified.`);\r\n }\r\n // rewrite the first entry to be able to directly set the state as\r\n // well as any other path\r\n path[0] = '$state';\r\n isTimelineActive = false;\r\n payload.set(store, path, payload.state.value);\r\n isTimelineActive = true;\r\n }\r\n });\r\n });\r\n}\r\nfunction addStoreToDevtools(app, store) {\r\n if (!componentStateTypes.includes(getStoreType(store.$id))) {\r\n componentStateTypes.push(getStoreType(store.$id));\r\n }\r\n setupDevtoolsPlugin({\r\n id: 'dev.esm.pinia',\r\n label: 'Pinia 🍍',\r\n logo: 'https://pinia.vuejs.org/logo.svg',\r\n packageName: 'pinia',\r\n homepage: 'https://pinia.vuejs.org',\r\n componentStateTypes,\r\n app,\r\n settings: {\r\n logStoreChanges: {\r\n label: 'Notify about new/deleted stores',\r\n type: 'boolean',\r\n defaultValue: true,\r\n },\r\n // useEmojis: {\r\n // label: 'Use emojis in messages ⚡️',\r\n // type: 'boolean',\r\n // defaultValue: true,\r\n // },\r\n },\r\n }, (api) => {\r\n // gracefully handle errors\r\n const now = typeof api.now === 'function' ? api.now.bind(api) : Date.now;\r\n store.$onAction(({ after, onError, name, args }) => {\r\n const groupId = runningActionId++;\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: {\r\n time: now(),\r\n title: '🛫 ' + name,\r\n subtitle: 'start',\r\n data: {\r\n store: formatDisplay(store.$id),\r\n action: formatDisplay(name),\r\n args,\r\n },\r\n groupId,\r\n },\r\n });\r\n after((result) => {\r\n activeAction = undefined;\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: {\r\n time: now(),\r\n title: '🛬 ' + name,\r\n subtitle: 'end',\r\n data: {\r\n store: formatDisplay(store.$id),\r\n action: formatDisplay(name),\r\n args,\r\n result,\r\n },\r\n groupId,\r\n },\r\n });\r\n });\r\n onError((error) => {\r\n activeAction = undefined;\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: {\r\n time: now(),\r\n logType: 'error',\r\n title: '💥 ' + name,\r\n subtitle: 'end',\r\n data: {\r\n store: formatDisplay(store.$id),\r\n action: formatDisplay(name),\r\n args,\r\n error,\r\n },\r\n groupId,\r\n },\r\n });\r\n });\r\n }, true);\r\n store._customProperties.forEach((name) => {\r\n watch(() => unref(store[name]), (newValue, oldValue) => {\r\n api.notifyComponentUpdate();\r\n api.sendInspectorState(INSPECTOR_ID);\r\n if (isTimelineActive) {\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: {\r\n time: now(),\r\n title: 'Change',\r\n subtitle: name,\r\n data: {\r\n newValue,\r\n oldValue,\r\n },\r\n groupId: activeAction,\r\n },\r\n });\r\n }\r\n }, { deep: true });\r\n });\r\n store.$subscribe(({ events, type }, state) => {\r\n api.notifyComponentUpdate();\r\n api.sendInspectorState(INSPECTOR_ID);\r\n if (!isTimelineActive)\r\n return;\r\n // rootStore.state[store.id] = state\r\n const eventData = {\r\n time: now(),\r\n title: formatMutationType(type),\r\n data: {\r\n store: formatDisplay(store.$id),\r\n ...formatEventData(events),\r\n },\r\n groupId: activeAction,\r\n };\r\n // reset for the next mutation\r\n activeAction = undefined;\r\n if (type === MutationType.patchFunction) {\r\n eventData.subtitle = '⤵️';\r\n }\r\n else if (type === MutationType.patchObject) {\r\n eventData.subtitle = '🧩';\r\n }\r\n else if (events && !Array.isArray(events)) {\r\n eventData.subtitle = events.type;\r\n }\r\n if (events) {\r\n eventData.data['rawEvent(s)'] = {\r\n _custom: {\r\n display: 'DebuggerEvent',\r\n type: 'object',\r\n tooltip: 'raw DebuggerEvent[]',\r\n value: events,\r\n },\r\n };\r\n }\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: eventData,\r\n });\r\n }, { detached: true, flush: 'sync' });\r\n const hotUpdate = store._hotUpdate;\r\n store._hotUpdate = markRaw((newStore) => {\r\n hotUpdate(newStore);\r\n api.addTimelineEvent({\r\n layerId: MUTATIONS_LAYER_ID,\r\n event: {\r\n time: now(),\r\n title: '🔥 ' + store.$id,\r\n subtitle: 'HMR update',\r\n data: {\r\n store: formatDisplay(store.$id),\r\n info: formatDisplay(`HMR update`),\r\n },\r\n },\r\n });\r\n // update the devtools too\r\n api.notifyComponentUpdate();\r\n api.sendInspectorTree(INSPECTOR_ID);\r\n api.sendInspectorState(INSPECTOR_ID);\r\n });\r\n const { $dispose } = store;\r\n store.$dispose = () => {\r\n $dispose();\r\n api.notifyComponentUpdate();\r\n api.sendInspectorTree(INSPECTOR_ID);\r\n api.sendInspectorState(INSPECTOR_ID);\r\n api.getSettings().logStoreChanges &&\r\n toastMessage(`Disposed \"${store.$id}\" store 🗑`);\r\n };\r\n // trigger an update so it can display new registered stores\r\n api.notifyComponentUpdate();\r\n api.sendInspectorTree(INSPECTOR_ID);\r\n api.sendInspectorState(INSPECTOR_ID);\r\n api.getSettings().logStoreChanges &&\r\n toastMessage(`\"${store.$id}\" store installed 🆕`);\r\n });\r\n}\r\nlet runningActionId = 0;\r\nlet activeAction;\r\n/**\r\n * Patches a store to enable action grouping in devtools by wrapping the store with a Proxy that is passed as the\r\n * context of all actions, allowing us to set `runningAction` on each access and effectively associating any state\r\n * mutation to the action.\r\n *\r\n * @param store - store to patch\r\n * @param actionNames - list of actionst to patch\r\n */\r\nfunction patchActionForGrouping(store, actionNames) {\r\n // original actions of the store as they are given by pinia. We are going to override them\r\n const actions = actionNames.reduce((storeActions, actionName) => {\r\n // use toRaw to avoid tracking #541\r\n storeActions[actionName] = toRaw(store)[actionName];\r\n return storeActions;\r\n }, {});\r\n for (const actionName in actions) {\r\n store[actionName] = function () {\r\n // setActivePinia(store._p)\r\n // the running action id is incremented in a before action hook\r\n const _actionId = runningActionId;\r\n const trackedStore = new Proxy(store, {\r\n get(...args) {\r\n activeAction = _actionId;\r\n return Reflect.get(...args);\r\n },\r\n set(...args) {\r\n activeAction = _actionId;\r\n return Reflect.set(...args);\r\n },\r\n });\r\n return actions[actionName].apply(trackedStore, arguments);\r\n };\r\n }\r\n}\r\n/**\r\n * pinia.use(devtoolsPlugin)\r\n */\r\nfunction devtoolsPlugin({ app, store, options }) {\r\n // HMR module\r\n if (store.$id.startsWith('__hot:')) {\r\n return;\r\n }\r\n // detect option api vs setup api\r\n if (options.state) {\r\n store._isOptionsAPI = true;\r\n }\r\n // only wrap actions in option-defined stores as this technique relies on\r\n // wrapping the context of the action with a proxy\r\n if (typeof options.state === 'function') {\r\n patchActionForGrouping(\r\n // @ts-expect-error: can cast the store...\r\n store, Object.keys(options.actions));\r\n const originalHotUpdate = store._hotUpdate;\r\n // Upgrade the HMR to also update the new actions\r\n toRaw(store)._hotUpdate = function (newStore) {\r\n originalHotUpdate.apply(this, arguments);\r\n patchActionForGrouping(store, Object.keys(newStore._hmrPayload.actions));\r\n };\r\n }\r\n addStoreToDevtools(app, \r\n // FIXME: is there a way to allow the assignment from Store to StoreGeneric?\r\n store);\r\n}\n\n/**\r\n * Creates a Pinia instance to be used by the application\r\n */\r\nfunction createPinia() {\r\n const scope = effectScope(true);\r\n // NOTE: here we could check the window object for a state and directly set it\r\n // if there is anything like it with Vue 3 SSR\r\n const state = scope.run(() => ref({}));\r\n let _p = [];\r\n // plugins added before calling app.use(pinia)\r\n let toBeInstalled = [];\r\n const pinia = markRaw({\r\n install(app) {\r\n // this allows calling useStore() outside of a component setup after\r\n // installing pinia's plugin\r\n setActivePinia(pinia);\r\n if (!isVue2) {\r\n pinia._a = app;\r\n app.provide(piniaSymbol, pinia);\r\n app.config.globalProperties.$pinia = pinia;\r\n /* istanbul ignore else */\r\n if (USE_DEVTOOLS) {\r\n registerPiniaDevtools(app, pinia);\r\n }\r\n toBeInstalled.forEach((plugin) => _p.push(plugin));\r\n toBeInstalled = [];\r\n }\r\n },\r\n use(plugin) {\r\n if (!this._a && !isVue2) {\r\n toBeInstalled.push(plugin);\r\n }\r\n else {\r\n _p.push(plugin);\r\n }\r\n return this;\r\n },\r\n _p,\r\n // it's actually undefined here\r\n // @ts-expect-error\r\n _a: null,\r\n _e: scope,\r\n _s: new Map(),\r\n state,\r\n });\r\n // pinia devtools rely on dev only features so they cannot be forced unless\r\n // the dev build of Vue is used. Avoid old browsers like IE11.\r\n if (USE_DEVTOOLS && typeof Proxy !== 'undefined') {\r\n pinia.use(devtoolsPlugin);\r\n }\r\n return pinia;\r\n}\n\n/**\r\n * Checks if a function is a `StoreDefinition`.\r\n *\r\n * @param fn - object to test\r\n * @returns true if `fn` is a StoreDefinition\r\n */\r\nconst isUseStore = (fn) => {\r\n return typeof fn === 'function' && typeof fn.$id === 'string';\r\n};\r\n/**\r\n * Mutates in place `newState` with `oldState` to _hot update_ it. It will\r\n * remove any key not existing in `newState` and recursively merge plain\r\n * objects.\r\n *\r\n * @param newState - new state object to be patched\r\n * @param oldState - old state that should be used to patch newState\r\n * @returns - newState\r\n */\r\nfunction patchObject(newState, oldState) {\r\n // no need to go through symbols because they cannot be serialized anyway\r\n for (const key in oldState) {\r\n const subPatch = oldState[key];\r\n // skip the whole sub tree\r\n if (!(key in newState)) {\r\n continue;\r\n }\r\n const targetValue = newState[key];\r\n if (isPlainObject(targetValue) &&\r\n isPlainObject(subPatch) &&\r\n !isRef(subPatch) &&\r\n !isReactive(subPatch)) {\r\n newState[key] = patchObject(targetValue, subPatch);\r\n }\r\n else {\r\n // objects are either a bit more complex (e.g. refs) or primitives, so we\r\n // just set the whole thing\r\n if (isVue2) {\r\n set(newState, key, subPatch);\r\n }\r\n else {\r\n newState[key] = subPatch;\r\n }\r\n }\r\n }\r\n return newState;\r\n}\r\n/**\r\n * Creates an _accept_ function to pass to `import.meta.hot` in Vite applications.\r\n *\r\n * @example\r\n * ```js\r\n * const useUser = defineStore(...)\r\n * if (import.meta.hot) {\r\n * import.meta.hot.accept(acceptHMRUpdate(useUser, import.meta.hot))\r\n * }\r\n * ```\r\n *\r\n * @param initialUseStore - return of the defineStore to hot update\r\n * @param hot - `import.meta.hot`\r\n */\r\nfunction acceptHMRUpdate(initialUseStore, hot) {\r\n // strip as much as possible from iife.prod\r\n if (!(process.env.NODE_ENV !== 'production')) {\r\n return () => { };\r\n }\r\n return (newModule) => {\r\n const pinia = hot.data.pinia || initialUseStore._pinia;\r\n if (!pinia) {\r\n // this store is still not used\r\n return;\r\n }\r\n // preserve the pinia instance across loads\r\n hot.data.pinia = pinia;\r\n // console.log('got data', newStore)\r\n for (const exportName in newModule) {\r\n const useStore = newModule[exportName];\r\n // console.log('checking for', exportName)\r\n if (isUseStore(useStore) && pinia._s.has(useStore.$id)) {\r\n // console.log('Accepting update for', useStore.$id)\r\n const id = useStore.$id;\r\n if (id !== initialUseStore.$id) {\r\n console.warn(`The id of the store changed from \"${initialUseStore.$id}\" to \"${id}\". Reloading.`);\r\n // return import.meta.hot.invalidate()\r\n return hot.invalidate();\r\n }\r\n const existingStore = pinia._s.get(id);\r\n if (!existingStore) {\r\n console.log(`[Pinia]: skipping hmr because store doesn't exist yet`);\r\n return;\r\n }\r\n useStore(pinia, existingStore);\r\n }\r\n }\r\n };\r\n}\n\nconst noop = () => { };\r\nfunction addSubscription(subscriptions, callback, detached, onCleanup = noop) {\r\n subscriptions.push(callback);\r\n const removeSubscription = () => {\r\n const idx = subscriptions.indexOf(callback);\r\n if (idx > -1) {\r\n subscriptions.splice(idx, 1);\r\n onCleanup();\r\n }\r\n };\r\n if (!detached && getCurrentInstance()) {\r\n onUnmounted(removeSubscription);\r\n }\r\n return removeSubscription;\r\n}\r\nfunction triggerSubscriptions(subscriptions, ...args) {\r\n subscriptions.slice().forEach((callback) => {\r\n callback(...args);\r\n });\r\n}\n\nfunction mergeReactiveObjects(target, patchToApply) {\r\n // Handle Map instances\r\n if (target instanceof Map && patchToApply instanceof Map) {\r\n patchToApply.forEach((value, key) => target.set(key, value));\r\n }\r\n // Handle Set instances\r\n if (target instanceof Set && patchToApply instanceof Set) {\r\n patchToApply.forEach(target.add, target);\r\n }\r\n // no need to go through symbols because they cannot be serialized anyway\r\n for (const key in patchToApply) {\r\n if (!patchToApply.hasOwnProperty(key))\r\n continue;\r\n const subPatch = patchToApply[key];\r\n const targetValue = target[key];\r\n if (isPlainObject(targetValue) &&\r\n isPlainObject(subPatch) &&\r\n target.hasOwnProperty(key) &&\r\n !isRef(subPatch) &&\r\n !isReactive(subPatch)) {\r\n // NOTE: here I wanted to warn about inconsistent types but it's not possible because in setup stores one might\r\n // start the value of a property as a certain type e.g. a Map, and then for some reason, during SSR, change that\r\n // to `undefined`. When trying to hydrate, we want to override the Map with `undefined`.\r\n target[key] = mergeReactiveObjects(targetValue, subPatch);\r\n }\r\n else {\r\n // @ts-expect-error: subPatch is a valid value\r\n target[key] = subPatch;\r\n }\r\n }\r\n return target;\r\n}\r\nconst skipHydrateSymbol = (process.env.NODE_ENV !== 'production')\r\n ? Symbol('pinia:skipHydration')\r\n : /* istanbul ignore next */ Symbol();\r\nconst skipHydrateMap = /*#__PURE__*/ new WeakMap();\r\n/**\r\n * Tells Pinia to skip the hydration process of a given object. This is useful in setup stores (only) when you return a\r\n * stateful object in the store but it isn't really state. e.g. returning a router instance in a setup store.\r\n *\r\n * @param obj - target object\r\n * @returns obj\r\n */\r\nfunction skipHydrate(obj) {\r\n return isVue2\r\n ? // in @vue/composition-api, the refs are sealed so defineProperty doesn't work...\r\n /* istanbul ignore next */ skipHydrateMap.set(obj, 1) && obj\r\n : Object.defineProperty(obj, skipHydrateSymbol, {});\r\n}\r\nfunction shouldHydrate(obj) {\r\n return isVue2\r\n ? /* istanbul ignore next */ !skipHydrateMap.has(obj)\r\n : !isPlainObject(obj) || !obj.hasOwnProperty(skipHydrateSymbol);\r\n}\r\nconst { assign } = Object;\r\nfunction isComputed(o) {\r\n return !!(isRef(o) && o.effect);\r\n}\r\nfunction createOptionsStore(id, options, pinia, hot) {\r\n const { state, actions, getters } = options;\r\n const initialState = pinia.state.value[id];\r\n let store;\r\n function setup() {\r\n if (!initialState && (!(process.env.NODE_ENV !== 'production') || !hot)) {\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n set(pinia.state.value, id, state ? state() : {});\r\n }\r\n else {\r\n pinia.state.value[id] = state ? state() : {};\r\n }\r\n }\r\n // avoid creating a state in pinia.state.value\r\n const localState = (process.env.NODE_ENV !== 'production') && hot\r\n ? // use ref() to unwrap refs inside state TODO: check if this is still necessary\r\n toRefs(ref(state ? state() : {}).value)\r\n : toRefs(pinia.state.value[id]);\r\n return assign(localState, actions, Object.keys(getters || {}).reduce((computedGetters, name) => {\r\n if ((process.env.NODE_ENV !== 'production') && name in localState) {\r\n console.warn(`[🍍]: A getter cannot have the same name as another state property. Rename one of them. Found with \"${name}\" in store \"${id}\".`);\r\n }\r\n computedGetters[name] = markRaw(computed(() => {\r\n setActivePinia(pinia);\r\n // it was created just before\r\n const store = pinia._s.get(id);\r\n // allow cross using stores\r\n /* istanbul ignore next */\r\n if (isVue2 && !store._r)\r\n return;\r\n // @ts-expect-error\r\n // return getters![name].call(context, context)\r\n // TODO: avoid reading the getter while assigning with a global variable\r\n return getters[name].call(store, store);\r\n }));\r\n return computedGetters;\r\n }, {}));\r\n }\r\n store = createSetupStore(id, setup, options, pinia, hot, true);\r\n store.$reset = function $reset() {\r\n const newState = state ? state() : {};\r\n // we use a patch to group all changes into one single subscription\r\n this.$patch(($state) => {\r\n assign($state, newState);\r\n });\r\n };\r\n return store;\r\n}\r\nfunction createSetupStore($id, setup, options = {}, pinia, hot, isOptionsStore) {\r\n let scope;\r\n const optionsForPlugin = assign({ actions: {} }, options);\r\n /* istanbul ignore if */\r\n // @ts-expect-error: active is an internal property\r\n if ((process.env.NODE_ENV !== 'production') && !pinia._e.active) {\r\n throw new Error('Pinia destroyed');\r\n }\r\n // watcher options for $subscribe\r\n const $subscribeOptions = {\r\n deep: true,\r\n // flush: 'post',\r\n };\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production') && !isVue2) {\r\n $subscribeOptions.onTrigger = (event) => {\r\n /* istanbul ignore else */\r\n if (isListening) {\r\n debuggerEvents = event;\r\n // avoid triggering this while the store is being built and the state is being set in pinia\r\n }\r\n else if (isListening == false && !store._hotUpdating) {\r\n // let patch send all the events together later\r\n /* istanbul ignore else */\r\n if (Array.isArray(debuggerEvents)) {\r\n debuggerEvents.push(event);\r\n }\r\n else {\r\n console.error('🍍 debuggerEvents should be an array. This is most likely an internal Pinia bug.');\r\n }\r\n }\r\n };\r\n }\r\n // internal state\r\n let isListening; // set to true at the end\r\n let isSyncListening; // set to true at the end\r\n let subscriptions = markRaw([]);\r\n let actionSubscriptions = markRaw([]);\r\n let debuggerEvents;\r\n const initialState = pinia.state.value[$id];\r\n // avoid setting the state for option stores if it is set\r\n // by the setup\r\n if (!isOptionsStore && !initialState && (!(process.env.NODE_ENV !== 'production') || !hot)) {\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n set(pinia.state.value, $id, {});\r\n }\r\n else {\r\n pinia.state.value[$id] = {};\r\n }\r\n }\r\n const hotState = ref({});\r\n // avoid triggering too many listeners\r\n // https://github.com/vuejs/pinia/issues/1129\r\n let activeListener;\r\n function $patch(partialStateOrMutator) {\r\n let subscriptionMutation;\r\n isListening = isSyncListening = false;\r\n // reset the debugger events since patches are sync\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production')) {\r\n debuggerEvents = [];\r\n }\r\n if (typeof partialStateOrMutator === 'function') {\r\n partialStateOrMutator(pinia.state.value[$id]);\r\n subscriptionMutation = {\r\n type: MutationType.patchFunction,\r\n storeId: $id,\r\n events: debuggerEvents,\r\n };\r\n }\r\n else {\r\n mergeReactiveObjects(pinia.state.value[$id], partialStateOrMutator);\r\n subscriptionMutation = {\r\n type: MutationType.patchObject,\r\n payload: partialStateOrMutator,\r\n storeId: $id,\r\n events: debuggerEvents,\r\n };\r\n }\r\n const myListenerId = (activeListener = Symbol());\r\n nextTick().then(() => {\r\n if (activeListener === myListenerId) {\r\n isListening = true;\r\n }\r\n });\r\n isSyncListening = true;\r\n // because we paused the watcher, we need to manually call the subscriptions\r\n triggerSubscriptions(subscriptions, subscriptionMutation, pinia.state.value[$id]);\r\n }\r\n /* istanbul ignore next */\r\n const $reset = (process.env.NODE_ENV !== 'production')\r\n ? () => {\r\n throw new Error(`🍍: Store \"${$id}\" is built using the setup syntax and does not implement $reset().`);\r\n }\r\n : noop;\r\n function $dispose() {\r\n scope.stop();\r\n subscriptions = [];\r\n actionSubscriptions = [];\r\n pinia._s.delete($id);\r\n }\r\n /**\r\n * Wraps an action to handle subscriptions.\r\n *\r\n * @param name - name of the action\r\n * @param action - action to wrap\r\n * @returns a wrapped action to handle subscriptions\r\n */\r\n function wrapAction(name, action) {\r\n return function () {\r\n setActivePinia(pinia);\r\n const args = Array.from(arguments);\r\n const afterCallbackList = [];\r\n const onErrorCallbackList = [];\r\n function after(callback) {\r\n afterCallbackList.push(callback);\r\n }\r\n function onError(callback) {\r\n onErrorCallbackList.push(callback);\r\n }\r\n // @ts-expect-error\r\n triggerSubscriptions(actionSubscriptions, {\r\n args,\r\n name,\r\n store,\r\n after,\r\n onError,\r\n });\r\n let ret;\r\n try {\r\n ret = action.apply(this && this.$id === $id ? this : store, args);\r\n // handle sync errors\r\n }\r\n catch (error) {\r\n triggerSubscriptions(onErrorCallbackList, error);\r\n throw error;\r\n }\r\n if (ret instanceof Promise) {\r\n return ret\r\n .then((value) => {\r\n triggerSubscriptions(afterCallbackList, value);\r\n return value;\r\n })\r\n .catch((error) => {\r\n triggerSubscriptions(onErrorCallbackList, error);\r\n return Promise.reject(error);\r\n });\r\n }\r\n // allow the afterCallback to override the return value\r\n triggerSubscriptions(afterCallbackList, ret);\r\n return ret;\r\n };\r\n }\r\n const _hmrPayload = /*#__PURE__*/ markRaw({\r\n actions: {},\r\n getters: {},\r\n state: [],\r\n hotState,\r\n });\r\n const partialStore = {\r\n _p: pinia,\r\n // _s: scope,\r\n $id,\r\n $onAction: addSubscription.bind(null, actionSubscriptions),\r\n $patch,\r\n $reset,\r\n $subscribe(callback, options = {}) {\r\n const removeSubscription = addSubscription(subscriptions, callback, options.detached, () => stopWatcher());\r\n const stopWatcher = scope.run(() => watch(() => pinia.state.value[$id], (state) => {\r\n if (options.flush === 'sync' ? isSyncListening : isListening) {\r\n callback({\r\n storeId: $id,\r\n type: MutationType.direct,\r\n events: debuggerEvents,\r\n }, state);\r\n }\r\n }, assign({}, $subscribeOptions, options)));\r\n return removeSubscription;\r\n },\r\n $dispose,\r\n };\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n // start as non ready\r\n partialStore._r = false;\r\n }\r\n const store = reactive(assign((process.env.NODE_ENV !== 'production') && IS_CLIENT\r\n ? // devtools custom properties\r\n {\r\n _customProperties: markRaw(new Set()),\r\n _hmrPayload,\r\n }\r\n : {}, partialStore\r\n // must be added later\r\n // setupStore\r\n ));\r\n // store the partial store now so the setup of stores can instantiate each other before they are finished without\r\n // creating infinite loops.\r\n pinia._s.set($id, store);\r\n // TODO: idea create skipSerialize that marks properties as non serializable and they are skipped\r\n const setupStore = pinia._e.run(() => {\r\n scope = effectScope();\r\n return scope.run(() => setup());\r\n });\r\n // overwrite existing actions to support $onAction\r\n for (const key in setupStore) {\r\n const prop = setupStore[key];\r\n if ((isRef(prop) && !isComputed(prop)) || isReactive(prop)) {\r\n // mark it as a piece of state to be serialized\r\n if ((process.env.NODE_ENV !== 'production') && hot) {\r\n set(hotState.value, key, toRef(setupStore, key));\r\n // createOptionStore directly sets the state in pinia.state.value so we\r\n // can just skip that\r\n }\r\n else if (!isOptionsStore) {\r\n // in setup stores we must hydrate the state and sync pinia state tree with the refs the user just created\r\n if (initialState && shouldHydrate(prop)) {\r\n if (isRef(prop)) {\r\n prop.value = initialState[key];\r\n }\r\n else {\r\n // probably a reactive object, lets recursively assign\r\n mergeReactiveObjects(prop, initialState[key]);\r\n }\r\n }\r\n // transfer the ref to the pinia state to keep everything in sync\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n set(pinia.state.value[$id], key, prop);\r\n }\r\n else {\r\n pinia.state.value[$id][key] = prop;\r\n }\r\n }\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production')) {\r\n _hmrPayload.state.push(key);\r\n }\r\n // action\r\n }\r\n else if (typeof prop === 'function') {\r\n // @ts-expect-error: we are overriding the function we avoid wrapping if\r\n const actionValue = (process.env.NODE_ENV !== 'production') && hot ? prop : wrapAction(key, prop);\r\n // this a hot module replacement store because the hotUpdate method needs\r\n // to do it with the right context\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n set(setupStore, key, actionValue);\r\n }\r\n else {\r\n // @ts-expect-error\r\n setupStore[key] = actionValue;\r\n }\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production')) {\r\n _hmrPayload.actions[key] = prop;\r\n }\r\n // list actions so they can be used in plugins\r\n // @ts-expect-error\r\n optionsForPlugin.actions[key] = prop;\r\n }\r\n else if ((process.env.NODE_ENV !== 'production')) {\r\n // add getters for devtools\r\n if (isComputed(prop)) {\r\n _hmrPayload.getters[key] = isOptionsStore\r\n ? // @ts-expect-error\r\n options.getters[key]\r\n : prop;\r\n if (IS_CLIENT) {\r\n const getters = \r\n // @ts-expect-error: it should be on the store\r\n setupStore._getters || (setupStore._getters = markRaw([]));\r\n getters.push(key);\r\n }\r\n }\r\n }\r\n }\r\n // add the state, getters, and action properties\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n Object.keys(setupStore).forEach((key) => {\r\n set(store, key, \r\n // @ts-expect-error: valid key indexing\r\n setupStore[key]);\r\n });\r\n }\r\n else {\r\n assign(store, setupStore);\r\n // allows retrieving reactive objects with `storeToRefs()`. Must be called after assigning to the reactive object.\r\n // Make `storeToRefs()` work with `reactive()` #799\r\n assign(toRaw(store), setupStore);\r\n }\r\n // use this instead of a computed with setter to be able to create it anywhere\r\n // without linking the computed lifespan to wherever the store is first\r\n // created.\r\n Object.defineProperty(store, '$state', {\r\n get: () => ((process.env.NODE_ENV !== 'production') && hot ? hotState.value : pinia.state.value[$id]),\r\n set: (state) => {\r\n /* istanbul ignore if */\r\n if ((process.env.NODE_ENV !== 'production') && hot) {\r\n throw new Error('cannot set hotState');\r\n }\r\n $patch(($state) => {\r\n assign($state, state);\r\n });\r\n },\r\n });\r\n // add the hotUpdate before plugins to allow them to override it\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production')) {\r\n store._hotUpdate = markRaw((newStore) => {\r\n store._hotUpdating = true;\r\n newStore._hmrPayload.state.forEach((stateKey) => {\r\n if (stateKey in store.$state) {\r\n const newStateTarget = newStore.$state[stateKey];\r\n const oldStateSource = store.$state[stateKey];\r\n if (typeof newStateTarget === 'object' &&\r\n isPlainObject(newStateTarget) &&\r\n isPlainObject(oldStateSource)) {\r\n patchObject(newStateTarget, oldStateSource);\r\n }\r\n else {\r\n // transfer the ref\r\n newStore.$state[stateKey] = oldStateSource;\r\n }\r\n }\r\n // patch direct access properties to allow store.stateProperty to work as\r\n // store.$state.stateProperty\r\n set(store, stateKey, toRef(newStore.$state, stateKey));\r\n });\r\n // remove deleted state properties\r\n Object.keys(store.$state).forEach((stateKey) => {\r\n if (!(stateKey in newStore.$state)) {\r\n del(store, stateKey);\r\n }\r\n });\r\n // avoid devtools logging this as a mutation\r\n isListening = false;\r\n isSyncListening = false;\r\n pinia.state.value[$id] = toRef(newStore._hmrPayload, 'hotState');\r\n isSyncListening = true;\r\n nextTick().then(() => {\r\n isListening = true;\r\n });\r\n for (const actionName in newStore._hmrPayload.actions) {\r\n const action = newStore[actionName];\r\n set(store, actionName, wrapAction(actionName, action));\r\n }\r\n // TODO: does this work in both setup and option store?\r\n for (const getterName in newStore._hmrPayload.getters) {\r\n const getter = newStore._hmrPayload.getters[getterName];\r\n const getterValue = isOptionsStore\r\n ? // special handling of options api\r\n computed(() => {\r\n setActivePinia(pinia);\r\n return getter.call(store, store);\r\n })\r\n : getter;\r\n set(store, getterName, getterValue);\r\n }\r\n // remove deleted getters\r\n Object.keys(store._hmrPayload.getters).forEach((key) => {\r\n if (!(key in newStore._hmrPayload.getters)) {\r\n del(store, key);\r\n }\r\n });\r\n // remove old actions\r\n Object.keys(store._hmrPayload.actions).forEach((key) => {\r\n if (!(key in newStore._hmrPayload.actions)) {\r\n del(store, key);\r\n }\r\n });\r\n // update the values used in devtools and to allow deleting new properties later on\r\n store._hmrPayload = newStore._hmrPayload;\r\n store._getters = newStore._getters;\r\n store._hotUpdating = false;\r\n });\r\n const nonEnumerable = {\r\n writable: true,\r\n configurable: true,\r\n // avoid warning on devtools trying to display this property\r\n enumerable: false,\r\n };\r\n if (IS_CLIENT) {\r\n ['_p', '_hmrPayload', '_getters', '_customProperties'].forEach((p) => {\r\n Object.defineProperty(store, p, {\r\n value: store[p],\r\n ...nonEnumerable,\r\n });\r\n });\r\n }\r\n }\r\n /* istanbul ignore if */\r\n if (isVue2) {\r\n // mark the store as ready before plugins\r\n store._r = true;\r\n }\r\n // apply all plugins\r\n pinia._p.forEach((extender) => {\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production') && IS_CLIENT) {\r\n const extensions = scope.run(() => extender({\r\n store,\r\n app: pinia._a,\r\n pinia,\r\n options: optionsForPlugin,\r\n }));\r\n Object.keys(extensions || {}).forEach((key) => store._customProperties.add(key));\r\n assign(store, extensions);\r\n }\r\n else {\r\n assign(store, scope.run(() => extender({\r\n store,\r\n app: pinia._a,\r\n pinia,\r\n options: optionsForPlugin,\r\n })));\r\n }\r\n });\r\n if ((process.env.NODE_ENV !== 'production') &&\r\n store.$state &&\r\n typeof store.$state === 'object' &&\r\n typeof store.$state.constructor === 'function' &&\r\n !store.$state.constructor.toString().includes('[native code]')) {\r\n console.warn(`[🍍]: The \"state\" must be a plain object. It cannot be\\n` +\r\n `\\tstate: () => new MyClass()\\n` +\r\n `Found in store \"${store.$id}\".`);\r\n }\r\n // only apply hydrate to option stores with an initial state in pinia\r\n if (initialState &&\r\n isOptionsStore &&\r\n options.hydrate) {\r\n options.hydrate(store.$state, initialState);\r\n }\r\n isListening = true;\r\n isSyncListening = true;\r\n return store;\r\n}\r\nfunction defineStore(\r\n// TODO: add proper types from above\r\nidOrOptions, setup, setupOptions) {\r\n let id;\r\n let options;\r\n const isSetupStore = typeof setup === 'function';\r\n if (typeof idOrOptions === 'string') {\r\n id = idOrOptions;\r\n // the option store setup will contain the actual options in this case\r\n options = isSetupStore ? setupOptions : setup;\r\n }\r\n else {\r\n options = idOrOptions;\r\n id = idOrOptions.id;\r\n }\r\n function useStore(pinia, hot) {\r\n const currentInstance = getCurrentInstance();\r\n pinia =\r\n // in test mode, ignore the argument provided as we can always retrieve a\r\n // pinia instance with getActivePinia()\r\n ((process.env.NODE_ENV === 'test') && activePinia && activePinia._testing ? null : pinia) ||\r\n (currentInstance && inject(piniaSymbol));\r\n if (pinia)\r\n setActivePinia(pinia);\r\n if ((process.env.NODE_ENV !== 'production') && !activePinia) {\r\n throw new Error(`[🍍]: getActivePinia was called with no active Pinia. Did you forget to install pinia?\\n` +\r\n `\\tconst pinia = createPinia()\\n` +\r\n `\\tapp.use(pinia)\\n` +\r\n `This will fail in production.`);\r\n }\r\n pinia = activePinia;\r\n if (!pinia._s.has(id)) {\r\n // creating the store registers it in `pinia._s`\r\n if (isSetupStore) {\r\n createSetupStore(id, setup, options, pinia);\r\n }\r\n else {\r\n createOptionsStore(id, options, pinia);\r\n }\r\n /* istanbul ignore else */\r\n if ((process.env.NODE_ENV !== 'production')) {\r\n // @ts-expect-error: not the right inferred type\r\n useStore._pinia = pinia;\r\n }\r\n }\r\n const store = pinia._s.get(id);\r\n if ((process.env.NODE_ENV !== 'production') && hot) {\r\n const hotId = '__hot:' + id;\r\n const newStore = isSetupStore\r\n ? createSetupStore(hotId, setup, options, pinia, true)\r\n : createOptionsStore(hotId, assign({}, options), pinia, true);\r\n hot._hotUpdate(newStore);\r\n // cleanup the state properties and the store from the cache\r\n delete pinia.state.value[hotId];\r\n pinia._s.delete(hotId);\r\n }\r\n // save stores in instances to access them devtools\r\n if ((process.env.NODE_ENV !== 'production') &&\r\n IS_CLIENT &&\r\n currentInstance &&\r\n currentInstance.proxy &&\r\n // avoid adding stores that are just built for hot module replacement\r\n !hot) {\r\n const vm = currentInstance.proxy;\r\n const cache = '_pStores' in vm ? vm._pStores : (vm._pStores = {});\r\n cache[id] = store;\r\n }\r\n // StoreGeneric cannot be casted towards Store\r\n return store;\r\n }\r\n useStore.$id = id;\r\n return useStore;\r\n}\n\nlet mapStoreSuffix = 'Store';\r\n/**\r\n * Changes the suffix added by `mapStores()`. Can be set to an empty string.\r\n * Defaults to `\"Store\"`. Make sure to extend the MapStoresCustomization\r\n * interface if you are using TypeScript.\r\n *\r\n * @param suffix - new suffix\r\n */\r\nfunction setMapStoreSuffix(suffix // could be 'Store' but that would be annoying for JS\r\n) {\r\n mapStoreSuffix = suffix;\r\n}\r\n/**\r\n * Allows using stores without the composition API (`setup()`) by generating an\r\n * object to be spread in the `computed` field of a component. It accepts a list\r\n * of store definitions.\r\n *\r\n * @example\r\n * ```js\r\n * export default {\r\n * computed: {\r\n * // other computed properties\r\n * ...mapStores(useUserStore, useCartStore)\r\n * },\r\n *\r\n * created() {\r\n * this.userStore // store with id \"user\"\r\n * this.cartStore // store with id \"cart\"\r\n * }\r\n * }\r\n * ```\r\n *\r\n * @param stores - list of stores to map to an object\r\n */\r\nfunction mapStores(...stores) {\r\n if ((process.env.NODE_ENV !== 'production') && Array.isArray(stores[0])) {\r\n console.warn(`[🍍]: Directly pass all stores to \"mapStores()\" without putting them in an array:\\n` +\r\n `Replace\\n` +\r\n `\\tmapStores([useAuthStore, useCartStore])\\n` +\r\n `with\\n` +\r\n `\\tmapStores(useAuthStore, useCartStore)\\n` +\r\n `This will fail in production if not fixed.`);\r\n stores = stores[0];\r\n }\r\n return stores.reduce((reduced, useStore) => {\r\n // @ts-expect-error: $id is added by defineStore\r\n reduced[useStore.$id + mapStoreSuffix] = function () {\r\n return useStore(this.$pinia);\r\n };\r\n return reduced;\r\n }, {});\r\n}\r\n/**\r\n * Allows using state and getters from one store without using the composition\r\n * API (`setup()`) by generating an object to be spread in the `computed` field\r\n * of a component.\r\n *\r\n * @param useStore - store to map from\r\n * @param keysOrMapper - array or object\r\n */\r\nfunction mapState(useStore, keysOrMapper) {\r\n return Array.isArray(keysOrMapper)\r\n ? keysOrMapper.reduce((reduced, key) => {\r\n reduced[key] = function () {\r\n return useStore(this.$pinia)[key];\r\n };\r\n return reduced;\r\n }, {})\r\n : Object.keys(keysOrMapper).reduce((reduced, key) => {\r\n // @ts-expect-error\r\n reduced[key] = function () {\r\n const store = useStore(this.$pinia);\r\n const storeKey = keysOrMapper[key];\r\n // for some reason TS is unable to infer the type of storeKey to be a\r\n // function\r\n return typeof storeKey === 'function'\r\n ? storeKey.call(this, store)\r\n : store[storeKey];\r\n };\r\n return reduced;\r\n }, {});\r\n}\r\n/**\r\n * Alias for `mapState()`. You should use `mapState()` instead.\r\n * @deprecated use `mapState()` instead.\r\n */\r\nconst mapGetters = mapState;\r\n/**\r\n * Allows directly using actions from your store without using the composition\r\n * API (`setup()`) by generating an object to be spread in the `methods` field\r\n * of a component.\r\n *\r\n * @param useStore - store to map from\r\n * @param keysOrMapper - array or object\r\n */\r\nfunction mapActions(useStore, keysOrMapper) {\r\n return Array.isArray(keysOrMapper)\r\n ? keysOrMapper.reduce((reduced, key) => {\r\n // @ts-expect-error\r\n reduced[key] = function (...args) {\r\n return useStore(this.$pinia)[key](...args);\r\n };\r\n return reduced;\r\n }, {})\r\n : Object.keys(keysOrMapper).reduce((reduced, key) => {\r\n // @ts-expect-error\r\n reduced[key] = function (...args) {\r\n return useStore(this.$pinia)[keysOrMapper[key]](...args);\r\n };\r\n return reduced;\r\n }, {});\r\n}\r\n/**\r\n * Allows using state and getters from one store without using the composition\r\n * API (`setup()`) by generating an object to be spread in the `computed` field\r\n * of a component.\r\n *\r\n * @param useStore - store to map from\r\n * @param keysOrMapper - array or object\r\n */\r\nfunction mapWritableState(useStore, keysOrMapper) {\r\n return Array.isArray(keysOrMapper)\r\n ? keysOrMapper.reduce((reduced, key) => {\r\n // @ts-ignore\r\n reduced[key] = {\r\n get() {\r\n return useStore(this.$pinia)[key];\r\n },\r\n set(value) {\r\n // it's easier to type it here as any\r\n return (useStore(this.$pinia)[key] = value);\r\n },\r\n };\r\n return reduced;\r\n }, {})\r\n : Object.keys(keysOrMapper).reduce((reduced, key) => {\r\n // @ts-ignore\r\n reduced[key] = {\r\n get() {\r\n return useStore(this.$pinia)[keysOrMapper[key]];\r\n },\r\n set(value) {\r\n // it's easier to type it here as any\r\n return (useStore(this.$pinia)[keysOrMapper[key]] = value);\r\n },\r\n };\r\n return reduced;\r\n }, {});\r\n}\n\n/**\r\n * Creates an object of references with all the state, getters, and plugin-added\r\n * state properties of the store. Similar to `toRefs()` but specifically\r\n * designed for Pinia stores so methods and non reactive properties are\r\n * completely ignored.\r\n *\r\n * @param store - store to extract the refs from\r\n */\r\nfunction storeToRefs(store) {\r\n // See https://github.com/vuejs/pinia/issues/852\r\n // It's easier to just use toRefs() even if it includes more stuff\r\n if (isVue2) {\r\n // @ts-expect-error: toRefs include methods and others\r\n return toRefs(store);\r\n }\r\n else {\r\n store = toRaw(store);\r\n const refs = {};\r\n for (const key in store) {\r\n const value = store[key];\r\n if (isRef(value) || isReactive(value)) {\r\n // @ts-expect-error: the key is state or getter\r\n refs[key] =\r\n // ---\r\n toRef(store, key);\r\n }\r\n }\r\n return refs;\r\n }\r\n}\n\n/**\r\n * Vue 2 Plugin that must be installed for pinia to work. Note **you don't need\r\n * this plugin if you are using Nuxt.js**. Use the `buildModule` instead:\r\n * https://pinia.vuejs.org/ssr/nuxt.html.\r\n *\r\n * @example\r\n * ```js\r\n * import Vue from 'vue'\r\n * import { PiniaVuePlugin, createPinia } from 'pinia'\r\n *\r\n * Vue.use(PiniaVuePlugin)\r\n * const pinia = createPinia()\r\n *\r\n * new Vue({\r\n * el: '#app',\r\n * // ...\r\n * pinia,\r\n * })\r\n * ```\r\n *\r\n * @param _Vue - `Vue` imported from 'vue'.\r\n */\r\nconst PiniaVuePlugin = function (_Vue) {\r\n // Equivalent of\r\n // app.config.globalProperties.$pinia = pinia\r\n _Vue.mixin({\r\n beforeCreate() {\r\n const options = this.$options;\r\n if (options.pinia) {\r\n const pinia = options.pinia;\r\n // HACK: taken from provide(): https://github.com/vuejs/composition-api/blob/main/src/apis/inject.ts#L31\r\n /* istanbul ignore else */\r\n if (!this._provided) {\r\n const provideCache = {};\r\n Object.defineProperty(this, '_provided', {\r\n get: () => provideCache,\r\n set: (v) => Object.assign(provideCache, v),\r\n });\r\n }\r\n this._provided[piniaSymbol] = pinia;\r\n // propagate the pinia instance in an SSR friendly way\r\n // avoid adding it to nuxt twice\r\n /* istanbul ignore else */\r\n if (!this.$pinia) {\r\n this.$pinia = pinia;\r\n }\r\n pinia._a = this;\r\n if (IS_CLIENT) {\r\n // this allows calling useStore() outside of a component setup after\r\n // installing pinia's plugin\r\n setActivePinia(pinia);\r\n }\r\n if (USE_DEVTOOLS) {\r\n registerPiniaDevtools(pinia._a, pinia);\r\n }\r\n }\r\n else if (!this.$pinia && options.parent && options.parent.$pinia) {\r\n this.$pinia = options.parent.$pinia;\r\n }\r\n },\r\n destroyed() {\r\n delete this._pStores;\r\n },\r\n });\r\n};\n\nexport { MutationType, PiniaVuePlugin, acceptHMRUpdate, createPinia, defineStore, getActivePinia, mapActions, mapGetters, mapState, mapStores, mapWritableState, setActivePinia, setMapStoreSuffix, skipHydrate, storeToRefs };\n","import { SavedAsset } from \"@/types\";\nimport { DataBinding, PublishMetadata, UserDefinedAppData } from \"@/types/data\";\nimport { WidgetWithConditions } from \"@/components/widgets/Widget\";\n\nexport type AppMode = \"render\" | \"edit\";\n\nexport interface AppState {\n assets: SavedAsset[];\n widgets: Record;\n dataBindings: DataBinding[];\n /**\n * Mapping of parent widget id -> child widget array\n */\n parents: Record;\n width: number;\n height: number;\n uuid: string;\n appVersionUuid: string;\n appSchemaVersion: number;\n name: string;\n custom: UserDefinedAppData;\n checksum: number | null;\n publishMeta: PublishMetadata | null;\n ianaTimeZone: string | null;\n isTemplate: boolean;\n feed: any;\n appMode: AppMode;\n maxAllowedConnectionCount: number;\n\n fallbackImageTimeoutSec?: number;\n introImageTimeoutSec?: number;\n}\n\nexport const makeInitialAppState = (): AppState => {\n return {\n assets: [],\n widgets: {},\n dataBindings: [],\n parents: {},\n custom: {\n fonts: [],\n userUploadedAssets: [],\n },\n uuid: \"\",\n appSchemaVersion: 1,\n appVersionUuid: \"\",\n name: \"\",\n width: 1920,\n height: 1080,\n checksum: null,\n publishMeta: null,\n ianaTimeZone: null,\n isTemplate: false,\n feed: {},\n appMode: \"render\",\n maxAllowedConnectionCount: 5,\n };\n};\n","import { WidgetWithConditions } from \"@/components/widgets/Widget\";\nimport { SavedAsset } from \".\";\nimport { FaultErrorCode } from \"./faultErrorCode\";\nimport { Logic } from \"./logic\";\n\n/**\n * A data connection represents\n * - A data source (e.g. Google Sheet, CSV, JSON url)\n * - Authorization for that data source\n * - The expected data schema\n */\nexport interface DataConnection {\n uuid: string;\n appUuid?: string;\n authProvider: AuthProvider;\n authProviderDisplay: string;\n name: string;\n canConfigureRequest: boolean;\n canEditData: boolean;\n canEditSchema: boolean;\n canModerate: boolean;\n dataProvider: string;\n dataProviderDisplay: string;\n refreshRateSec: number;\n refreshRateTimeSpan: number;\n source: string;\n providerAuthId?: string;\n isUpload: boolean;\n createdOn: string;\n modifiedAt: string;\n nodes: Node[] | SchemaNode[];\n nodeSets: NodeSet[];\n logic: Logic[];\n allowFaultedSchema: boolean;\n shouldAlertOnFaultedSchema: boolean;\n moderationMode?: ModerationMode;\n shouldRemoveModeration?: boolean;\n appConnections: AppConnection[];\n isFaulted: boolean;\n isSyncable: boolean;\n faultErrorCode?: FaultErrorCode;\n faultErrorMessage?: string;\n isCollection?: boolean;\n schemaType?: string;\n options: ConnectionOption[];\n logicNodeReferences: LogicNodeReference[];\n iconUrl: string;\n}\n\nexport interface LogicNodeReference {\n conditionGroupUuid: string | null;\n logicUuid: string;\n nodeUuids: string[];\n type: \"Filter\" | \"Condition\";\n}\n\nexport class ConnectionOption {\n id: number | null;\n type: string;\n key: string;\n value: string;\n}\n\nexport enum ConnectionOptionType {\n Method = 0,\n Header = 1,\n Parameter = 2,\n Body = 3,\n}\n\nexport interface ProviderAuthorization {\n providerAuthId: number;\n username: string;\n name: string;\n}\n\nexport type AuthProvider = \"Microsoft\" | \"Google\" | \"None\";\n\nexport interface AppConnection {\n appUuid: string;\n appName: string;\n isPublished: boolean;\n}\n\nexport interface DataValue {\n /**\n * The value as a string with formatting.\n */\n formattedValue: string;\n\n /**\n * The parsed data value from the data source\n */\n value: any;\n}\n\nexport interface SchemaNode {\n /**\n * An array of potential dataTypes that the schemaNode could be.\n */\n altTypes: DataType[];\n\n /**\n * Describes the schemaNode type. See the Data Types Enum model above.\n */\n dataType: DataType;\n\n /**\n * Only used on client side to display XML / JSON query paths. Server knows nothing about it.\n */\n displayQuery: string;\n\n /**\n * Indicates if the node is generated by the BE. Currently the Index node is artificial.\n */\n isArtificial: boolean;\n\n /**\n * If moderation is enabled for the connection, if isPrimaryKey is True, then this nodes value is used as the reference key to lookup the associated moderation value.\n */\n isPrimaryKey: boolean;\n\n /**\n * Indicates if the node is required to be present when inserting data via the API. Also used to enforce a schema for a custom widget, i.e. Calendar widgets.\n */\n isRequired: boolean;\n\n /**\n * Indicates if the schemaNode is selected. If not selected it will still be persisted but will not be included when fetching the schema and data for building an app.\n */\n isSelected: boolean;\n\n /**\n * Indicates if the node exists in the original schema but no longer exists in the sync'ed data schema.\n */\n isMissing?: boolean;\n\n /**\n * Indicates if the schemaNode is not supported for selecting. The primary use case for this currently is to flag nested arrays in Json/Xml.\n *\n * TODO: DELETE THIS. IT WONT BE NEEDED\n */\n isUnsupported?: boolean;\n\n /**\n * Enum Values: Node, NodeSet Indicates if the schemaNode is a Node or NodeSet\n */\n kind: NodeKind;\n\n /**\n * The name of the schema node.\n */\n name: string;\n\n /**\n * The uuid of this node's parent node.\n */\n parentUuid: string;\n\n /**\n * The reference value associated with the schemaNode that is used to query the node data.\n */\n query: string;\n\n /**\n * An alternative name used to identify a column or entity.\n * The property is not editable and is populated by the server.\n *\n * - For tabular data, the reference field is the actual column name\n * created by the user in the data source (not to be confused with\n * query, which is the actual spreadsheet column letter).\n * - For json/xml the reference field is the path to the specific\n * properties parent in the document.\n */\n referenceName: string;\n\n /**\n * An array of values { formattedValue: string, value: any } from the data source. The formattedValue is the string value from the data source; value is the parsed data value from the data source.\n */\n sampleValues: DataValue[];\n\n /**\n * Explicit indication if the schemaNode should be deleted and removed from the underlying schema.\n */\n shouldDelete: boolean;\n\n /**\n * Used to represent a header row(s) offset.\n */\n startIndex?: number;\n\n /**\n * The node uuid.\n */\n uuid?: string;\n}\n\nexport type ColumnPreviewInfo = SchemaNode & {\n importedRecordCount?: number;\n};\n\nexport interface TreeSchemaNode extends SchemaNode {\n /**\n * Children of the current SchemaNode. Only present for TreeData.\n */\n children: TreeSchemaNode[];\n\n /**\n * Number of items in children array\n */\n length?: number;\n\n /**\n * The node value. Only applies to TreeData, should not be present for tabular data\n */\n value: DataValue;\n}\n\n/**\n * This represents\n * - The schema for a 2D array of values\n * - Instructions for the backend to query data from the underlying source\n */\nexport interface NodeSet {\n uuid?: string;\n name: string;\n startIndex?: number;\n endIndex?: number;\n nodes?: Node[];\n query: string;\n customQuery?: string;\n dataType?: DataType;\n parentDataNodeSetUuid?: string;\n isSelected: boolean;\n}\n\n/**\n * This represents\n * - The schema for a single data value\n * - Instructions for the backend to query data from the underlying source\n */\nexport interface Node {\n uuid?: string;\n name: string;\n dataType: DataType;\n query: string;\n isSelected: boolean;\n isPrimaryKey: boolean;\n isRequired?: boolean;\n isArtificial?: boolean;\n}\n\nexport class NodeKind {\n static node = \"Node\";\n static nodeSet = \"NodeSet\";\n}\n\n/**\n * This represents\n * - A data value\n * - Metadata about that value (name, kind)\n */\nexport interface NodeData {\n /**\n * The unique ID for the column (not the actual data row)\n */\n uuid: string | null;\n dataType: DataType;\n query: string;\n kind: NodeKind;\n displayName: string;\n isArtificial: boolean;\n isPrimaryKey: boolean;\n value: string | boolean | number | Date | null;\n formattedValue: string;\n groupUuid?: string;\n assetUuid?: string;\n}\n\n/**\n * This represents\n * - A data tree\n * - Metadata about that value (name, kind)\n */\nexport interface NodeSetData {\n uuid: string;\n dataType: DataType;\n query: string;\n kind: NodeKind;\n children: (NodeData | NodeSetData)[][];\n importedRecordCount?: number;\n}\n\nexport interface ManualDataRow {\n rowUuid: string;\n columns: NodeData[];\n /**\n * Used to control moderation settings (removed or approved e.g.)\n */\n isSelected?: boolean;\n isHidden?: boolean;\n}\n\nexport type ModerationMode = \"Approval\" | \"Removal\" | null;\n\nexport type BindingType = \"Collection\" | \"Scalar\";\n\nexport type SchemaType = \"Undefined\" | \"Tabular\" | \"Tree\" | \"Calendar\";\n\nexport type DataType =\n | \"Number\"\n | \"String\"\n | \"Bool\"\n | \"Date\"\n | \"Time\"\n | \"DateTime\"\n | \"ImageUrl\"\n | \"ImageUpload\"\n | \"Color\"\n | \"Object\"\n | \"Url\"\n | \"PrimativeArray\"\n | \"ObjectArray\"\n | \"NestedArray\";\n\nexport const DataTypes: DataType[] = [\n \"Number\",\n \"String\",\n \"Bool\",\n \"Date\",\n \"Time\",\n \"DateTime\",\n \"ImageUrl\",\n \"ImageUpload\",\n \"Color\",\n \"Object\",\n \"PrimativeArray\",\n \"ObjectArray\",\n \"NestedArray\",\n];\n\nexport const UserDefinedDataTypes: DataType[] = [\n \"Number\",\n \"String\",\n \"Bool\",\n \"Date\",\n \"Time\",\n \"DateTime\",\n \"ImageUrl\",\n \"ImageUpload\",\n \"Color\",\n];\n\nexport type Lookup = { [key: string]: T };\n\nexport interface UserUploadedAsset {\n uuid: string;\n url: string;\n name?: string;\n}\n\nexport type UserDefinedAppData = {\n fonts: string[];\n userUploadedAssets: UserUploadedAsset[];\n};\n\nexport interface AppModel {\n appSchemaVersion: number;\n widgets: Record;\n parents: { [key: string]: string[] };\n custom: UserDefinedAppData;\n}\n\nexport interface SavedAppInfo extends AppModel {\n dataBindings: DataBinding[];\n name: string;\n data: Record>;\n}\n\n// Pretty sure the idea is that singleDataNode is like \"Location\",\n// DataSetNode is like \"CalendarEventInfo\",\n// and DataSet would be like \"CalendarInfo\" -- an array of CalendarEventInfo objects, each of which is a \"row\" in a table\n// Wonder if different naming might bring that out more\nexport type DataBindingType =\n | \"Scalar\"\n | \"DataSetNode\"\n | \"DataSet\"\n | \"Asset\"\n | \"Filter\"\n | \"DataSetParent\";\n\nexport type DataSortDirection = \"None\" | \"Ascending\" | \"Descending\" | \"Random\";\n\nexport interface DataBinding {\n widgetId: string;\n property: DataBindingProperty;\n dataConnectionUuid?: string;\n dataParentUuid?: string | null;\n dataUuid: string;\n bindingType: DataBindingType;\n query?: string;\n dataName?: string;\n parentWidgetId?: string;\n filterUuid?: string;\n sortDirection?: DataSortDirection;\n orderByDataUuid?: string;\n shouldRandomize?: boolean;\n conditionUuid?: string;\n conditionGroupUuid?: string | null;\n}\n\nexport interface DataBindingAppInfo {\n appName: string;\n appUuid: string;\n widgetIds: string[];\n isPublished: boolean;\n}\n\nexport interface DataBindingFilterInfo {\n appName: string;\n appUuid: string;\n filterUuid: string;\n filterName: string;\n}\n\nexport interface DataBindingUsageInfo {\n dataUuid: string;\n dataParentUui?: string;\n isUsedInFilter: boolean;\n bindingType: DataBindingType;\n appInfo: DataBindingAppInfo[];\n filterInfo: DataBindingFilterInfo[];\n}\nexport interface AppPayload {\n appVersionUuid: string;\n assets: SavedAsset[];\n categories: any[];\n checksum: number | null;\n createdOn: string;\n dataBindings: DataBinding[];\n description: string;\n feed: Feed;\n height: number;\n ianaTimeZone: string | null;\n isPublic: boolean;\n isTemplate: boolean;\n model: AppModel;\n modifiedAt: string | null;\n name: string;\n previewUrl: string;\n tags: any[];\n uuid: string;\n width: number;\n fallbackImageTimeoutSec?: number;\n introImageTimeoutSec?: number;\n}\n\nexport class FeedDeliveryMethod {\n static MediaRss = \"MediaRss\";\n static Html = \"Html\";\n static DirectUrl = \"DirectUrl\";\n}\n\n/**\n * PublishedUrl - url associated with the published version of the app. If the\n * app is unpublished the url will return the draft version.\n *\n * PreviewUrl - url associated with the current draft version\n */\nexport interface Feed {\n name: string;\n publishedUrl: string;\n previewUrl: string;\n deliveryMethod: FeedDeliveryMethod;\n}\n\nexport type DataStruct = \"Tabular\" | \"Tree\" | \"Undefined\";\n\nexport interface SourceDataSchema {\n url: string;\n schema: SchemaNode[];\n connectionUuid: string;\n schemaInfoUuid: string;\n lastDataSourceUpdate: string;\n recordCount: number;\n schemaType: SchemaType;\n clientComponentName: string;\n dataStruct: DataStruct;\n}\nexport interface TabularDataSchema {\n url: string;\n sheets: TableSchema[];\n connectionUuid: string;\n schemaInfoUuid: string;\n lastDataSourceUpdate: string;\n schemaType: SchemaType;\n dataStruct: DataStruct;\n spreadsheetName: string;\n messages: string[] | null;\n}\n\nexport interface GetConnectionBindingsResult {\n dataConnection: DataConnection;\n appBindingInfo: DataBindingUsageInfo[];\n}\n\nexport interface EditSchemaResult {\n dataConnection: DataConnection;\n appBindings: DataBindingUsageInfo[];\n brokenBindings: DataBindingUsageInfo[];\n hasSchemaChanged: boolean;\n schema: any;\n missingNodes: SchemaNode[];\n schemaInfoUuid: string;\n lastDataSourceUpdate: Date | null;\n}\n\nexport interface TableSchema {\n id: string;\n title: string;\n rangeId: string;\n hasHeader: boolean;\n schema: SchemaNode[];\n recordCount: number;\n}\n\nexport interface ConnectionDataResponse {\n data: NodeSetData | NodeData;\n lastUpdated: string;\n}\n\nexport type ConnectionRefreshRate =\n | \"Never\"\n | \"OnDemand\"\n | \"FifteenMin\"\n | \"Hourly\"\n | \"Daily\"\n | \"Weekly\";\n\n/**\n * This is the model that is sent to and from the server\n */\nexport interface FilterOptions {\n logic: Logic;\n types: string[];\n schema: SchemaNode[];\n message: string;\n dataConnectionName: string;\n}\n\nexport interface ModerationDataNode {\n refUuid: string;\n primaryKeyId: string;\n primaryKeyValue?: string;\n isSelected: boolean;\n isHidden?: boolean;\n moderatedByUsername?: string;\n moderatedOn?: Date;\n data: NodeData[];\n}\n\nexport interface ModerationDataNodeSet {\n parentDataNodeSet: NodeSetData;\n moderationDataNodes: ModerationDataNode[];\n}\n\nexport interface ModerationDataContainer {\n lastDataSourceUpdate: Date;\n data: ModerationDataNodeSet;\n}\n\nexport interface ModerationDataQueryResult {\n data: ModerationDataContainer;\n hasPrimaryKey: boolean;\n hasSchemaChanged: boolean;\n isDataValid: boolean;\n}\n\nexport interface PublishMetadata {\n isPublished: boolean;\n publishedOn: string | null;\n unpublishedOn: string | null;\n}\n\nexport interface CanPublishMetadata {\n canPublish: boolean;\n hasReseller: boolean;\n subscribeToConnectOneUrl: string;\n subscribeToConnectUnlimitedUrl: string;\n}\n\nexport interface AppInfo {\n uuid: string;\n name: string;\n description: string;\n createdByAccountId: number;\n createdOn: string;\n modifiedAt: string;\n appVersionUuid: string;\n featuredStart: string;\n featuredEnd: string;\n featuredOrder: string;\n width: number;\n height: number;\n previewUrl: string;\n previewAssetUuid: string | null;\n ianaTimeZone: string | null;\n isTemplate: boolean;\n isPublic: boolean;\n publishMeta: PublishMetadata;\n feed: Feed;\n checksum: number;\n categories: Keyword[];\n tags: Keyword[];\n maxAllowedConnectionCount: number;\n dataBindings?: DataBinding[];\n}\n\n// Keywords can be Categories or Tags\nexport interface Keyword {\n id: number | null;\n type: string;\n text: string;\n value: string;\n order: number;\n isPrimary: boolean;\n}\n\nexport type DataBindingProperty =\n | \"h\"\n | \"w\"\n | \"x\"\n | \"y\"\n | \"z\"\n | \"scaleX\"\n | \"scaleY\"\n | \"data\"\n | \"parentId\"\n | \"opacity\"\n | \"angle\"\n | \"locked\"\n | \"lockAspect\"\n | \"backgroundColor\"\n | \"backgroundImageUrl\"\n | \"backgroundSize\"\n | \"shadow\"\n | \"shadowX\"\n | \"shadowY\"\n | \"shadowBlur\"\n | \"shadowColor\"\n | \"border\"\n | \"borderWidth\"\n | \"borderColor\"\n | \"borderIsDashed\"\n | \"borderDashSize\"\n | \"borderGapSize\"\n | \"borderStyle\"\n | \"borderAlign\"\n | \"perspective\"\n | \"perspectiveAngle\"\n | \"color\"\n | \"blur\"\n | \"blurValue\"\n | \"blurDisplay\"\n | \"paddingTop\"\n | \"paddingRight\"\n | \"paddingBottom\"\n | \"paddingLeft\"\n | \"fill\"\n | \"font\"\n | \"textColor\"\n | \"fontFamily\"\n | \"fontSize\"\n | \"fontWeight\"\n | \"textTransform\"\n | \"fontStyle\"\n | \"letterSpacing\"\n | \"lineHeight\"\n | \"textAlign\"\n | \"textStyle\"\n | \"alignHorizontal\"\n | \"alignVertical\"\n | \"content\"\n | \"url\"\n | \"dataset\"\n | \"datasets\"\n | \"syncFonts\"\n | \"syncTextColors\"\n | \"paragraphSpacing\"\n | \"minValue\"\n | \"maxValue\"\n | \"lineColor\"\n | \"bgColor\"\n | \"barColor\"\n | \"plotlineColor\"\n | \"bgFill\"\n | \"fgFill\"\n | \"stroke\"\n | \"dividerFill\"\n | \"dividerStroke\"\n | \"currentValue\"\n | \"dayOffset\"\n | \"dayFormat\"\n | \"startHour\"\n | \"endHour\"\n | \"eventBackgroundColor\"\n | \"allDayEventBackgroundColor\"\n | \"verticalLinesColor\"\n | \"verticalLines_color\"\n | \"event_backgroundColor\"\n | \"today_backgroundColor\"\n | \"today_textColor\"\n | \"nowLineColor\"\n | \"calendarBackgroundColor\"\n | \"showEventLocation\"\n | \"showEventCreator\"\n | \"showEventTime\"\n | \"showEventDescription\"\n | \"trackLength\"\n | \"dividerColor\"\n | \"dividerFillColor\"\n | \"showDivider\"\n | \"progressBarBackgroundColor\"\n | \"progressBarFillColor\"\n | \"datetimeFormat\"\n | \"datetimeValue\"\n | \"userSpecifiedLocale\"\n | \"useBrowserLocale\"\n | \"useCurrentTime\"\n | \"verticalDynamism\"\n | \"verticalMargin\"\n | \"orderedChildIds\"\n | \"backgroundImageUrl2\"\n | \"cellBackgroundColor\";\n","/**\n * The default `parentId` value for widgets added\n * directly the canvas (not a child of another widget).\n */\nexport const BASE_PARENT_ID = \"base\";\nexport const PLACEHOLDER_IMAGE_PROPERTY = \"__placeholderImage\";\n\n/**\n * Default connection refresh rate in minutes\n */\nexport const DEFAULT_CONNECTION_REFRESH_RATE = 15;\n\n/**\n * Root path for app editor\n */\nexport const APP_EDITOR_ROUTE_PATH = \"app/edit\";\n\n/**\n * Unique string that is present in non-editor \"Connection\" paths.\n * It is used to determine which context the user is in when managing\n * or creating a connection.\n */\nexport const APP_EDITOR_CONNECTION_ROUTE_PATH = \"connections\";\n\n/**\n * The unique id for \"placeholder\" connections. For example,\n * Calendar, Graphs have fake data.\n *\n * This is used when a user presses \"Manage Data\" for a\n * Calendar or Graph that isn't yet databound.\n */\nexport const PLACEHOLDER_CONNECTION_UUID = \"placeholder\";\n\n/**\n * The DOM id for the component.\n *\n * It is used in a few places where we need to measure the DOM\n * at runtime.\n */\nexport const CANVAS_ID = \"canvas\";\n\n/**\n * The DOM id for the left-hand sidebar in the Dashboard.\n * Used when computing positions of cells in virtual grid collection.\n */\n\nexport const DASHBOARD_SIDEBAR_ID = \"dashboard_sidebar\";\n\n/**\n * The value for the \"default\" condition.\n */\nexport const DEFAULT_CONDITION_ID = \"00000000-0000-0000-0000-000000000000\";\n\nexport const DEFAULT_FALLBACK_IMAGE_TIMEOUT_SEC = 60 * 60; // 60 minutes\nexport const DEFAULT_INTRO_IMAGE_TIMEOUT_SEC = 2;\n\n// Default widget names for layer panel display\nexport const DEFAULT_LAYER_NAMES: { [key: string]: string } = {\n Text: \"Text\",\n Repeater: \"Repeater\",\n Group: \"Group\",\n CalendarEvent: \"Event\",\n CalendarDay: \"Day\",\n CalendarAgenda: \"Agenda\",\n CalendarRoomSchedule: \"Room Schedule\",\n CalendarWeek: \"Calendar Week\",\n ProgressBar: \"Progress Bar\",\n ProgressDonut: \"Progress Circle\",\n Countdown: \"Countdown / Up\",\n Datetime: \"Date / Time\",\n AnalogClock: \"Analog Clock\",\n ColumnGraph: \"Column Graph\",\n BarGraph: \"Bar Graph\",\n LineGraph: \"Line Graph\",\n PieGraph: \"Pie Graph\",\n StackedGraph: \"Stacked Graph\",\n Slide: \"Slide\",\n Multiframe: \"Multiframe\",\n Image: \"Image\",\n Video: \"Video\",\n Rectangle: \"Rectangle\",\n Ellipse: \"Ellipse\",\n Svg: \"Polygon\",\n Triangle: \"Triangle\",\n Line: \"Line\",\n};\n","'use strict';\n\nmodule.exports = function bind(fn, thisArg) {\n return function wrap() {\n var args = new Array(arguments.length);\n for (var i = 0; i < args.length; i++) {\n args[i] = arguments[i];\n }\n return fn.apply(thisArg, args);\n };\n};\n","'use strict';\n\nvar bind = require('./helpers/bind');\n\n// utils is a library of generic helper functions non-specific to axios\n\nvar toString = Object.prototype.toString;\n\n// eslint-disable-next-line func-names\nvar kindOf = (function(cache) {\n // eslint-disable-next-line func-names\n return function(thing) {\n var str = toString.call(thing);\n return cache[str] || (cache[str] = str.slice(8, -1).toLowerCase());\n };\n})(Object.create(null));\n\nfunction kindOfTest(type) {\n type = type.toLowerCase();\n return function isKindOf(thing) {\n return kindOf(thing) === type;\n };\n}\n\n/**\n * Determine if a value is an Array\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is an Array, otherwise false\n */\nfunction isArray(val) {\n return Array.isArray(val);\n}\n\n/**\n * Determine if a value is undefined\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if the value is undefined, otherwise false\n */\nfunction isUndefined(val) {\n return typeof val === 'undefined';\n}\n\n/**\n * Determine if a value is a Buffer\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Buffer, otherwise false\n */\nfunction isBuffer(val) {\n return val !== null && !isUndefined(val) && val.constructor !== null && !isUndefined(val.constructor)\n && typeof val.constructor.isBuffer === 'function' && val.constructor.isBuffer(val);\n}\n\n/**\n * Determine if a value is an ArrayBuffer\n *\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is an ArrayBuffer, otherwise false\n */\nvar isArrayBuffer = kindOfTest('ArrayBuffer');\n\n\n/**\n * Determine if a value is a view on an ArrayBuffer\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a view on an ArrayBuffer, otherwise false\n */\nfunction isArrayBufferView(val) {\n var result;\n if ((typeof ArrayBuffer !== 'undefined') && (ArrayBuffer.isView)) {\n result = ArrayBuffer.isView(val);\n } else {\n result = (val) && (val.buffer) && (isArrayBuffer(val.buffer));\n }\n return result;\n}\n\n/**\n * Determine if a value is a String\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a String, otherwise false\n */\nfunction isString(val) {\n return typeof val === 'string';\n}\n\n/**\n * Determine if a value is a Number\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Number, otherwise false\n */\nfunction isNumber(val) {\n return typeof val === 'number';\n}\n\n/**\n * Determine if a value is an Object\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is an Object, otherwise false\n */\nfunction isObject(val) {\n return val !== null && typeof val === 'object';\n}\n\n/**\n * Determine if a value is a plain Object\n *\n * @param {Object} val The value to test\n * @return {boolean} True if value is a plain Object, otherwise false\n */\nfunction isPlainObject(val) {\n if (kindOf(val) !== 'object') {\n return false;\n }\n\n var prototype = Object.getPrototypeOf(val);\n return prototype === null || prototype === Object.prototype;\n}\n\n/**\n * Determine if a value is a Date\n *\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Date, otherwise false\n */\nvar isDate = kindOfTest('Date');\n\n/**\n * Determine if a value is a File\n *\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a File, otherwise false\n */\nvar isFile = kindOfTest('File');\n\n/**\n * Determine if a value is a Blob\n *\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Blob, otherwise false\n */\nvar isBlob = kindOfTest('Blob');\n\n/**\n * Determine if a value is a FileList\n *\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a File, otherwise false\n */\nvar isFileList = kindOfTest('FileList');\n\n/**\n * Determine if a value is a Function\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Function, otherwise false\n */\nfunction isFunction(val) {\n return toString.call(val) === '[object Function]';\n}\n\n/**\n * Determine if a value is a Stream\n *\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a Stream, otherwise false\n */\nfunction isStream(val) {\n return isObject(val) && isFunction(val.pipe);\n}\n\n/**\n * Determine if a value is a FormData\n *\n * @param {Object} thing The value to test\n * @returns {boolean} True if value is an FormData, otherwise false\n */\nfunction isFormData(thing) {\n var pattern = '[object FormData]';\n return thing && (\n (typeof FormData === 'function' && thing instanceof FormData) ||\n toString.call(thing) === pattern ||\n (isFunction(thing.toString) && thing.toString() === pattern)\n );\n}\n\n/**\n * Determine if a value is a URLSearchParams object\n * @function\n * @param {Object} val The value to test\n * @returns {boolean} True if value is a URLSearchParams object, otherwise false\n */\nvar isURLSearchParams = kindOfTest('URLSearchParams');\n\n/**\n * Trim excess whitespace off the beginning and end of a string\n *\n * @param {String} str The String to trim\n * @returns {String} The String freed of excess whitespace\n */\nfunction trim(str) {\n return str.trim ? str.trim() : str.replace(/^\\s+|\\s+$/g, '');\n}\n\n/**\n * Determine if we're running in a standard browser environment\n *\n * This allows axios to run in a web worker, and react-native.\n * Both environments support XMLHttpRequest, but not fully standard globals.\n *\n * web workers:\n * typeof window -> undefined\n * typeof document -> undefined\n *\n * react-native:\n * navigator.product -> 'ReactNative'\n * nativescript\n * navigator.product -> 'NativeScript' or 'NS'\n */\nfunction isStandardBrowserEnv() {\n if (typeof navigator !== 'undefined' && (navigator.product === 'ReactNative' ||\n navigator.product === 'NativeScript' ||\n navigator.product === 'NS')) {\n return false;\n }\n return (\n typeof window !== 'undefined' &&\n typeof document !== 'undefined'\n );\n}\n\n/**\n * Iterate over an Array or an Object invoking a function for each item.\n *\n * If `obj` is an Array callback will be called passing\n * the value, index, and complete array for each item.\n *\n * If 'obj' is an Object callback will be called passing\n * the value, key, and complete object for each property.\n *\n * @param {Object|Array} obj The object to iterate\n * @param {Function} fn The callback to invoke for each item\n */\nfunction forEach(obj, fn) {\n // Don't bother if no value provided\n if (obj === null || typeof obj === 'undefined') {\n return;\n }\n\n // Force an array if not already something iterable\n if (typeof obj !== 'object') {\n /*eslint no-param-reassign:0*/\n obj = [obj];\n }\n\n if (isArray(obj)) {\n // Iterate over array values\n for (var i = 0, l = obj.length; i < l; i++) {\n fn.call(null, obj[i], i, obj);\n }\n } else {\n // Iterate over object keys\n for (var key in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, key)) {\n fn.call(null, obj[key], key, obj);\n }\n }\n }\n}\n\n/**\n * Accepts varargs expecting each argument to be an object, then\n * immutably merges the properties of each object and returns result.\n *\n * When multiple objects contain the same key the later object in\n * the arguments list will take precedence.\n *\n * Example:\n *\n * ```js\n * var result = merge({foo: 123}, {foo: 456});\n * console.log(result.foo); // outputs 456\n * ```\n *\n * @param {Object} obj1 Object to merge\n * @returns {Object} Result of all merge properties\n */\nfunction merge(/* obj1, obj2, obj3, ... */) {\n var result = {};\n function assignValue(val, key) {\n if (isPlainObject(result[key]) && isPlainObject(val)) {\n result[key] = merge(result[key], val);\n } else if (isPlainObject(val)) {\n result[key] = merge({}, val);\n } else if (isArray(val)) {\n result[key] = val.slice();\n } else {\n result[key] = val;\n }\n }\n\n for (var i = 0, l = arguments.length; i < l; i++) {\n forEach(arguments[i], assignValue);\n }\n return result;\n}\n\n/**\n * Extends object a by mutably adding to it the properties of object b.\n *\n * @param {Object} a The object to be extended\n * @param {Object} b The object to copy properties from\n * @param {Object} thisArg The object to bind function to\n * @return {Object} The resulting value of object a\n */\nfunction extend(a, b, thisArg) {\n forEach(b, function assignValue(val, key) {\n if (thisArg && typeof val === 'function') {\n a[key] = bind(val, thisArg);\n } else {\n a[key] = val;\n }\n });\n return a;\n}\n\n/**\n * Remove byte order marker. This catches EF BB BF (the UTF-8 BOM)\n *\n * @param {string} content with BOM\n * @return {string} content value without BOM\n */\nfunction stripBOM(content) {\n if (content.charCodeAt(0) === 0xFEFF) {\n content = content.slice(1);\n }\n return content;\n}\n\n/**\n * Inherit the prototype methods from one constructor into another\n * @param {function} constructor\n * @param {function} superConstructor\n * @param {object} [props]\n * @param {object} [descriptors]\n */\n\nfunction inherits(constructor, superConstructor, props, descriptors) {\n constructor.prototype = Object.create(superConstructor.prototype, descriptors);\n constructor.prototype.constructor = constructor;\n props && Object.assign(constructor.prototype, props);\n}\n\n/**\n * Resolve object with deep prototype chain to a flat object\n * @param {Object} sourceObj source object\n * @param {Object} [destObj]\n * @param {Function} [filter]\n * @returns {Object}\n */\n\nfunction toFlatObject(sourceObj, destObj, filter) {\n var props;\n var i;\n var prop;\n var merged = {};\n\n destObj = destObj || {};\n\n do {\n props = Object.getOwnPropertyNames(sourceObj);\n i = props.length;\n while (i-- > 0) {\n prop = props[i];\n if (!merged[prop]) {\n destObj[prop] = sourceObj[prop];\n merged[prop] = true;\n }\n }\n sourceObj = Object.getPrototypeOf(sourceObj);\n } while (sourceObj && (!filter || filter(sourceObj, destObj)) && sourceObj !== Object.prototype);\n\n return destObj;\n}\n\n/*\n * determines whether a string ends with the characters of a specified string\n * @param {String} str\n * @param {String} searchString\n * @param {Number} [position= 0]\n * @returns {boolean}\n */\nfunction endsWith(str, searchString, position) {\n str = String(str);\n if (position === undefined || position > str.length) {\n position = str.length;\n }\n position -= searchString.length;\n var lastIndex = str.indexOf(searchString, position);\n return lastIndex !== -1 && lastIndex === position;\n}\n\n\n/**\n * Returns new array from array like object\n * @param {*} [thing]\n * @returns {Array}\n */\nfunction toArray(thing) {\n if (!thing) return null;\n var i = thing.length;\n if (isUndefined(i)) return null;\n var arr = new Array(i);\n while (i-- > 0) {\n arr[i] = thing[i];\n }\n return arr;\n}\n\n// eslint-disable-next-line func-names\nvar isTypedArray = (function(TypedArray) {\n // eslint-disable-next-line func-names\n return function(thing) {\n return TypedArray && thing instanceof TypedArray;\n };\n})(typeof Uint8Array !== 'undefined' && Object.getPrototypeOf(Uint8Array));\n\nmodule.exports = {\n isArray: isArray,\n isArrayBuffer: isArrayBuffer,\n isBuffer: isBuffer,\n isFormData: isFormData,\n isArrayBufferView: isArrayBufferView,\n isString: isString,\n isNumber: isNumber,\n isObject: isObject,\n isPlainObject: isPlainObject,\n isUndefined: isUndefined,\n isDate: isDate,\n isFile: isFile,\n isBlob: isBlob,\n isFunction: isFunction,\n isStream: isStream,\n isURLSearchParams: isURLSearchParams,\n isStandardBrowserEnv: isStandardBrowserEnv,\n forEach: forEach,\n merge: merge,\n extend: extend,\n trim: trim,\n stripBOM: stripBOM,\n inherits: inherits,\n toFlatObject: toFlatObject,\n kindOf: kindOf,\n kindOfTest: kindOfTest,\n endsWith: endsWith,\n toArray: toArray,\n isTypedArray: isTypedArray,\n isFileList: isFileList\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\nfunction encode(val) {\n return encodeURIComponent(val).\n replace(/%3A/gi, ':').\n replace(/%24/g, '$').\n replace(/%2C/gi, ',').\n replace(/%20/g, '+').\n replace(/%5B/gi, '[').\n replace(/%5D/gi, ']');\n}\n\n/**\n * Build a URL by appending params to the end\n *\n * @param {string} url The base of the url (e.g., http://www.google.com)\n * @param {object} [params] The params to be appended\n * @returns {string} The formatted url\n */\nmodule.exports = function buildURL(url, params, paramsSerializer) {\n /*eslint no-param-reassign:0*/\n if (!params) {\n return url;\n }\n\n var serializedParams;\n if (paramsSerializer) {\n serializedParams = paramsSerializer(params);\n } else if (utils.isURLSearchParams(params)) {\n serializedParams = params.toString();\n } else {\n var parts = [];\n\n utils.forEach(params, function serialize(val, key) {\n if (val === null || typeof val === 'undefined') {\n return;\n }\n\n if (utils.isArray(val)) {\n key = key + '[]';\n } else {\n val = [val];\n }\n\n utils.forEach(val, function parseValue(v) {\n if (utils.isDate(v)) {\n v = v.toISOString();\n } else if (utils.isObject(v)) {\n v = JSON.stringify(v);\n }\n parts.push(encode(key) + '=' + encode(v));\n });\n });\n\n serializedParams = parts.join('&');\n }\n\n if (serializedParams) {\n var hashmarkIndex = url.indexOf('#');\n if (hashmarkIndex !== -1) {\n url = url.slice(0, hashmarkIndex);\n }\n\n url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams;\n }\n\n return url;\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\nfunction InterceptorManager() {\n this.handlers = [];\n}\n\n/**\n * Add a new interceptor to the stack\n *\n * @param {Function} fulfilled The function to handle `then` for a `Promise`\n * @param {Function} rejected The function to handle `reject` for a `Promise`\n *\n * @return {Number} An ID used to remove interceptor later\n */\nInterceptorManager.prototype.use = function use(fulfilled, rejected, options) {\n this.handlers.push({\n fulfilled: fulfilled,\n rejected: rejected,\n synchronous: options ? options.synchronous : false,\n runWhen: options ? options.runWhen : null\n });\n return this.handlers.length - 1;\n};\n\n/**\n * Remove an interceptor from the stack\n *\n * @param {Number} id The ID that was returned by `use`\n */\nInterceptorManager.prototype.eject = function eject(id) {\n if (this.handlers[id]) {\n this.handlers[id] = null;\n }\n};\n\n/**\n * Iterate over all the registered interceptors\n *\n * This method is particularly useful for skipping over any\n * interceptors that may have become `null` calling `eject`.\n *\n * @param {Function} fn The function to call for each interceptor\n */\nInterceptorManager.prototype.forEach = function forEach(fn) {\n utils.forEach(this.handlers, function forEachHandler(h) {\n if (h !== null) {\n fn(h);\n }\n });\n};\n\nmodule.exports = InterceptorManager;\n","'use strict';\n\nvar utils = require('../utils');\n\nmodule.exports = function normalizeHeaderName(headers, normalizedName) {\n utils.forEach(headers, function processHeader(value, name) {\n if (name !== normalizedName && name.toUpperCase() === normalizedName.toUpperCase()) {\n headers[normalizedName] = value;\n delete headers[name];\n }\n });\n};\n","'use strict';\n\nvar utils = require('../utils');\n\n/**\n * Create an Error with the specified message, config, error code, request and response.\n *\n * @param {string} message The error message.\n * @param {string} [code] The error code (for example, 'ECONNABORTED').\n * @param {Object} [config] The config.\n * @param {Object} [request] The request.\n * @param {Object} [response] The response.\n * @returns {Error} The created error.\n */\nfunction AxiosError(message, code, config, request, response) {\n Error.call(this);\n this.message = message;\n this.name = 'AxiosError';\n code && (this.code = code);\n config && (this.config = config);\n request && (this.request = request);\n response && (this.response = response);\n}\n\nutils.inherits(AxiosError, Error, {\n toJSON: function toJSON() {\n return {\n // Standard\n message: this.message,\n name: this.name,\n // Microsoft\n description: this.description,\n number: this.number,\n // Mozilla\n fileName: this.fileName,\n lineNumber: this.lineNumber,\n columnNumber: this.columnNumber,\n stack: this.stack,\n // Axios\n config: this.config,\n code: this.code,\n status: this.response && this.response.status ? this.response.status : null\n };\n }\n});\n\nvar prototype = AxiosError.prototype;\nvar descriptors = {};\n\n[\n 'ERR_BAD_OPTION_VALUE',\n 'ERR_BAD_OPTION',\n 'ECONNABORTED',\n 'ETIMEDOUT',\n 'ERR_NETWORK',\n 'ERR_FR_TOO_MANY_REDIRECTS',\n 'ERR_DEPRECATED',\n 'ERR_BAD_RESPONSE',\n 'ERR_BAD_REQUEST',\n 'ERR_CANCELED'\n// eslint-disable-next-line func-names\n].forEach(function(code) {\n descriptors[code] = {value: code};\n});\n\nObject.defineProperties(AxiosError, descriptors);\nObject.defineProperty(prototype, 'isAxiosError', {value: true});\n\n// eslint-disable-next-line func-names\nAxiosError.from = function(error, code, config, request, response, customProps) {\n var axiosError = Object.create(prototype);\n\n utils.toFlatObject(error, axiosError, function filter(obj) {\n return obj !== Error.prototype;\n });\n\n AxiosError.call(axiosError, error.message, code, config, request, response);\n\n axiosError.name = error.name;\n\n customProps && Object.assign(axiosError, customProps);\n\n return axiosError;\n};\n\nmodule.exports = AxiosError;\n","'use strict';\n\nmodule.exports = {\n silentJSONParsing: true,\n forcedJSONParsing: true,\n clarifyTimeoutError: false\n};\n","'use strict';\n\nvar utils = require('../utils');\n\n/**\n * Convert a data object to FormData\n * @param {Object} obj\n * @param {?Object} [formData]\n * @returns {Object}\n **/\n\nfunction toFormData(obj, formData) {\n // eslint-disable-next-line no-param-reassign\n formData = formData || new FormData();\n\n var stack = [];\n\n function convertValue(value) {\n if (value === null) return '';\n\n if (utils.isDate(value)) {\n return value.toISOString();\n }\n\n if (utils.isArrayBuffer(value) || utils.isTypedArray(value)) {\n return typeof Blob === 'function' ? new Blob([value]) : Buffer.from(value);\n }\n\n return value;\n }\n\n function build(data, parentKey) {\n if (utils.isPlainObject(data) || utils.isArray(data)) {\n if (stack.indexOf(data) !== -1) {\n throw Error('Circular reference detected in ' + parentKey);\n }\n\n stack.push(data);\n\n utils.forEach(data, function each(value, key) {\n if (utils.isUndefined(value)) return;\n var fullKey = parentKey ? parentKey + '.' + key : key;\n var arr;\n\n if (value && !parentKey && typeof value === 'object') {\n if (utils.endsWith(key, '{}')) {\n // eslint-disable-next-line no-param-reassign\n value = JSON.stringify(value);\n } else if (utils.endsWith(key, '[]') && (arr = utils.toArray(value))) {\n // eslint-disable-next-line func-names\n arr.forEach(function(el) {\n !utils.isUndefined(el) && formData.append(fullKey, convertValue(el));\n });\n return;\n }\n }\n\n build(value, fullKey);\n });\n\n stack.pop();\n } else {\n formData.append(parentKey, convertValue(data));\n }\n }\n\n build(obj);\n\n return formData;\n}\n\nmodule.exports = toFormData;\n","'use strict';\n\nvar AxiosError = require('./AxiosError');\n\n/**\n * Resolve or reject a Promise based on response status.\n *\n * @param {Function} resolve A function that resolves the promise.\n * @param {Function} reject A function that rejects the promise.\n * @param {object} response The response.\n */\nmodule.exports = function settle(resolve, reject, response) {\n var validateStatus = response.config.validateStatus;\n if (!response.status || !validateStatus || validateStatus(response.status)) {\n resolve(response);\n } else {\n reject(new AxiosError(\n 'Request failed with status code ' + response.status,\n [AxiosError.ERR_BAD_REQUEST, AxiosError.ERR_BAD_RESPONSE][Math.floor(response.status / 100) - 4],\n response.config,\n response.request,\n response\n ));\n }\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\nmodule.exports = (\n utils.isStandardBrowserEnv() ?\n\n // Standard browser envs support document.cookie\n (function standardBrowserEnv() {\n return {\n write: function write(name, value, expires, path, domain, secure) {\n var cookie = [];\n cookie.push(name + '=' + encodeURIComponent(value));\n\n if (utils.isNumber(expires)) {\n cookie.push('expires=' + new Date(expires).toGMTString());\n }\n\n if (utils.isString(path)) {\n cookie.push('path=' + path);\n }\n\n if (utils.isString(domain)) {\n cookie.push('domain=' + domain);\n }\n\n if (secure === true) {\n cookie.push('secure');\n }\n\n document.cookie = cookie.join('; ');\n },\n\n read: function read(name) {\n var match = document.cookie.match(new RegExp('(^|;\\\\s*)(' + name + ')=([^;]*)'));\n return (match ? decodeURIComponent(match[3]) : null);\n },\n\n remove: function remove(name) {\n this.write(name, '', Date.now() - 86400000);\n }\n };\n })() :\n\n // Non standard browser env (web workers, react-native) lack needed support.\n (function nonStandardBrowserEnv() {\n return {\n write: function write() {},\n read: function read() { return null; },\n remove: function remove() {}\n };\n })()\n);\n","'use strict';\n\n/**\n * Determines whether the specified URL is absolute\n *\n * @param {string} url The URL to test\n * @returns {boolean} True if the specified URL is absolute, otherwise false\n */\nmodule.exports = function isAbsoluteURL(url) {\n // A URL is considered absolute if it begins with \"://\" or \"//\" (protocol-relative URL).\n // RFC 3986 defines scheme name as a sequence of characters beginning with a letter and followed\n // by any combination of letters, digits, plus, period, or hyphen.\n return /^([a-z][a-z\\d+\\-.]*:)?\\/\\//i.test(url);\n};\n","'use strict';\n\n/**\n * Creates a new URL by combining the specified URLs\n *\n * @param {string} baseURL The base URL\n * @param {string} relativeURL The relative URL\n * @returns {string} The combined URL\n */\nmodule.exports = function combineURLs(baseURL, relativeURL) {\n return relativeURL\n ? baseURL.replace(/\\/+$/, '') + '/' + relativeURL.replace(/^\\/+/, '')\n : baseURL;\n};\n","'use strict';\n\nvar isAbsoluteURL = require('../helpers/isAbsoluteURL');\nvar combineURLs = require('../helpers/combineURLs');\n\n/**\n * Creates a new URL by combining the baseURL with the requestedURL,\n * only when the requestedURL is not already an absolute URL.\n * If the requestURL is absolute, this function returns the requestedURL untouched.\n *\n * @param {string} baseURL The base URL\n * @param {string} requestedURL Absolute or relative URL to combine\n * @returns {string} The combined full path\n */\nmodule.exports = function buildFullPath(baseURL, requestedURL) {\n if (baseURL && !isAbsoluteURL(requestedURL)) {\n return combineURLs(baseURL, requestedURL);\n }\n return requestedURL;\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\n// Headers whose duplicates are ignored by node\n// c.f. https://nodejs.org/api/http.html#http_message_headers\nvar ignoreDuplicateOf = [\n 'age', 'authorization', 'content-length', 'content-type', 'etag',\n 'expires', 'from', 'host', 'if-modified-since', 'if-unmodified-since',\n 'last-modified', 'location', 'max-forwards', 'proxy-authorization',\n 'referer', 'retry-after', 'user-agent'\n];\n\n/**\n * Parse headers into an object\n *\n * ```\n * Date: Wed, 27 Aug 2014 08:58:49 GMT\n * Content-Type: application/json\n * Connection: keep-alive\n * Transfer-Encoding: chunked\n * ```\n *\n * @param {String} headers Headers needing to be parsed\n * @returns {Object} Headers parsed into an object\n */\nmodule.exports = function parseHeaders(headers) {\n var parsed = {};\n var key;\n var val;\n var i;\n\n if (!headers) { return parsed; }\n\n utils.forEach(headers.split('\\n'), function parser(line) {\n i = line.indexOf(':');\n key = utils.trim(line.substr(0, i)).toLowerCase();\n val = utils.trim(line.substr(i + 1));\n\n if (key) {\n if (parsed[key] && ignoreDuplicateOf.indexOf(key) >= 0) {\n return;\n }\n if (key === 'set-cookie') {\n parsed[key] = (parsed[key] ? parsed[key] : []).concat([val]);\n } else {\n parsed[key] = parsed[key] ? parsed[key] + ', ' + val : val;\n }\n }\n });\n\n return parsed;\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\nmodule.exports = (\n utils.isStandardBrowserEnv() ?\n\n // Standard browser envs have full support of the APIs needed to test\n // whether the request URL is of the same origin as current location.\n (function standardBrowserEnv() {\n var msie = /(msie|trident)/i.test(navigator.userAgent);\n var urlParsingNode = document.createElement('a');\n var originURL;\n\n /**\n * Parse a URL to discover it's components\n *\n * @param {String} url The URL to be parsed\n * @returns {Object}\n */\n function resolveURL(url) {\n var href = url;\n\n if (msie) {\n // IE needs attribute set twice to normalize properties\n urlParsingNode.setAttribute('href', href);\n href = urlParsingNode.href;\n }\n\n urlParsingNode.setAttribute('href', href);\n\n // urlParsingNode provides the UrlUtils interface - http://url.spec.whatwg.org/#urlutils\n return {\n href: urlParsingNode.href,\n protocol: urlParsingNode.protocol ? urlParsingNode.protocol.replace(/:$/, '') : '',\n host: urlParsingNode.host,\n search: urlParsingNode.search ? urlParsingNode.search.replace(/^\\?/, '') : '',\n hash: urlParsingNode.hash ? urlParsingNode.hash.replace(/^#/, '') : '',\n hostname: urlParsingNode.hostname,\n port: urlParsingNode.port,\n pathname: (urlParsingNode.pathname.charAt(0) === '/') ?\n urlParsingNode.pathname :\n '/' + urlParsingNode.pathname\n };\n }\n\n originURL = resolveURL(window.location.href);\n\n /**\n * Determine if a URL shares the same origin as the current location\n *\n * @param {String} requestURL The URL to test\n * @returns {boolean} True if URL shares the same origin, otherwise false\n */\n return function isURLSameOrigin(requestURL) {\n var parsed = (utils.isString(requestURL)) ? resolveURL(requestURL) : requestURL;\n return (parsed.protocol === originURL.protocol &&\n parsed.host === originURL.host);\n };\n })() :\n\n // Non standard browser envs (web workers, react-native) lack needed support.\n (function nonStandardBrowserEnv() {\n return function isURLSameOrigin() {\n return true;\n };\n })()\n);\n","'use strict';\n\nvar AxiosError = require('../core/AxiosError');\nvar utils = require('../utils');\n\n/**\n * A `CanceledError` is an object that is thrown when an operation is canceled.\n *\n * @class\n * @param {string=} message The message.\n */\nfunction CanceledError(message) {\n // eslint-disable-next-line no-eq-null,eqeqeq\n AxiosError.call(this, message == null ? 'canceled' : message, AxiosError.ERR_CANCELED);\n this.name = 'CanceledError';\n}\n\nutils.inherits(CanceledError, AxiosError, {\n __CANCEL__: true\n});\n\nmodule.exports = CanceledError;\n","'use strict';\n\nmodule.exports = function parseProtocol(url) {\n var match = /^([-+\\w]{1,25})(:?\\/\\/|:)/.exec(url);\n return match && match[1] || '';\n};\n","'use strict';\n\nvar utils = require('./../utils');\nvar settle = require('./../core/settle');\nvar cookies = require('./../helpers/cookies');\nvar buildURL = require('./../helpers/buildURL');\nvar buildFullPath = require('../core/buildFullPath');\nvar parseHeaders = require('./../helpers/parseHeaders');\nvar isURLSameOrigin = require('./../helpers/isURLSameOrigin');\nvar transitionalDefaults = require('../defaults/transitional');\nvar AxiosError = require('../core/AxiosError');\nvar CanceledError = require('../cancel/CanceledError');\nvar parseProtocol = require('../helpers/parseProtocol');\n\nmodule.exports = function xhrAdapter(config) {\n return new Promise(function dispatchXhrRequest(resolve, reject) {\n var requestData = config.data;\n var requestHeaders = config.headers;\n var responseType = config.responseType;\n var onCanceled;\n function done() {\n if (config.cancelToken) {\n config.cancelToken.unsubscribe(onCanceled);\n }\n\n if (config.signal) {\n config.signal.removeEventListener('abort', onCanceled);\n }\n }\n\n if (utils.isFormData(requestData) && utils.isStandardBrowserEnv()) {\n delete requestHeaders['Content-Type']; // Let the browser set it\n }\n\n var request = new XMLHttpRequest();\n\n // HTTP basic authentication\n if (config.auth) {\n var username = config.auth.username || '';\n var password = config.auth.password ? unescape(encodeURIComponent(config.auth.password)) : '';\n requestHeaders.Authorization = 'Basic ' + btoa(username + ':' + password);\n }\n\n var fullPath = buildFullPath(config.baseURL, config.url);\n\n request.open(config.method.toUpperCase(), buildURL(fullPath, config.params, config.paramsSerializer), true);\n\n // Set the request timeout in MS\n request.timeout = config.timeout;\n\n function onloadend() {\n if (!request) {\n return;\n }\n // Prepare the response\n var responseHeaders = 'getAllResponseHeaders' in request ? parseHeaders(request.getAllResponseHeaders()) : null;\n var responseData = !responseType || responseType === 'text' || responseType === 'json' ?\n request.responseText : request.response;\n var response = {\n data: responseData,\n status: request.status,\n statusText: request.statusText,\n headers: responseHeaders,\n config: config,\n request: request\n };\n\n settle(function _resolve(value) {\n resolve(value);\n done();\n }, function _reject(err) {\n reject(err);\n done();\n }, response);\n\n // Clean up request\n request = null;\n }\n\n if ('onloadend' in request) {\n // Use onloadend if available\n request.onloadend = onloadend;\n } else {\n // Listen for ready state to emulate onloadend\n request.onreadystatechange = function handleLoad() {\n if (!request || request.readyState !== 4) {\n return;\n }\n\n // The request errored out and we didn't get a response, this will be\n // handled by onerror instead\n // With one exception: request that using file: protocol, most browsers\n // will return status as 0 even though it's a successful request\n if (request.status === 0 && !(request.responseURL && request.responseURL.indexOf('file:') === 0)) {\n return;\n }\n // readystate handler is calling before onerror or ontimeout handlers,\n // so we should call onloadend on the next 'tick'\n setTimeout(onloadend);\n };\n }\n\n // Handle browser request cancellation (as opposed to a manual cancellation)\n request.onabort = function handleAbort() {\n if (!request) {\n return;\n }\n\n reject(new AxiosError('Request aborted', AxiosError.ECONNABORTED, config, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle low level network errors\n request.onerror = function handleError() {\n // Real errors are hidden from us by the browser\n // onerror should only fire if it's a network error\n reject(new AxiosError('Network Error', AxiosError.ERR_NETWORK, config, request, request));\n\n // Clean up request\n request = null;\n };\n\n // Handle timeout\n request.ontimeout = function handleTimeout() {\n var timeoutErrorMessage = config.timeout ? 'timeout of ' + config.timeout + 'ms exceeded' : 'timeout exceeded';\n var transitional = config.transitional || transitionalDefaults;\n if (config.timeoutErrorMessage) {\n timeoutErrorMessage = config.timeoutErrorMessage;\n }\n reject(new AxiosError(\n timeoutErrorMessage,\n transitional.clarifyTimeoutError ? AxiosError.ETIMEDOUT : AxiosError.ECONNABORTED,\n config,\n request));\n\n // Clean up request\n request = null;\n };\n\n // Add xsrf header\n // This is only done if running in a standard browser environment.\n // Specifically not if we're in a web worker, or react-native.\n if (utils.isStandardBrowserEnv()) {\n // Add xsrf header\n var xsrfValue = (config.withCredentials || isURLSameOrigin(fullPath)) && config.xsrfCookieName ?\n cookies.read(config.xsrfCookieName) :\n undefined;\n\n if (xsrfValue) {\n requestHeaders[config.xsrfHeaderName] = xsrfValue;\n }\n }\n\n // Add headers to the request\n if ('setRequestHeader' in request) {\n utils.forEach(requestHeaders, function setRequestHeader(val, key) {\n if (typeof requestData === 'undefined' && key.toLowerCase() === 'content-type') {\n // Remove Content-Type if data is undefined\n delete requestHeaders[key];\n } else {\n // Otherwise add header to the request\n request.setRequestHeader(key, val);\n }\n });\n }\n\n // Add withCredentials to request if needed\n if (!utils.isUndefined(config.withCredentials)) {\n request.withCredentials = !!config.withCredentials;\n }\n\n // Add responseType to request if needed\n if (responseType && responseType !== 'json') {\n request.responseType = config.responseType;\n }\n\n // Handle progress if needed\n if (typeof config.onDownloadProgress === 'function') {\n request.addEventListener('progress', config.onDownloadProgress);\n }\n\n // Not all browsers support upload events\n if (typeof config.onUploadProgress === 'function' && request.upload) {\n request.upload.addEventListener('progress', config.onUploadProgress);\n }\n\n if (config.cancelToken || config.signal) {\n // Handle cancellation\n // eslint-disable-next-line func-names\n onCanceled = function(cancel) {\n if (!request) {\n return;\n }\n reject(!cancel || (cancel && cancel.type) ? new CanceledError() : cancel);\n request.abort();\n request = null;\n };\n\n config.cancelToken && config.cancelToken.subscribe(onCanceled);\n if (config.signal) {\n config.signal.aborted ? onCanceled() : config.signal.addEventListener('abort', onCanceled);\n }\n }\n\n if (!requestData) {\n requestData = null;\n }\n\n var protocol = parseProtocol(fullPath);\n\n if (protocol && [ 'http', 'https', 'file' ].indexOf(protocol) === -1) {\n reject(new AxiosError('Unsupported protocol ' + protocol + ':', AxiosError.ERR_BAD_REQUEST, config));\n return;\n }\n\n\n // Send the request\n request.send(requestData);\n });\n};\n","// eslint-disable-next-line strict\nmodule.exports = null;\n","'use strict';\n\nvar utils = require('../utils');\nvar normalizeHeaderName = require('../helpers/normalizeHeaderName');\nvar AxiosError = require('../core/AxiosError');\nvar transitionalDefaults = require('./transitional');\nvar toFormData = require('../helpers/toFormData');\n\nvar DEFAULT_CONTENT_TYPE = {\n 'Content-Type': 'application/x-www-form-urlencoded'\n};\n\nfunction setContentTypeIfUnset(headers, value) {\n if (!utils.isUndefined(headers) && utils.isUndefined(headers['Content-Type'])) {\n headers['Content-Type'] = value;\n }\n}\n\nfunction getDefaultAdapter() {\n var adapter;\n if (typeof XMLHttpRequest !== 'undefined') {\n // For browsers use XHR adapter\n adapter = require('../adapters/xhr');\n } else if (typeof process !== 'undefined' && Object.prototype.toString.call(process) === '[object process]') {\n // For node use HTTP adapter\n adapter = require('../adapters/http');\n }\n return adapter;\n}\n\nfunction stringifySafely(rawValue, parser, encoder) {\n if (utils.isString(rawValue)) {\n try {\n (parser || JSON.parse)(rawValue);\n return utils.trim(rawValue);\n } catch (e) {\n if (e.name !== 'SyntaxError') {\n throw e;\n }\n }\n }\n\n return (encoder || JSON.stringify)(rawValue);\n}\n\nvar defaults = {\n\n transitional: transitionalDefaults,\n\n adapter: getDefaultAdapter(),\n\n transformRequest: [function transformRequest(data, headers) {\n normalizeHeaderName(headers, 'Accept');\n normalizeHeaderName(headers, 'Content-Type');\n\n if (utils.isFormData(data) ||\n utils.isArrayBuffer(data) ||\n utils.isBuffer(data) ||\n utils.isStream(data) ||\n utils.isFile(data) ||\n utils.isBlob(data)\n ) {\n return data;\n }\n if (utils.isArrayBufferView(data)) {\n return data.buffer;\n }\n if (utils.isURLSearchParams(data)) {\n setContentTypeIfUnset(headers, 'application/x-www-form-urlencoded;charset=utf-8');\n return data.toString();\n }\n\n var isObjectPayload = utils.isObject(data);\n var contentType = headers && headers['Content-Type'];\n\n var isFileList;\n\n if ((isFileList = utils.isFileList(data)) || (isObjectPayload && contentType === 'multipart/form-data')) {\n var _FormData = this.env && this.env.FormData;\n return toFormData(isFileList ? {'files[]': data} : data, _FormData && new _FormData());\n } else if (isObjectPayload || contentType === 'application/json') {\n setContentTypeIfUnset(headers, 'application/json');\n return stringifySafely(data);\n }\n\n return data;\n }],\n\n transformResponse: [function transformResponse(data) {\n var transitional = this.transitional || defaults.transitional;\n var silentJSONParsing = transitional && transitional.silentJSONParsing;\n var forcedJSONParsing = transitional && transitional.forcedJSONParsing;\n var strictJSONParsing = !silentJSONParsing && this.responseType === 'json';\n\n if (strictJSONParsing || (forcedJSONParsing && utils.isString(data) && data.length)) {\n try {\n return JSON.parse(data);\n } catch (e) {\n if (strictJSONParsing) {\n if (e.name === 'SyntaxError') {\n throw AxiosError.from(e, AxiosError.ERR_BAD_RESPONSE, this, null, this.response);\n }\n throw e;\n }\n }\n }\n\n return data;\n }],\n\n /**\n * A timeout in milliseconds to abort a request. If set to 0 (default) a\n * timeout is not created.\n */\n timeout: 0,\n\n xsrfCookieName: 'XSRF-TOKEN',\n xsrfHeaderName: 'X-XSRF-TOKEN',\n\n maxContentLength: -1,\n maxBodyLength: -1,\n\n env: {\n FormData: require('./env/FormData')\n },\n\n validateStatus: function validateStatus(status) {\n return status >= 200 && status < 300;\n },\n\n headers: {\n common: {\n 'Accept': 'application/json, text/plain, */*'\n }\n }\n};\n\nutils.forEach(['delete', 'get', 'head'], function forEachMethodNoData(method) {\n defaults.headers[method] = {};\n});\n\nutils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {\n defaults.headers[method] = utils.merge(DEFAULT_CONTENT_TYPE);\n});\n\nmodule.exports = defaults;\n","'use strict';\n\nvar utils = require('./../utils');\nvar defaults = require('../defaults');\n\n/**\n * Transform the data for a request or a response\n *\n * @param {Object|String} data The data to be transformed\n * @param {Array} headers The headers for the request or response\n * @param {Array|Function} fns A single function or Array of functions\n * @returns {*} The resulting transformed data\n */\nmodule.exports = function transformData(data, headers, fns) {\n var context = this || defaults;\n /*eslint no-param-reassign:0*/\n utils.forEach(fns, function transform(fn) {\n data = fn.call(context, data, headers);\n });\n\n return data;\n};\n","'use strict';\n\nmodule.exports = function isCancel(value) {\n return !!(value && value.__CANCEL__);\n};\n","'use strict';\n\nvar utils = require('./../utils');\nvar transformData = require('./transformData');\nvar isCancel = require('../cancel/isCancel');\nvar defaults = require('../defaults');\nvar CanceledError = require('../cancel/CanceledError');\n\n/**\n * Throws a `CanceledError` if cancellation has been requested.\n */\nfunction throwIfCancellationRequested(config) {\n if (config.cancelToken) {\n config.cancelToken.throwIfRequested();\n }\n\n if (config.signal && config.signal.aborted) {\n throw new CanceledError();\n }\n}\n\n/**\n * Dispatch a request to the server using the configured adapter.\n *\n * @param {object} config The config that is to be used for the request\n * @returns {Promise} The Promise to be fulfilled\n */\nmodule.exports = function dispatchRequest(config) {\n throwIfCancellationRequested(config);\n\n // Ensure headers exist\n config.headers = config.headers || {};\n\n // Transform request data\n config.data = transformData.call(\n config,\n config.data,\n config.headers,\n config.transformRequest\n );\n\n // Flatten headers\n config.headers = utils.merge(\n config.headers.common || {},\n config.headers[config.method] || {},\n config.headers\n );\n\n utils.forEach(\n ['delete', 'get', 'head', 'post', 'put', 'patch', 'common'],\n function cleanHeaderConfig(method) {\n delete config.headers[method];\n }\n );\n\n var adapter = config.adapter || defaults.adapter;\n\n return adapter(config).then(function onAdapterResolution(response) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n response.data = transformData.call(\n config,\n response.data,\n response.headers,\n config.transformResponse\n );\n\n return response;\n }, function onAdapterRejection(reason) {\n if (!isCancel(reason)) {\n throwIfCancellationRequested(config);\n\n // Transform response data\n if (reason && reason.response) {\n reason.response.data = transformData.call(\n config,\n reason.response.data,\n reason.response.headers,\n config.transformResponse\n );\n }\n }\n\n return Promise.reject(reason);\n });\n};\n","'use strict';\n\nvar utils = require('../utils');\n\n/**\n * Config-specific merge-function which creates a new config-object\n * by merging two configuration objects together.\n *\n * @param {Object} config1\n * @param {Object} config2\n * @returns {Object} New object resulting from merging config2 to config1\n */\nmodule.exports = function mergeConfig(config1, config2) {\n // eslint-disable-next-line no-param-reassign\n config2 = config2 || {};\n var config = {};\n\n function getMergedValue(target, source) {\n if (utils.isPlainObject(target) && utils.isPlainObject(source)) {\n return utils.merge(target, source);\n } else if (utils.isPlainObject(source)) {\n return utils.merge({}, source);\n } else if (utils.isArray(source)) {\n return source.slice();\n }\n return source;\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDeepProperties(prop) {\n if (!utils.isUndefined(config2[prop])) {\n return getMergedValue(config1[prop], config2[prop]);\n } else if (!utils.isUndefined(config1[prop])) {\n return getMergedValue(undefined, config1[prop]);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function valueFromConfig2(prop) {\n if (!utils.isUndefined(config2[prop])) {\n return getMergedValue(undefined, config2[prop]);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function defaultToConfig2(prop) {\n if (!utils.isUndefined(config2[prop])) {\n return getMergedValue(undefined, config2[prop]);\n } else if (!utils.isUndefined(config1[prop])) {\n return getMergedValue(undefined, config1[prop]);\n }\n }\n\n // eslint-disable-next-line consistent-return\n function mergeDirectKeys(prop) {\n if (prop in config2) {\n return getMergedValue(config1[prop], config2[prop]);\n } else if (prop in config1) {\n return getMergedValue(undefined, config1[prop]);\n }\n }\n\n var mergeMap = {\n 'url': valueFromConfig2,\n 'method': valueFromConfig2,\n 'data': valueFromConfig2,\n 'baseURL': defaultToConfig2,\n 'transformRequest': defaultToConfig2,\n 'transformResponse': defaultToConfig2,\n 'paramsSerializer': defaultToConfig2,\n 'timeout': defaultToConfig2,\n 'timeoutMessage': defaultToConfig2,\n 'withCredentials': defaultToConfig2,\n 'adapter': defaultToConfig2,\n 'responseType': defaultToConfig2,\n 'xsrfCookieName': defaultToConfig2,\n 'xsrfHeaderName': defaultToConfig2,\n 'onUploadProgress': defaultToConfig2,\n 'onDownloadProgress': defaultToConfig2,\n 'decompress': defaultToConfig2,\n 'maxContentLength': defaultToConfig2,\n 'maxBodyLength': defaultToConfig2,\n 'beforeRedirect': defaultToConfig2,\n 'transport': defaultToConfig2,\n 'httpAgent': defaultToConfig2,\n 'httpsAgent': defaultToConfig2,\n 'cancelToken': defaultToConfig2,\n 'socketPath': defaultToConfig2,\n 'responseEncoding': defaultToConfig2,\n 'validateStatus': mergeDirectKeys\n };\n\n utils.forEach(Object.keys(config1).concat(Object.keys(config2)), function computeConfigValue(prop) {\n var merge = mergeMap[prop] || mergeDeepProperties;\n var configValue = merge(prop);\n (utils.isUndefined(configValue) && merge !== mergeDirectKeys) || (config[prop] = configValue);\n });\n\n return config;\n};\n","module.exports = {\n \"version\": \"0.27.2\"\n};","'use strict';\n\nvar VERSION = require('../env/data').version;\nvar AxiosError = require('../core/AxiosError');\n\nvar validators = {};\n\n// eslint-disable-next-line func-names\n['object', 'boolean', 'number', 'function', 'string', 'symbol'].forEach(function(type, i) {\n validators[type] = function validator(thing) {\n return typeof thing === type || 'a' + (i < 1 ? 'n ' : ' ') + type;\n };\n});\n\nvar deprecatedWarnings = {};\n\n/**\n * Transitional option validator\n * @param {function|boolean?} validator - set to false if the transitional option has been removed\n * @param {string?} version - deprecated version / removed since version\n * @param {string?} message - some message with additional info\n * @returns {function}\n */\nvalidators.transitional = function transitional(validator, version, message) {\n function formatMessage(opt, desc) {\n return '[Axios v' + VERSION + '] Transitional option \\'' + opt + '\\'' + desc + (message ? '. ' + message : '');\n }\n\n // eslint-disable-next-line func-names\n return function(value, opt, opts) {\n if (validator === false) {\n throw new AxiosError(\n formatMessage(opt, ' has been removed' + (version ? ' in ' + version : '')),\n AxiosError.ERR_DEPRECATED\n );\n }\n\n if (version && !deprecatedWarnings[opt]) {\n deprecatedWarnings[opt] = true;\n // eslint-disable-next-line no-console\n console.warn(\n formatMessage(\n opt,\n ' has been deprecated since v' + version + ' and will be removed in the near future'\n )\n );\n }\n\n return validator ? validator(value, opt, opts) : true;\n };\n};\n\n/**\n * Assert object's properties type\n * @param {object} options\n * @param {object} schema\n * @param {boolean?} allowUnknown\n */\n\nfunction assertOptions(options, schema, allowUnknown) {\n if (typeof options !== 'object') {\n throw new AxiosError('options must be an object', AxiosError.ERR_BAD_OPTION_VALUE);\n }\n var keys = Object.keys(options);\n var i = keys.length;\n while (i-- > 0) {\n var opt = keys[i];\n var validator = schema[opt];\n if (validator) {\n var value = options[opt];\n var result = value === undefined || validator(value, opt, options);\n if (result !== true) {\n throw new AxiosError('option ' + opt + ' must be ' + result, AxiosError.ERR_BAD_OPTION_VALUE);\n }\n continue;\n }\n if (allowUnknown !== true) {\n throw new AxiosError('Unknown option ' + opt, AxiosError.ERR_BAD_OPTION);\n }\n }\n}\n\nmodule.exports = {\n assertOptions: assertOptions,\n validators: validators\n};\n","'use strict';\n\nvar utils = require('./../utils');\nvar buildURL = require('../helpers/buildURL');\nvar InterceptorManager = require('./InterceptorManager');\nvar dispatchRequest = require('./dispatchRequest');\nvar mergeConfig = require('./mergeConfig');\nvar buildFullPath = require('./buildFullPath');\nvar validator = require('../helpers/validator');\n\nvar validators = validator.validators;\n/**\n * Create a new instance of Axios\n *\n * @param {Object} instanceConfig The default config for the instance\n */\nfunction Axios(instanceConfig) {\n this.defaults = instanceConfig;\n this.interceptors = {\n request: new InterceptorManager(),\n response: new InterceptorManager()\n };\n}\n\n/**\n * Dispatch a request\n *\n * @param {Object} config The config specific for this request (merged with this.defaults)\n */\nAxios.prototype.request = function request(configOrUrl, config) {\n /*eslint no-param-reassign:0*/\n // Allow for axios('example/url'[, config]) a la fetch API\n if (typeof configOrUrl === 'string') {\n config = config || {};\n config.url = configOrUrl;\n } else {\n config = configOrUrl || {};\n }\n\n config = mergeConfig(this.defaults, config);\n\n // Set config.method\n if (config.method) {\n config.method = config.method.toLowerCase();\n } else if (this.defaults.method) {\n config.method = this.defaults.method.toLowerCase();\n } else {\n config.method = 'get';\n }\n\n var transitional = config.transitional;\n\n if (transitional !== undefined) {\n validator.assertOptions(transitional, {\n silentJSONParsing: validators.transitional(validators.boolean),\n forcedJSONParsing: validators.transitional(validators.boolean),\n clarifyTimeoutError: validators.transitional(validators.boolean)\n }, false);\n }\n\n // filter out skipped interceptors\n var requestInterceptorChain = [];\n var synchronousRequestInterceptors = true;\n this.interceptors.request.forEach(function unshiftRequestInterceptors(interceptor) {\n if (typeof interceptor.runWhen === 'function' && interceptor.runWhen(config) === false) {\n return;\n }\n\n synchronousRequestInterceptors = synchronousRequestInterceptors && interceptor.synchronous;\n\n requestInterceptorChain.unshift(interceptor.fulfilled, interceptor.rejected);\n });\n\n var responseInterceptorChain = [];\n this.interceptors.response.forEach(function pushResponseInterceptors(interceptor) {\n responseInterceptorChain.push(interceptor.fulfilled, interceptor.rejected);\n });\n\n var promise;\n\n if (!synchronousRequestInterceptors) {\n var chain = [dispatchRequest, undefined];\n\n Array.prototype.unshift.apply(chain, requestInterceptorChain);\n chain = chain.concat(responseInterceptorChain);\n\n promise = Promise.resolve(config);\n while (chain.length) {\n promise = promise.then(chain.shift(), chain.shift());\n }\n\n return promise;\n }\n\n\n var newConfig = config;\n while (requestInterceptorChain.length) {\n var onFulfilled = requestInterceptorChain.shift();\n var onRejected = requestInterceptorChain.shift();\n try {\n newConfig = onFulfilled(newConfig);\n } catch (error) {\n onRejected(error);\n break;\n }\n }\n\n try {\n promise = dispatchRequest(newConfig);\n } catch (error) {\n return Promise.reject(error);\n }\n\n while (responseInterceptorChain.length) {\n promise = promise.then(responseInterceptorChain.shift(), responseInterceptorChain.shift());\n }\n\n return promise;\n};\n\nAxios.prototype.getUri = function getUri(config) {\n config = mergeConfig(this.defaults, config);\n var fullPath = buildFullPath(config.baseURL, config.url);\n return buildURL(fullPath, config.params, config.paramsSerializer);\n};\n\n// Provide aliases for supported request methods\nutils.forEach(['delete', 'get', 'head', 'options'], function forEachMethodNoData(method) {\n /*eslint func-names:0*/\n Axios.prototype[method] = function(url, config) {\n return this.request(mergeConfig(config || {}, {\n method: method,\n url: url,\n data: (config || {}).data\n }));\n };\n});\n\nutils.forEach(['post', 'put', 'patch'], function forEachMethodWithData(method) {\n /*eslint func-names:0*/\n\n function generateHTTPMethod(isForm) {\n return function httpMethod(url, data, config) {\n return this.request(mergeConfig(config || {}, {\n method: method,\n headers: isForm ? {\n 'Content-Type': 'multipart/form-data'\n } : {},\n url: url,\n data: data\n }));\n };\n }\n\n Axios.prototype[method] = generateHTTPMethod();\n\n Axios.prototype[method + 'Form'] = generateHTTPMethod(true);\n});\n\nmodule.exports = Axios;\n","'use strict';\n\nvar CanceledError = require('./CanceledError');\n\n/**\n * A `CancelToken` is an object that can be used to request cancellation of an operation.\n *\n * @class\n * @param {Function} executor The executor function.\n */\nfunction CancelToken(executor) {\n if (typeof executor !== 'function') {\n throw new TypeError('executor must be a function.');\n }\n\n var resolvePromise;\n\n this.promise = new Promise(function promiseExecutor(resolve) {\n resolvePromise = resolve;\n });\n\n var token = this;\n\n // eslint-disable-next-line func-names\n this.promise.then(function(cancel) {\n if (!token._listeners) return;\n\n var i;\n var l = token._listeners.length;\n\n for (i = 0; i < l; i++) {\n token._listeners[i](cancel);\n }\n token._listeners = null;\n });\n\n // eslint-disable-next-line func-names\n this.promise.then = function(onfulfilled) {\n var _resolve;\n // eslint-disable-next-line func-names\n var promise = new Promise(function(resolve) {\n token.subscribe(resolve);\n _resolve = resolve;\n }).then(onfulfilled);\n\n promise.cancel = function reject() {\n token.unsubscribe(_resolve);\n };\n\n return promise;\n };\n\n executor(function cancel(message) {\n if (token.reason) {\n // Cancellation has already been requested\n return;\n }\n\n token.reason = new CanceledError(message);\n resolvePromise(token.reason);\n });\n}\n\n/**\n * Throws a `CanceledError` if cancellation has been requested.\n */\nCancelToken.prototype.throwIfRequested = function throwIfRequested() {\n if (this.reason) {\n throw this.reason;\n }\n};\n\n/**\n * Subscribe to the cancel signal\n */\n\nCancelToken.prototype.subscribe = function subscribe(listener) {\n if (this.reason) {\n listener(this.reason);\n return;\n }\n\n if (this._listeners) {\n this._listeners.push(listener);\n } else {\n this._listeners = [listener];\n }\n};\n\n/**\n * Unsubscribe from the cancel signal\n */\n\nCancelToken.prototype.unsubscribe = function unsubscribe(listener) {\n if (!this._listeners) {\n return;\n }\n var index = this._listeners.indexOf(listener);\n if (index !== -1) {\n this._listeners.splice(index, 1);\n }\n};\n\n/**\n * Returns an object that contains a new `CancelToken` and a function that, when called,\n * cancels the `CancelToken`.\n */\nCancelToken.source = function source() {\n var cancel;\n var token = new CancelToken(function executor(c) {\n cancel = c;\n });\n return {\n token: token,\n cancel: cancel\n };\n};\n\nmodule.exports = CancelToken;\n","'use strict';\n\n/**\n * Syntactic sugar for invoking a function and expanding an array for arguments.\n *\n * Common use case would be to use `Function.prototype.apply`.\n *\n * ```js\n * function f(x, y, z) {}\n * var args = [1, 2, 3];\n * f.apply(null, args);\n * ```\n *\n * With `spread` this example can be re-written.\n *\n * ```js\n * spread(function(x, y, z) {})([1, 2, 3]);\n * ```\n *\n * @param {Function} callback\n * @returns {Function}\n */\nmodule.exports = function spread(callback) {\n return function wrap(arr) {\n return callback.apply(null, arr);\n };\n};\n","'use strict';\n\nvar utils = require('./../utils');\n\n/**\n * Determines whether the payload is an error thrown by Axios\n *\n * @param {*} payload The value to test\n * @returns {boolean} True if the payload is an error thrown by Axios, otherwise false\n */\nmodule.exports = function isAxiosError(payload) {\n return utils.isObject(payload) && (payload.isAxiosError === true);\n};\n","'use strict';\n\nvar utils = require('./utils');\nvar bind = require('./helpers/bind');\nvar Axios = require('./core/Axios');\nvar mergeConfig = require('./core/mergeConfig');\nvar defaults = require('./defaults');\n\n/**\n * Create an instance of Axios\n *\n * @param {Object} defaultConfig The default config for the instance\n * @return {Axios} A new instance of Axios\n */\nfunction createInstance(defaultConfig) {\n var context = new Axios(defaultConfig);\n var instance = bind(Axios.prototype.request, context);\n\n // Copy axios.prototype to instance\n utils.extend(instance, Axios.prototype, context);\n\n // Copy context to instance\n utils.extend(instance, context);\n\n // Factory for creating new instances\n instance.create = function create(instanceConfig) {\n return createInstance(mergeConfig(defaultConfig, instanceConfig));\n };\n\n return instance;\n}\n\n// Create the default instance to be exported\nvar axios = createInstance(defaults);\n\n// Expose Axios class to allow class inheritance\naxios.Axios = Axios;\n\n// Expose Cancel & CancelToken\naxios.CanceledError = require('./cancel/CanceledError');\naxios.CancelToken = require('./cancel/CancelToken');\naxios.isCancel = require('./cancel/isCancel');\naxios.VERSION = require('./env/data').version;\naxios.toFormData = require('./helpers/toFormData');\n\n// Expose AxiosError class\naxios.AxiosError = require('../lib/core/AxiosError');\n\n// alias for CanceledError for backward compatibility\naxios.Cancel = axios.CanceledError;\n\n// Expose all/spread\naxios.all = function all(promises) {\n return Promise.all(promises);\n};\naxios.spread = require('./helpers/spread');\n\n// Expose isAxiosError\naxios.isAxiosError = require('./helpers/isAxiosError');\n\nmodule.exports = axios;\n\n// Allow use of default import syntax in TypeScript\nmodule.exports.default = axios;\n","module.exports = require('./lib/axios');","export const appSettings = {\n backEnd: {\n baseUrlAndPath: `${import.meta.env.VITE_APP_BACKEND_BASE_URL}/${\n import.meta.env.VITE_APP_BACKEND_BASE_PATH\n }`,\n },\n openId: {\n baseUrl: import.meta.env.VITE_APP_OPENID_BASE_URL,\n },\n i18n: {\n locale: import.meta.env.VITE_APP_I18N_LOCALE || \"en\",\n fallbackLocale: import.meta.env.VITE_APP_I18N_FALLBACK_LOCALE || \"en\",\n },\n environment: {\n isProduction: import.meta.env.PROD,\n },\n};\n\nexport default appSettings;\n","export default {}",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory();\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\troot.CryptoJS = factory();\n\t}\n}(this, function () {\n\n\t/*globals window, global, require*/\n\n\t/**\n\t * CryptoJS core components.\n\t */\n\tvar CryptoJS = CryptoJS || (function (Math, undefined) {\n\n\t var crypto;\n\n\t // Native crypto from window (Browser)\n\t if (typeof window !== 'undefined' && window.crypto) {\n\t crypto = window.crypto;\n\t }\n\n\t // Native crypto in web worker (Browser)\n\t if (typeof self !== 'undefined' && self.crypto) {\n\t crypto = self.crypto;\n\t }\n\n\t // Native crypto from worker\n\t if (typeof globalThis !== 'undefined' && globalThis.crypto) {\n\t crypto = globalThis.crypto;\n\t }\n\n\t // Native (experimental IE 11) crypto from window (Browser)\n\t if (!crypto && typeof window !== 'undefined' && window.msCrypto) {\n\t crypto = window.msCrypto;\n\t }\n\n\t // Native crypto from global (NodeJS)\n\t if (!crypto && typeof global !== 'undefined' && global.crypto) {\n\t crypto = global.crypto;\n\t }\n\n\t // Native crypto import via require (NodeJS)\n\t if (!crypto && typeof require === 'function') {\n\t try {\n\t crypto = require('crypto');\n\t } catch (err) {}\n\t }\n\n\t /*\n\t * Cryptographically secure pseudorandom number generator\n\t *\n\t * As Math.random() is cryptographically not safe to use\n\t */\n\t var cryptoSecureRandomInt = function () {\n\t if (crypto) {\n\t // Use getRandomValues method (Browser)\n\t if (typeof crypto.getRandomValues === 'function') {\n\t try {\n\t return crypto.getRandomValues(new Uint32Array(1))[0];\n\t } catch (err) {}\n\t }\n\n\t // Use randomBytes method (NodeJS)\n\t if (typeof crypto.randomBytes === 'function') {\n\t try {\n\t return crypto.randomBytes(4).readInt32LE();\n\t } catch (err) {}\n\t }\n\t }\n\n\t throw new Error('Native crypto module could not be used to get secure random number.');\n\t };\n\n\t /*\n\t * Local polyfill of Object.create\n\n\t */\n\t var create = Object.create || (function () {\n\t function F() {}\n\n\t return function (obj) {\n\t var subtype;\n\n\t F.prototype = obj;\n\n\t subtype = new F();\n\n\t F.prototype = null;\n\n\t return subtype;\n\t };\n\t }());\n\n\t /**\n\t * CryptoJS namespace.\n\t */\n\t var C = {};\n\n\t /**\n\t * Library namespace.\n\t */\n\t var C_lib = C.lib = {};\n\n\t /**\n\t * Base object for prototypal inheritance.\n\t */\n\t var Base = C_lib.Base = (function () {\n\n\n\t return {\n\t /**\n\t * Creates a new object that inherits from this object.\n\t *\n\t * @param {Object} overrides Properties to copy into the new object.\n\t *\n\t * @return {Object} The new object.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var MyType = CryptoJS.lib.Base.extend({\n\t * field: 'value',\n\t *\n\t * method: function () {\n\t * }\n\t * });\n\t */\n\t extend: function (overrides) {\n\t // Spawn\n\t var subtype = create(this);\n\n\t // Augment\n\t if (overrides) {\n\t subtype.mixIn(overrides);\n\t }\n\n\t // Create default initializer\n\t if (!subtype.hasOwnProperty('init') || this.init === subtype.init) {\n\t subtype.init = function () {\n\t subtype.$super.init.apply(this, arguments);\n\t };\n\t }\n\n\t // Initializer's prototype is the subtype object\n\t subtype.init.prototype = subtype;\n\n\t // Reference supertype\n\t subtype.$super = this;\n\n\t return subtype;\n\t },\n\n\t /**\n\t * Extends this object and runs the init method.\n\t * Arguments to create() will be passed to init().\n\t *\n\t * @return {Object} The new object.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var instance = MyType.create();\n\t */\n\t create: function () {\n\t var instance = this.extend();\n\t instance.init.apply(instance, arguments);\n\n\t return instance;\n\t },\n\n\t /**\n\t * Initializes a newly created object.\n\t * Override this method to add some logic when your objects are created.\n\t *\n\t * @example\n\t *\n\t * var MyType = CryptoJS.lib.Base.extend({\n\t * init: function () {\n\t * // ...\n\t * }\n\t * });\n\t */\n\t init: function () {\n\t },\n\n\t /**\n\t * Copies properties into this object.\n\t *\n\t * @param {Object} properties The properties to mix in.\n\t *\n\t * @example\n\t *\n\t * MyType.mixIn({\n\t * field: 'value'\n\t * });\n\t */\n\t mixIn: function (properties) {\n\t for (var propertyName in properties) {\n\t if (properties.hasOwnProperty(propertyName)) {\n\t this[propertyName] = properties[propertyName];\n\t }\n\t }\n\n\t // IE won't copy toString using the loop above\n\t if (properties.hasOwnProperty('toString')) {\n\t this.toString = properties.toString;\n\t }\n\t },\n\n\t /**\n\t * Creates a copy of this object.\n\t *\n\t * @return {Object} The clone.\n\t *\n\t * @example\n\t *\n\t * var clone = instance.clone();\n\t */\n\t clone: function () {\n\t return this.init.prototype.extend(this);\n\t }\n\t };\n\t }());\n\n\t /**\n\t * An array of 32-bit words.\n\t *\n\t * @property {Array} words The array of 32-bit words.\n\t * @property {number} sigBytes The number of significant bytes in this word array.\n\t */\n\t var WordArray = C_lib.WordArray = Base.extend({\n\t /**\n\t * Initializes a newly created word array.\n\t *\n\t * @param {Array} words (Optional) An array of 32-bit words.\n\t * @param {number} sigBytes (Optional) The number of significant bytes in the words.\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.lib.WordArray.create();\n\t * var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);\n\t * var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);\n\t */\n\t init: function (words, sigBytes) {\n\t words = this.words = words || [];\n\n\t if (sigBytes != undefined) {\n\t this.sigBytes = sigBytes;\n\t } else {\n\t this.sigBytes = words.length * 4;\n\t }\n\t },\n\n\t /**\n\t * Converts this word array to a string.\n\t *\n\t * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex\n\t *\n\t * @return {string} The stringified word array.\n\t *\n\t * @example\n\t *\n\t * var string = wordArray + '';\n\t * var string = wordArray.toString();\n\t * var string = wordArray.toString(CryptoJS.enc.Utf8);\n\t */\n\t toString: function (encoder) {\n\t return (encoder || Hex).stringify(this);\n\t },\n\n\t /**\n\t * Concatenates a word array to this word array.\n\t *\n\t * @param {WordArray} wordArray The word array to append.\n\t *\n\t * @return {WordArray} This word array.\n\t *\n\t * @example\n\t *\n\t * wordArray1.concat(wordArray2);\n\t */\n\t concat: function (wordArray) {\n\t // Shortcuts\n\t var thisWords = this.words;\n\t var thatWords = wordArray.words;\n\t var thisSigBytes = this.sigBytes;\n\t var thatSigBytes = wordArray.sigBytes;\n\n\t // Clamp excess bits\n\t this.clamp();\n\n\t // Concat\n\t if (thisSigBytes % 4) {\n\t // Copy one byte at a time\n\t for (var i = 0; i < thatSigBytes; i++) {\n\t var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);\n\t }\n\t } else {\n\t // Copy one word at a time\n\t for (var j = 0; j < thatSigBytes; j += 4) {\n\t thisWords[(thisSigBytes + j) >>> 2] = thatWords[j >>> 2];\n\t }\n\t }\n\t this.sigBytes += thatSigBytes;\n\n\t // Chainable\n\t return this;\n\t },\n\n\t /**\n\t * Removes insignificant bits.\n\t *\n\t * @example\n\t *\n\t * wordArray.clamp();\n\t */\n\t clamp: function () {\n\t // Shortcuts\n\t var words = this.words;\n\t var sigBytes = this.sigBytes;\n\n\t // Clamp\n\t words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);\n\t words.length = Math.ceil(sigBytes / 4);\n\t },\n\n\t /**\n\t * Creates a copy of this word array.\n\t *\n\t * @return {WordArray} The clone.\n\t *\n\t * @example\n\t *\n\t * var clone = wordArray.clone();\n\t */\n\t clone: function () {\n\t var clone = Base.clone.call(this);\n\t clone.words = this.words.slice(0);\n\n\t return clone;\n\t },\n\n\t /**\n\t * Creates a word array filled with random bytes.\n\t *\n\t * @param {number} nBytes The number of random bytes to generate.\n\t *\n\t * @return {WordArray} The random word array.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.lib.WordArray.random(16);\n\t */\n\t random: function (nBytes) {\n\t var words = [];\n\n\t for (var i = 0; i < nBytes; i += 4) {\n\t words.push(cryptoSecureRandomInt());\n\t }\n\n\t return new WordArray.init(words, nBytes);\n\t }\n\t });\n\n\t /**\n\t * Encoder namespace.\n\t */\n\t var C_enc = C.enc = {};\n\n\t /**\n\t * Hex encoding strategy.\n\t */\n\t var Hex = C_enc.Hex = {\n\t /**\n\t * Converts a word array to a hex string.\n\t *\n\t * @param {WordArray} wordArray The word array.\n\t *\n\t * @return {string} The hex string.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var hexString = CryptoJS.enc.Hex.stringify(wordArray);\n\t */\n\t stringify: function (wordArray) {\n\t // Shortcuts\n\t var words = wordArray.words;\n\t var sigBytes = wordArray.sigBytes;\n\n\t // Convert\n\t var hexChars = [];\n\t for (var i = 0; i < sigBytes; i++) {\n\t var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t hexChars.push((bite >>> 4).toString(16));\n\t hexChars.push((bite & 0x0f).toString(16));\n\t }\n\n\t return hexChars.join('');\n\t },\n\n\t /**\n\t * Converts a hex string to a word array.\n\t *\n\t * @param {string} hexStr The hex string.\n\t *\n\t * @return {WordArray} The word array.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.enc.Hex.parse(hexString);\n\t */\n\t parse: function (hexStr) {\n\t // Shortcut\n\t var hexStrLength = hexStr.length;\n\n\t // Convert\n\t var words = [];\n\t for (var i = 0; i < hexStrLength; i += 2) {\n\t words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);\n\t }\n\n\t return new WordArray.init(words, hexStrLength / 2);\n\t }\n\t };\n\n\t /**\n\t * Latin1 encoding strategy.\n\t */\n\t var Latin1 = C_enc.Latin1 = {\n\t /**\n\t * Converts a word array to a Latin1 string.\n\t *\n\t * @param {WordArray} wordArray The word array.\n\t *\n\t * @return {string} The Latin1 string.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);\n\t */\n\t stringify: function (wordArray) {\n\t // Shortcuts\n\t var words = wordArray.words;\n\t var sigBytes = wordArray.sigBytes;\n\n\t // Convert\n\t var latin1Chars = [];\n\t for (var i = 0; i < sigBytes; i++) {\n\t var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t latin1Chars.push(String.fromCharCode(bite));\n\t }\n\n\t return latin1Chars.join('');\n\t },\n\n\t /**\n\t * Converts a Latin1 string to a word array.\n\t *\n\t * @param {string} latin1Str The Latin1 string.\n\t *\n\t * @return {WordArray} The word array.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.enc.Latin1.parse(latin1String);\n\t */\n\t parse: function (latin1Str) {\n\t // Shortcut\n\t var latin1StrLength = latin1Str.length;\n\n\t // Convert\n\t var words = [];\n\t for (var i = 0; i < latin1StrLength; i++) {\n\t words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);\n\t }\n\n\t return new WordArray.init(words, latin1StrLength);\n\t }\n\t };\n\n\t /**\n\t * UTF-8 encoding strategy.\n\t */\n\t var Utf8 = C_enc.Utf8 = {\n\t /**\n\t * Converts a word array to a UTF-8 string.\n\t *\n\t * @param {WordArray} wordArray The word array.\n\t *\n\t * @return {string} The UTF-8 string.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);\n\t */\n\t stringify: function (wordArray) {\n\t try {\n\t return decodeURIComponent(escape(Latin1.stringify(wordArray)));\n\t } catch (e) {\n\t throw new Error('Malformed UTF-8 data');\n\t }\n\t },\n\n\t /**\n\t * Converts a UTF-8 string to a word array.\n\t *\n\t * @param {string} utf8Str The UTF-8 string.\n\t *\n\t * @return {WordArray} The word array.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.enc.Utf8.parse(utf8String);\n\t */\n\t parse: function (utf8Str) {\n\t return Latin1.parse(unescape(encodeURIComponent(utf8Str)));\n\t }\n\t };\n\n\t /**\n\t * Abstract buffered block algorithm template.\n\t *\n\t * The property blockSize must be implemented in a concrete subtype.\n\t *\n\t * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0\n\t */\n\t var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({\n\t /**\n\t * Resets this block algorithm's data buffer to its initial state.\n\t *\n\t * @example\n\t *\n\t * bufferedBlockAlgorithm.reset();\n\t */\n\t reset: function () {\n\t // Initial values\n\t this._data = new WordArray.init();\n\t this._nDataBytes = 0;\n\t },\n\n\t /**\n\t * Adds new data to this block algorithm's buffer.\n\t *\n\t * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.\n\t *\n\t * @example\n\t *\n\t * bufferedBlockAlgorithm._append('data');\n\t * bufferedBlockAlgorithm._append(wordArray);\n\t */\n\t _append: function (data) {\n\t // Convert string to WordArray, else assume WordArray already\n\t if (typeof data == 'string') {\n\t data = Utf8.parse(data);\n\t }\n\n\t // Append\n\t this._data.concat(data);\n\t this._nDataBytes += data.sigBytes;\n\t },\n\n\t /**\n\t * Processes available data blocks.\n\t *\n\t * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.\n\t *\n\t * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.\n\t *\n\t * @return {WordArray} The processed data.\n\t *\n\t * @example\n\t *\n\t * var processedData = bufferedBlockAlgorithm._process();\n\t * var processedData = bufferedBlockAlgorithm._process(!!'flush');\n\t */\n\t _process: function (doFlush) {\n\t var processedWords;\n\n\t // Shortcuts\n\t var data = this._data;\n\t var dataWords = data.words;\n\t var dataSigBytes = data.sigBytes;\n\t var blockSize = this.blockSize;\n\t var blockSizeBytes = blockSize * 4;\n\n\t // Count blocks ready\n\t var nBlocksReady = dataSigBytes / blockSizeBytes;\n\t if (doFlush) {\n\t // Round up to include partial blocks\n\t nBlocksReady = Math.ceil(nBlocksReady);\n\t } else {\n\t // Round down to include only full blocks,\n\t // less the number of blocks that must remain in the buffer\n\t nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);\n\t }\n\n\t // Count words ready\n\t var nWordsReady = nBlocksReady * blockSize;\n\n\t // Count bytes ready\n\t var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);\n\n\t // Process blocks\n\t if (nWordsReady) {\n\t for (var offset = 0; offset < nWordsReady; offset += blockSize) {\n\t // Perform concrete-algorithm logic\n\t this._doProcessBlock(dataWords, offset);\n\t }\n\n\t // Remove processed words\n\t processedWords = dataWords.splice(0, nWordsReady);\n\t data.sigBytes -= nBytesReady;\n\t }\n\n\t // Return processed words\n\t return new WordArray.init(processedWords, nBytesReady);\n\t },\n\n\t /**\n\t * Creates a copy of this object.\n\t *\n\t * @return {Object} The clone.\n\t *\n\t * @example\n\t *\n\t * var clone = bufferedBlockAlgorithm.clone();\n\t */\n\t clone: function () {\n\t var clone = Base.clone.call(this);\n\t clone._data = this._data.clone();\n\n\t return clone;\n\t },\n\n\t _minBufferSize: 0\n\t });\n\n\t /**\n\t * Abstract hasher template.\n\t *\n\t * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)\n\t */\n\t var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({\n\t /**\n\t * Configuration options.\n\t */\n\t cfg: Base.extend(),\n\n\t /**\n\t * Initializes a newly created hasher.\n\t *\n\t * @param {Object} cfg (Optional) The configuration options to use for this hash computation.\n\t *\n\t * @example\n\t *\n\t * var hasher = CryptoJS.algo.SHA256.create();\n\t */\n\t init: function (cfg) {\n\t // Apply config defaults\n\t this.cfg = this.cfg.extend(cfg);\n\n\t // Set initial values\n\t this.reset();\n\t },\n\n\t /**\n\t * Resets this hasher to its initial state.\n\t *\n\t * @example\n\t *\n\t * hasher.reset();\n\t */\n\t reset: function () {\n\t // Reset data buffer\n\t BufferedBlockAlgorithm.reset.call(this);\n\n\t // Perform concrete-hasher logic\n\t this._doReset();\n\t },\n\n\t /**\n\t * Updates this hasher with a message.\n\t *\n\t * @param {WordArray|string} messageUpdate The message to append.\n\t *\n\t * @return {Hasher} This hasher.\n\t *\n\t * @example\n\t *\n\t * hasher.update('message');\n\t * hasher.update(wordArray);\n\t */\n\t update: function (messageUpdate) {\n\t // Append\n\t this._append(messageUpdate);\n\n\t // Update the hash\n\t this._process();\n\n\t // Chainable\n\t return this;\n\t },\n\n\t /**\n\t * Finalizes the hash computation.\n\t * Note that the finalize operation is effectively a destructive, read-once operation.\n\t *\n\t * @param {WordArray|string} messageUpdate (Optional) A final message update.\n\t *\n\t * @return {WordArray} The hash.\n\t *\n\t * @example\n\t *\n\t * var hash = hasher.finalize();\n\t * var hash = hasher.finalize('message');\n\t * var hash = hasher.finalize(wordArray);\n\t */\n\t finalize: function (messageUpdate) {\n\t // Final message update\n\t if (messageUpdate) {\n\t this._append(messageUpdate);\n\t }\n\n\t // Perform concrete-hasher logic\n\t var hash = this._doFinalize();\n\n\t return hash;\n\t },\n\n\t blockSize: 512/32,\n\n\t /**\n\t * Creates a shortcut function to a hasher's object interface.\n\t *\n\t * @param {Hasher} hasher The hasher to create a helper for.\n\t *\n\t * @return {Function} The shortcut function.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);\n\t */\n\t _createHelper: function (hasher) {\n\t return function (message, cfg) {\n\t return new hasher.init(cfg).finalize(message);\n\t };\n\t },\n\n\t /**\n\t * Creates a shortcut function to the HMAC's object interface.\n\t *\n\t * @param {Hasher} hasher The hasher to use in this HMAC helper.\n\t *\n\t * @return {Function} The shortcut function.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);\n\t */\n\t _createHmacHelper: function (hasher) {\n\t return function (message, key) {\n\t return new C_algo.HMAC.init(hasher, key).finalize(message);\n\t };\n\t }\n\t });\n\n\t /**\n\t * Algorithm namespace.\n\t */\n\t var C_algo = C.algo = {};\n\n\t return C;\n\t}(Math));\n\n\n\treturn CryptoJS;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function (Math) {\n\t // Shortcuts\n\t var C = CryptoJS;\n\t var C_lib = C.lib;\n\t var WordArray = C_lib.WordArray;\n\t var Hasher = C_lib.Hasher;\n\t var C_algo = C.algo;\n\n\t // Initialization and round constants tables\n\t var H = [];\n\t var K = [];\n\n\t // Compute constants\n\t (function () {\n\t function isPrime(n) {\n\t var sqrtN = Math.sqrt(n);\n\t for (var factor = 2; factor <= sqrtN; factor++) {\n\t if (!(n % factor)) {\n\t return false;\n\t }\n\t }\n\n\t return true;\n\t }\n\n\t function getFractionalBits(n) {\n\t return ((n - (n | 0)) * 0x100000000) | 0;\n\t }\n\n\t var n = 2;\n\t var nPrime = 0;\n\t while (nPrime < 64) {\n\t if (isPrime(n)) {\n\t if (nPrime < 8) {\n\t H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));\n\t }\n\t K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));\n\n\t nPrime++;\n\t }\n\n\t n++;\n\t }\n\t }());\n\n\t // Reusable object\n\t var W = [];\n\n\t /**\n\t * SHA-256 hash algorithm.\n\t */\n\t var SHA256 = C_algo.SHA256 = Hasher.extend({\n\t _doReset: function () {\n\t this._hash = new WordArray.init(H.slice(0));\n\t },\n\n\t _doProcessBlock: function (M, offset) {\n\t // Shortcut\n\t var H = this._hash.words;\n\n\t // Working variables\n\t var a = H[0];\n\t var b = H[1];\n\t var c = H[2];\n\t var d = H[3];\n\t var e = H[4];\n\t var f = H[5];\n\t var g = H[6];\n\t var h = H[7];\n\n\t // Computation\n\t for (var i = 0; i < 64; i++) {\n\t if (i < 16) {\n\t W[i] = M[offset + i] | 0;\n\t } else {\n\t var gamma0x = W[i - 15];\n\t var gamma0 = ((gamma0x << 25) | (gamma0x >>> 7)) ^\n\t ((gamma0x << 14) | (gamma0x >>> 18)) ^\n\t (gamma0x >>> 3);\n\n\t var gamma1x = W[i - 2];\n\t var gamma1 = ((gamma1x << 15) | (gamma1x >>> 17)) ^\n\t ((gamma1x << 13) | (gamma1x >>> 19)) ^\n\t (gamma1x >>> 10);\n\n\t W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];\n\t }\n\n\t var ch = (e & f) ^ (~e & g);\n\t var maj = (a & b) ^ (a & c) ^ (b & c);\n\n\t var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));\n\t var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7) | (e >>> 25));\n\n\t var t1 = h + sigma1 + ch + K[i] + W[i];\n\t var t2 = sigma0 + maj;\n\n\t h = g;\n\t g = f;\n\t f = e;\n\t e = (d + t1) | 0;\n\t d = c;\n\t c = b;\n\t b = a;\n\t a = (t1 + t2) | 0;\n\t }\n\n\t // Intermediate hash value\n\t H[0] = (H[0] + a) | 0;\n\t H[1] = (H[1] + b) | 0;\n\t H[2] = (H[2] + c) | 0;\n\t H[3] = (H[3] + d) | 0;\n\t H[4] = (H[4] + e) | 0;\n\t H[5] = (H[5] + f) | 0;\n\t H[6] = (H[6] + g) | 0;\n\t H[7] = (H[7] + h) | 0;\n\t },\n\n\t _doFinalize: function () {\n\t // Shortcuts\n\t var data = this._data;\n\t var dataWords = data.words;\n\n\t var nBitsTotal = this._nDataBytes * 8;\n\t var nBitsLeft = data.sigBytes * 8;\n\n\t // Add padding\n\t dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);\n\t dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);\n\t dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;\n\t data.sigBytes = dataWords.length * 4;\n\n\t // Hash final blocks\n\t this._process();\n\n\t // Return final computed hash\n\t return this._hash;\n\t },\n\n\t clone: function () {\n\t var clone = Hasher.clone.call(this);\n\t clone._hash = this._hash.clone();\n\n\t return clone;\n\t }\n\t });\n\n\t /**\n\t * Shortcut function to the hasher's object interface.\n\t *\n\t * @param {WordArray|string} message The message to hash.\n\t *\n\t * @return {WordArray} The hash.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var hash = CryptoJS.SHA256('message');\n\t * var hash = CryptoJS.SHA256(wordArray);\n\t */\n\t C.SHA256 = Hasher._createHelper(SHA256);\n\n\t /**\n\t * Shortcut function to the HMAC's object interface.\n\t *\n\t * @param {WordArray|string} message The message to hash.\n\t * @param {WordArray|string} key The secret key.\n\t *\n\t * @return {WordArray} The HMAC.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var hmac = CryptoJS.HmacSHA256(message, key);\n\t */\n\t C.HmacSHA256 = Hasher._createHmacHelper(SHA256);\n\t}(Math));\n\n\n\treturn CryptoJS.SHA256;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\t(function () {\n\t // Shortcuts\n\t var C = CryptoJS;\n\t var C_lib = C.lib;\n\t var WordArray = C_lib.WordArray;\n\t var C_enc = C.enc;\n\n\t /**\n\t * Base64 encoding strategy.\n\t */\n\t var Base64 = C_enc.Base64 = {\n\t /**\n\t * Converts a word array to a Base64 string.\n\t *\n\t * @param {WordArray} wordArray The word array.\n\t *\n\t * @return {string} The Base64 string.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var base64String = CryptoJS.enc.Base64.stringify(wordArray);\n\t */\n\t stringify: function (wordArray) {\n\t // Shortcuts\n\t var words = wordArray.words;\n\t var sigBytes = wordArray.sigBytes;\n\t var map = this._map;\n\n\t // Clamp excess bits\n\t wordArray.clamp();\n\n\t // Convert\n\t var base64Chars = [];\n\t for (var i = 0; i < sigBytes; i += 3) {\n\t var byte1 = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;\n\t var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;\n\t var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;\n\n\t var triplet = (byte1 << 16) | (byte2 << 8) | byte3;\n\n\t for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {\n\t base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));\n\t }\n\t }\n\n\t // Add padding\n\t var paddingChar = map.charAt(64);\n\t if (paddingChar) {\n\t while (base64Chars.length % 4) {\n\t base64Chars.push(paddingChar);\n\t }\n\t }\n\n\t return base64Chars.join('');\n\t },\n\n\t /**\n\t * Converts a Base64 string to a word array.\n\t *\n\t * @param {string} base64Str The Base64 string.\n\t *\n\t * @return {WordArray} The word array.\n\t *\n\t * @static\n\t *\n\t * @example\n\t *\n\t * var wordArray = CryptoJS.enc.Base64.parse(base64String);\n\t */\n\t parse: function (base64Str) {\n\t // Shortcuts\n\t var base64StrLength = base64Str.length;\n\t var map = this._map;\n\t var reverseMap = this._reverseMap;\n\n\t if (!reverseMap) {\n\t reverseMap = this._reverseMap = [];\n\t for (var j = 0; j < map.length; j++) {\n\t reverseMap[map.charCodeAt(j)] = j;\n\t }\n\t }\n\n\t // Ignore padding\n\t var paddingChar = map.charAt(64);\n\t if (paddingChar) {\n\t var paddingIndex = base64Str.indexOf(paddingChar);\n\t if (paddingIndex !== -1) {\n\t base64StrLength = paddingIndex;\n\t }\n\t }\n\n\t // Convert\n\t return parseLoop(base64Str, base64StrLength, reverseMap);\n\n\t },\n\n\t _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='\n\t };\n\n\t function parseLoop(base64Str, base64StrLength, reverseMap) {\n\t var words = [];\n\t var nBytes = 0;\n\t for (var i = 0; i < base64StrLength; i++) {\n\t if (i % 4) {\n\t var bits1 = reverseMap[base64Str.charCodeAt(i - 1)] << ((i % 4) * 2);\n\t var bits2 = reverseMap[base64Str.charCodeAt(i)] >>> (6 - (i % 4) * 2);\n\t var bitsCombined = bits1 | bits2;\n\t words[nBytes >>> 2] |= bitsCombined << (24 - (nBytes % 4) * 8);\n\t nBytes++;\n\t }\n\t }\n\t return WordArray.create(words, nBytes);\n\t }\n\t}());\n\n\n\treturn CryptoJS.enc.Base64;\n\n}));",";(function (root, factory) {\n\tif (typeof exports === \"object\") {\n\t\t// CommonJS\n\t\tmodule.exports = exports = factory(require(\"./core\"));\n\t}\n\telse if (typeof define === \"function\" && define.amd) {\n\t\t// AMD\n\t\tdefine([\"./core\"], factory);\n\t}\n\telse {\n\t\t// Global (browser)\n\t\tfactory(root.CryptoJS);\n\t}\n}(this, function (CryptoJS) {\n\n\treturn CryptoJS.enc.Utf8;\n\n}));","function e(e){this.message=e}e.prototype=new Error,e.prototype.name=\"InvalidCharacterError\";var r=\"undefined\"!=typeof window&&window.atob&&window.atob.bind(window)||function(r){var t=String(r).replace(/=+$/,\"\");if(t.length%4==1)throw new e(\"'atob' failed: The string to be decoded is not correctly encoded.\");for(var n,o,a=0,i=0,c=\"\";o=t.charAt(i++);~o&&(n=a%4?64*n+o:o,a++%4)?c+=String.fromCharCode(255&n>>(-2*a&6)):0)o=\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\".indexOf(o);return c};function t(e){var t=e.replace(/-/g,\"+\").replace(/_/g,\"/\");switch(t.length%4){case 0:break;case 2:t+=\"==\";break;case 3:t+=\"=\";break;default:throw\"Illegal base64url string!\"}try{return function(e){return decodeURIComponent(r(e).replace(/(.)/g,(function(e,r){var t=r.charCodeAt(0).toString(16).toUpperCase();return t.length<2&&(t=\"0\"+t),\"%\"+t})))}(t)}catch(e){return r(t)}}function n(e){this.message=e}function o(e,r){if(\"string\"!=typeof e)throw new n(\"Invalid token specified\");var o=!0===(r=r||{}).header?0:1;try{return JSON.parse(t(e.split(\".\")[o]))}catch(e){throw new n(\"Invalid token specified: \"+e.message)}}n.prototype=new Error,n.prototype.name=\"InvalidTokenError\";export default o;export{n as InvalidTokenError};\n//# sourceMappingURL=jwt-decode.esm.js.map\n","// src/utils/CryptoUtils.ts\nimport CryptoJS from \"crypto-js/core.js\";\nimport sha256 from \"crypto-js/sha256.js\";\nimport Base64 from \"crypto-js/enc-base64.js\";\nimport Utf8 from \"crypto-js/enc-utf8.js\";\n\n// src/utils/Logger.ts\nvar nopLogger = {\n debug: () => void 0,\n info: () => void 0,\n warn: () => void 0,\n error: () => void 0\n};\nvar level;\nvar logger;\nvar Log = /* @__PURE__ */ ((Log2) => {\n Log2[Log2[\"NONE\"] = 0] = \"NONE\";\n Log2[Log2[\"ERROR\"] = 1] = \"ERROR\";\n Log2[Log2[\"WARN\"] = 2] = \"WARN\";\n Log2[Log2[\"INFO\"] = 3] = \"INFO\";\n Log2[Log2[\"DEBUG\"] = 4] = \"DEBUG\";\n return Log2;\n})(Log || {});\n((Log2) => {\n function reset() {\n level = 3 /* INFO */;\n logger = nopLogger;\n }\n Log2.reset = reset;\n function setLevel(value) {\n if (!(0 /* NONE */ <= value && value <= 4 /* DEBUG */)) {\n throw new Error(\"Invalid log level\");\n }\n level = value;\n }\n Log2.setLevel = setLevel;\n function setLogger(value) {\n logger = value;\n }\n Log2.setLogger = setLogger;\n})(Log || (Log = {}));\nvar Logger = class {\n constructor(_name) {\n this._name = _name;\n }\n debug(...args) {\n if (level >= 4 /* DEBUG */) {\n logger.debug(Logger._format(this._name, this._method), ...args);\n }\n }\n info(...args) {\n if (level >= 3 /* INFO */) {\n logger.info(Logger._format(this._name, this._method), ...args);\n }\n }\n warn(...args) {\n if (level >= 2 /* WARN */) {\n logger.warn(Logger._format(this._name, this._method), ...args);\n }\n }\n error(...args) {\n if (level >= 1 /* ERROR */) {\n logger.error(Logger._format(this._name, this._method), ...args);\n }\n }\n throw(err) {\n this.error(err);\n throw err;\n }\n create(method) {\n const methodLogger = Object.create(this);\n methodLogger._method = method;\n methodLogger.debug(\"begin\");\n return methodLogger;\n }\n static createStatic(name, staticMethod) {\n const staticLogger = new Logger(`${name}.${staticMethod}`);\n staticLogger.debug(\"begin\");\n return staticLogger;\n }\n static _format(name, method) {\n const prefix = `[${name}]`;\n return method ? `${prefix} ${method}:` : prefix;\n }\n // helpers for static class methods\n static debug(name, ...args) {\n if (level >= 4 /* DEBUG */) {\n logger.debug(Logger._format(name), ...args);\n }\n }\n static info(name, ...args) {\n if (level >= 3 /* INFO */) {\n logger.info(Logger._format(name), ...args);\n }\n }\n static warn(name, ...args) {\n if (level >= 2 /* WARN */) {\n logger.warn(Logger._format(name), ...args);\n }\n }\n static error(name, ...args) {\n if (level >= 1 /* ERROR */) {\n logger.error(Logger._format(name), ...args);\n }\n }\n};\nLog.reset();\n\n// src/utils/CryptoUtils.ts\nvar UUID_V4_TEMPLATE = \"10000000-1000-4000-8000-100000000000\";\nvar CryptoUtils = class {\n static _randomWord() {\n return CryptoJS.lib.WordArray.random(1).words[0];\n }\n /**\n * Generates RFC4122 version 4 guid\n */\n static generateUUIDv4() {\n const uuid = UUID_V4_TEMPLATE.replace(\n /[018]/g,\n (c) => (+c ^ CryptoUtils._randomWord() & 15 >> +c / 4).toString(16)\n );\n return uuid.replace(/-/g, \"\");\n }\n /**\n * PKCE: Generate a code verifier\n */\n static generateCodeVerifier() {\n return CryptoUtils.generateUUIDv4() + CryptoUtils.generateUUIDv4() + CryptoUtils.generateUUIDv4();\n }\n /**\n * PKCE: Generate a code challenge\n */\n static generateCodeChallenge(code_verifier) {\n try {\n const hashed = sha256(code_verifier);\n return Base64.stringify(hashed).replace(/\\+/g, \"-\").replace(/\\//g, \"_\").replace(/=+$/, \"\");\n } catch (err) {\n Logger.error(\"CryptoUtils.generateCodeChallenge\", err);\n throw err;\n }\n }\n /**\n * Generates a base64-encoded string for a basic auth header\n */\n static generateBasicAuth(client_id, client_secret) {\n const basicAuth = Utf8.parse([client_id, client_secret].join(\":\"));\n return Base64.stringify(basicAuth);\n }\n};\n\n// src/utils/Event.ts\nvar Event = class {\n constructor(_name) {\n this._name = _name;\n this._logger = new Logger(`Event('${this._name}')`);\n this._callbacks = [];\n }\n addHandler(cb) {\n this._callbacks.push(cb);\n return () => this.removeHandler(cb);\n }\n removeHandler(cb) {\n const idx = this._callbacks.lastIndexOf(cb);\n if (idx >= 0) {\n this._callbacks.splice(idx, 1);\n }\n }\n raise(...ev) {\n this._logger.debug(\"raise:\", ...ev);\n for (const cb of this._callbacks) {\n void cb(...ev);\n }\n }\n};\n\n// src/utils/JwtUtils.ts\nimport jwt_decode from \"jwt-decode\";\nvar JwtUtils = class {\n // IMPORTANT: doesn't validate the token\n static decode(token) {\n try {\n return jwt_decode(token);\n } catch (err) {\n Logger.error(\"JwtUtils.decode\", err);\n throw err;\n }\n }\n};\n\n// src/utils/PopupUtils.ts\nvar PopupUtils = class {\n /**\n * Populates a map of window features with a placement centered in front of\n * the current window. If no explicit width is given, a default value is\n * binned into [800, 720, 600, 480, 360] based on the current window's width.\n */\n static center({ ...features }) {\n var _a, _b, _c;\n if (features.width == null)\n features.width = (_a = [800, 720, 600, 480].find((width) => width <= window.outerWidth / 1.618)) != null ? _a : 360;\n (_b = features.left) != null ? _b : features.left = Math.max(0, Math.round(window.screenX + (window.outerWidth - features.width) / 2));\n if (features.height != null)\n (_c = features.top) != null ? _c : features.top = Math.max(0, Math.round(window.screenY + (window.outerHeight - features.height) / 2));\n return features;\n }\n static serialize(features) {\n return Object.entries(features).filter(([, value]) => value != null).map(([key, value]) => `${key}=${typeof value !== \"boolean\" ? value : value ? \"yes\" : \"no\"}`).join(\",\");\n }\n};\n\n// src/utils/Timer.ts\nvar Timer = class extends Event {\n constructor() {\n super(...arguments);\n this._logger = new Logger(`Timer('${this._name}')`);\n this._timerHandle = null;\n this._expiration = 0;\n this._callback = () => {\n const diff = this._expiration - Timer.getEpochTime();\n this._logger.debug(\"timer completes in\", diff);\n if (this._expiration <= Timer.getEpochTime()) {\n this.cancel();\n super.raise();\n }\n };\n }\n // get the time\n static getEpochTime() {\n return Math.floor(Date.now() / 1e3);\n }\n init(durationInSeconds) {\n const logger2 = this._logger.create(\"init\");\n durationInSeconds = Math.max(Math.floor(durationInSeconds), 1);\n const expiration = Timer.getEpochTime() + durationInSeconds;\n if (this.expiration === expiration && this._timerHandle) {\n logger2.debug(\"skipping since already initialized for expiration at\", this.expiration);\n return;\n }\n this.cancel();\n logger2.debug(\"using duration\", durationInSeconds);\n this._expiration = expiration;\n const timerDurationInSeconds = Math.min(durationInSeconds, 5);\n this._timerHandle = setInterval(this._callback, timerDurationInSeconds * 1e3);\n }\n get expiration() {\n return this._expiration;\n }\n cancel() {\n this._logger.create(\"cancel\");\n if (this._timerHandle) {\n clearInterval(this._timerHandle);\n this._timerHandle = null;\n }\n }\n};\n\n// src/utils/UrlUtils.ts\nvar UrlUtils = class {\n static readParams(url, responseMode = \"query\") {\n if (!url)\n throw new TypeError(\"Invalid URL\");\n const parsedUrl = new URL(url, \"http://127.0.0.1\");\n const params = parsedUrl[responseMode === \"fragment\" ? \"hash\" : \"search\"];\n return new URLSearchParams(params.slice(1));\n }\n};\n\n// src/errors/ErrorResponse.ts\nvar ErrorResponse = class extends Error {\n constructor(args, form) {\n var _a, _b, _c;\n super(args.error_description || args.error || \"\");\n this.form = form;\n /** Marker to detect class: \"ErrorResponse\" */\n this.name = \"ErrorResponse\";\n if (!args.error) {\n Logger.error(\"ErrorResponse\", \"No error passed\");\n throw new Error(\"No error passed\");\n }\n this.error = args.error;\n this.error_description = (_a = args.error_description) != null ? _a : null;\n this.error_uri = (_b = args.error_uri) != null ? _b : null;\n this.state = args.userState;\n this.session_state = (_c = args.session_state) != null ? _c : null;\n }\n};\n\n// src/errors/ErrorTimeout.ts\nvar ErrorTimeout = class extends Error {\n constructor(message) {\n super(message);\n /** Marker to detect class: \"ErrorTimeout\" */\n this.name = \"ErrorTimeout\";\n }\n};\n\n// src/AccessTokenEvents.ts\nvar AccessTokenEvents = class {\n constructor(args) {\n this._logger = new Logger(\"AccessTokenEvents\");\n this._expiringTimer = new Timer(\"Access token expiring\");\n this._expiredTimer = new Timer(\"Access token expired\");\n this._expiringNotificationTimeInSeconds = args.expiringNotificationTimeInSeconds;\n }\n load(container) {\n const logger2 = this._logger.create(\"load\");\n if (container.access_token && container.expires_in !== void 0) {\n const duration = container.expires_in;\n logger2.debug(\"access token present, remaining duration:\", duration);\n if (duration > 0) {\n let expiring = duration - this._expiringNotificationTimeInSeconds;\n if (expiring <= 0) {\n expiring = 1;\n }\n logger2.debug(\"registering expiring timer, raising in\", expiring, \"seconds\");\n this._expiringTimer.init(expiring);\n } else {\n logger2.debug(\"canceling existing expiring timer because we're past expiration.\");\n this._expiringTimer.cancel();\n }\n const expired = duration + 1;\n logger2.debug(\"registering expired timer, raising in\", expired, \"seconds\");\n this._expiredTimer.init(expired);\n } else {\n this._expiringTimer.cancel();\n this._expiredTimer.cancel();\n }\n }\n unload() {\n this._logger.debug(\"unload: canceling existing access token timers\");\n this._expiringTimer.cancel();\n this._expiredTimer.cancel();\n }\n /**\n * Add callback: Raised prior to the access token expiring.\n */\n addAccessTokenExpiring(cb) {\n return this._expiringTimer.addHandler(cb);\n }\n /**\n * Remove callback: Raised prior to the access token expiring.\n */\n removeAccessTokenExpiring(cb) {\n this._expiringTimer.removeHandler(cb);\n }\n /**\n * Add callback: Raised after the access token has expired.\n */\n addAccessTokenExpired(cb) {\n return this._expiredTimer.addHandler(cb);\n }\n /**\n * Remove callback: Raised after the access token has expired.\n */\n removeAccessTokenExpired(cb) {\n this._expiredTimer.removeHandler(cb);\n }\n};\n\n// src/CheckSessionIFrame.ts\nvar CheckSessionIFrame = class {\n constructor(_callback, _client_id, url, _intervalInSeconds, _stopOnError) {\n this._callback = _callback;\n this._client_id = _client_id;\n this._intervalInSeconds = _intervalInSeconds;\n this._stopOnError = _stopOnError;\n this._logger = new Logger(\"CheckSessionIFrame\");\n this._timer = null;\n this._session_state = null;\n this._message = (e) => {\n if (e.origin === this._frame_origin && e.source === this._frame.contentWindow) {\n if (e.data === \"error\") {\n this._logger.error(\"error message from check session op iframe\");\n if (this._stopOnError) {\n this.stop();\n }\n } else if (e.data === \"changed\") {\n this._logger.debug(\"changed message from check session op iframe\");\n this.stop();\n void this._callback();\n } else {\n this._logger.debug(e.data + \" message from check session op iframe\");\n }\n }\n };\n const parsedUrl = new URL(url);\n this._frame_origin = parsedUrl.origin;\n this._frame = window.document.createElement(\"iframe\");\n this._frame.style.visibility = \"hidden\";\n this._frame.style.position = \"fixed\";\n this._frame.style.left = \"-1000px\";\n this._frame.style.top = \"0\";\n this._frame.width = \"0\";\n this._frame.height = \"0\";\n this._frame.src = parsedUrl.href;\n }\n load() {\n return new Promise((resolve) => {\n this._frame.onload = () => {\n resolve();\n };\n window.document.body.appendChild(this._frame);\n window.addEventListener(\"message\", this._message, false);\n });\n }\n start(session_state) {\n if (this._session_state === session_state) {\n return;\n }\n this._logger.create(\"start\");\n this.stop();\n this._session_state = session_state;\n const send = () => {\n if (!this._frame.contentWindow || !this._session_state) {\n return;\n }\n this._frame.contentWindow.postMessage(this._client_id + \" \" + this._session_state, this._frame_origin);\n };\n send();\n this._timer = setInterval(send, this._intervalInSeconds * 1e3);\n }\n stop() {\n this._logger.create(\"stop\");\n this._session_state = null;\n if (this._timer) {\n clearInterval(this._timer);\n this._timer = null;\n }\n }\n};\n\n// src/InMemoryWebStorage.ts\nvar InMemoryWebStorage = class {\n constructor() {\n this._logger = new Logger(\"InMemoryWebStorage\");\n this._data = {};\n }\n clear() {\n this._logger.create(\"clear\");\n this._data = {};\n }\n getItem(key) {\n this._logger.create(`getItem('${key}')`);\n return this._data[key];\n }\n setItem(key, value) {\n this._logger.create(`setItem('${key}')`);\n this._data[key] = value;\n }\n removeItem(key) {\n this._logger.create(`removeItem('${key}')`);\n delete this._data[key];\n }\n get length() {\n return Object.getOwnPropertyNames(this._data).length;\n }\n key(index) {\n return Object.getOwnPropertyNames(this._data)[index];\n }\n};\n\n// src/JsonService.ts\nvar JsonService = class {\n constructor(additionalContentTypes = [], _jwtHandler = null, _extraHeaders = {}) {\n this._jwtHandler = _jwtHandler;\n this._extraHeaders = _extraHeaders;\n this._logger = new Logger(\"JsonService\");\n this._contentTypes = [];\n this._contentTypes.push(...additionalContentTypes, \"application/json\");\n if (_jwtHandler) {\n this._contentTypes.push(\"application/jwt\");\n }\n }\n async fetchWithTimeout(input, init = {}) {\n const { timeoutInSeconds, ...initFetch } = init;\n if (!timeoutInSeconds) {\n return await fetch(input, initFetch);\n }\n const controller = new AbortController();\n const timeoutId = setTimeout(() => controller.abort(), timeoutInSeconds * 1e3);\n try {\n const response = await fetch(input, {\n ...init,\n signal: controller.signal\n });\n return response;\n } catch (err) {\n if (err instanceof DOMException && err.name === \"AbortError\") {\n throw new ErrorTimeout(\"Network timed out\");\n }\n throw err;\n } finally {\n clearTimeout(timeoutId);\n }\n }\n async getJson(url, {\n token,\n credentials\n } = {}) {\n const logger2 = this._logger.create(\"getJson\");\n const headers = {\n \"Accept\": this._contentTypes.join(\", \")\n };\n if (token) {\n logger2.debug(\"token passed, setting Authorization header\");\n headers[\"Authorization\"] = \"Bearer \" + token;\n }\n this.appendExtraHeaders(headers);\n let response;\n try {\n logger2.debug(\"url:\", url);\n response = await this.fetchWithTimeout(url, { method: \"GET\", headers, credentials });\n } catch (err) {\n logger2.error(\"Network Error\");\n throw err;\n }\n logger2.debug(\"HTTP response received, status\", response.status);\n const contentType = response.headers.get(\"Content-Type\");\n if (contentType && !this._contentTypes.find((item) => contentType.startsWith(item))) {\n logger2.throw(new Error(`Invalid response Content-Type: ${contentType != null ? contentType : \"undefined\"}, from URL: ${url}`));\n }\n if (response.ok && this._jwtHandler && (contentType == null ? void 0 : contentType.startsWith(\"application/jwt\"))) {\n return await this._jwtHandler(await response.text());\n }\n let json;\n try {\n json = await response.json();\n } catch (err) {\n logger2.error(\"Error parsing JSON response\", err);\n if (response.ok)\n throw err;\n throw new Error(`${response.statusText} (${response.status})`);\n }\n if (!response.ok) {\n logger2.error(\"Error from server:\", json);\n if (json.error) {\n throw new ErrorResponse(json);\n }\n throw new Error(`${response.statusText} (${response.status}): ${JSON.stringify(json)}`);\n }\n return json;\n }\n async postForm(url, {\n body,\n basicAuth,\n timeoutInSeconds,\n initCredentials\n }) {\n const logger2 = this._logger.create(\"postForm\");\n const headers = {\n \"Accept\": this._contentTypes.join(\", \"),\n \"Content-Type\": \"application/x-www-form-urlencoded\"\n };\n if (basicAuth !== void 0) {\n headers[\"Authorization\"] = \"Basic \" + basicAuth;\n }\n this.appendExtraHeaders(headers);\n let response;\n try {\n logger2.debug(\"url:\", url);\n response = await this.fetchWithTimeout(url, { method: \"POST\", headers, body, timeoutInSeconds, credentials: initCredentials });\n } catch (err) {\n logger2.error(\"Network error\");\n throw err;\n }\n logger2.debug(\"HTTP response received, status\", response.status);\n const contentType = response.headers.get(\"Content-Type\");\n if (contentType && !this._contentTypes.find((item) => contentType.startsWith(item))) {\n throw new Error(`Invalid response Content-Type: ${contentType != null ? contentType : \"undefined\"}, from URL: ${url}`);\n }\n const responseText = await response.text();\n let json = {};\n if (responseText) {\n try {\n json = JSON.parse(responseText);\n } catch (err) {\n logger2.error(\"Error parsing JSON response\", err);\n if (response.ok)\n throw err;\n throw new Error(`${response.statusText} (${response.status})`);\n }\n }\n if (!response.ok) {\n logger2.error(\"Error from server:\", json);\n if (json.error) {\n throw new ErrorResponse(json, body);\n }\n throw new Error(`${response.statusText} (${response.status}): ${JSON.stringify(json)}`);\n }\n return json;\n }\n appendExtraHeaders(headers) {\n const logger2 = this._logger.create(\"appendExtraHeaders\");\n const customKeys = Object.keys(this._extraHeaders);\n const protectedHeaders = [\n \"authorization\",\n \"accept\",\n \"content-type\"\n ];\n if (customKeys.length === 0) {\n return;\n }\n customKeys.forEach((headerName) => {\n if (protectedHeaders.includes(headerName.toLocaleLowerCase())) {\n logger2.warn(\"Protected header could not be overridden\", headerName, protectedHeaders);\n return;\n }\n const content = typeof this._extraHeaders[headerName] === \"function\" ? this._extraHeaders[headerName]() : this._extraHeaders[headerName];\n if (content && content !== \"\") {\n headers[headerName] = content;\n }\n });\n }\n};\n\n// src/MetadataService.ts\nvar MetadataService = class {\n constructor(_settings) {\n this._settings = _settings;\n this._logger = new Logger(\"MetadataService\");\n this._signingKeys = null;\n this._metadata = null;\n this._metadataUrl = this._settings.metadataUrl;\n this._jsonService = new JsonService(\n [\"application/jwk-set+json\"],\n null,\n this._settings.extraHeaders\n );\n if (this._settings.signingKeys) {\n this._logger.debug(\"using signingKeys from settings\");\n this._signingKeys = this._settings.signingKeys;\n }\n if (this._settings.metadata) {\n this._logger.debug(\"using metadata from settings\");\n this._metadata = this._settings.metadata;\n }\n if (this._settings.fetchRequestCredentials) {\n this._logger.debug(\"using fetchRequestCredentials from settings\");\n this._fetchRequestCredentials = this._settings.fetchRequestCredentials;\n }\n }\n resetSigningKeys() {\n this._signingKeys = null;\n }\n async getMetadata() {\n const logger2 = this._logger.create(\"getMetadata\");\n if (this._metadata) {\n logger2.debug(\"using cached values\");\n return this._metadata;\n }\n if (!this._metadataUrl) {\n logger2.throw(new Error(\"No authority or metadataUrl configured on settings\"));\n throw null;\n }\n logger2.debug(\"getting metadata from\", this._metadataUrl);\n const metadata = await this._jsonService.getJson(this._metadataUrl, { credentials: this._fetchRequestCredentials });\n logger2.debug(\"merging remote JSON with seed metadata\");\n this._metadata = Object.assign({}, this._settings.metadataSeed, metadata);\n return this._metadata;\n }\n getIssuer() {\n return this._getMetadataProperty(\"issuer\");\n }\n getAuthorizationEndpoint() {\n return this._getMetadataProperty(\"authorization_endpoint\");\n }\n getUserInfoEndpoint() {\n return this._getMetadataProperty(\"userinfo_endpoint\");\n }\n getTokenEndpoint(optional = true) {\n return this._getMetadataProperty(\"token_endpoint\", optional);\n }\n getCheckSessionIframe() {\n return this._getMetadataProperty(\"check_session_iframe\", true);\n }\n getEndSessionEndpoint() {\n return this._getMetadataProperty(\"end_session_endpoint\", true);\n }\n getRevocationEndpoint(optional = true) {\n return this._getMetadataProperty(\"revocation_endpoint\", optional);\n }\n getKeysEndpoint(optional = true) {\n return this._getMetadataProperty(\"jwks_uri\", optional);\n }\n async _getMetadataProperty(name, optional = false) {\n const logger2 = this._logger.create(`_getMetadataProperty('${name}')`);\n const metadata = await this.getMetadata();\n logger2.debug(\"resolved\");\n if (metadata[name] === void 0) {\n if (optional === true) {\n logger2.warn(\"Metadata does not contain optional property\");\n return void 0;\n }\n logger2.throw(new Error(\"Metadata does not contain property \" + name));\n }\n return metadata[name];\n }\n async getSigningKeys() {\n const logger2 = this._logger.create(\"getSigningKeys\");\n if (this._signingKeys) {\n logger2.debug(\"returning signingKeys from cache\");\n return this._signingKeys;\n }\n const jwks_uri = await this.getKeysEndpoint(false);\n logger2.debug(\"got jwks_uri\", jwks_uri);\n const keySet = await this._jsonService.getJson(jwks_uri);\n logger2.debug(\"got key set\", keySet);\n if (!Array.isArray(keySet.keys)) {\n logger2.throw(new Error(\"Missing keys on keyset\"));\n throw null;\n }\n this._signingKeys = keySet.keys;\n return this._signingKeys;\n }\n};\n\n// src/WebStorageStateStore.ts\nvar WebStorageStateStore = class {\n constructor({\n prefix = \"oidc.\",\n store = localStorage\n } = {}) {\n this._logger = new Logger(\"WebStorageStateStore\");\n this._store = store;\n this._prefix = prefix;\n }\n async set(key, value) {\n this._logger.create(`set('${key}')`);\n key = this._prefix + key;\n await this._store.setItem(key, value);\n }\n async get(key) {\n this._logger.create(`get('${key}')`);\n key = this._prefix + key;\n const item = await this._store.getItem(key);\n return item;\n }\n async remove(key) {\n this._logger.create(`remove('${key}')`);\n key = this._prefix + key;\n const item = await this._store.getItem(key);\n await this._store.removeItem(key);\n return item;\n }\n async getAllKeys() {\n this._logger.create(\"getAllKeys\");\n const len = await this._store.length;\n const keys = [];\n for (let index = 0; index < len; index++) {\n const key = await this._store.key(index);\n if (key && key.indexOf(this._prefix) === 0) {\n keys.push(key.substr(this._prefix.length));\n }\n }\n return keys;\n }\n};\n\n// src/OidcClientSettings.ts\nvar DefaultResponseType = \"code\";\nvar DefaultScope = \"openid\";\nvar DefaultClientAuthentication = \"client_secret_post\";\nvar DefaultResponseMode = \"query\";\nvar DefaultStaleStateAgeInSeconds = 60 * 15;\nvar DefaultClockSkewInSeconds = 60 * 5;\nvar OidcClientSettingsStore = class {\n constructor({\n // metadata related\n authority,\n metadataUrl,\n metadata,\n signingKeys,\n metadataSeed,\n // client related\n client_id,\n client_secret,\n response_type = DefaultResponseType,\n scope = DefaultScope,\n redirect_uri,\n post_logout_redirect_uri,\n client_authentication = DefaultClientAuthentication,\n // optional protocol\n prompt,\n display,\n max_age,\n ui_locales,\n acr_values,\n resource,\n response_mode = DefaultResponseMode,\n // behavior flags\n filterProtocolClaims = true,\n loadUserInfo = false,\n staleStateAgeInSeconds = DefaultStaleStateAgeInSeconds,\n clockSkewInSeconds = DefaultClockSkewInSeconds,\n userInfoJwtIssuer = \"OP\",\n mergeClaims = false,\n disablePKCE = false,\n // other behavior\n stateStore,\n refreshTokenCredentials,\n revokeTokenAdditionalContentTypes,\n fetchRequestCredentials,\n refreshTokenAllowedScope,\n // extra\n extraQueryParams = {},\n extraTokenParams = {},\n extraHeaders = {}\n }) {\n this.authority = authority;\n if (metadataUrl) {\n this.metadataUrl = metadataUrl;\n } else {\n this.metadataUrl = authority;\n if (authority) {\n if (!this.metadataUrl.endsWith(\"/\")) {\n this.metadataUrl += \"/\";\n }\n this.metadataUrl += \".well-known/openid-configuration\";\n }\n }\n this.metadata = metadata;\n this.metadataSeed = metadataSeed;\n this.signingKeys = signingKeys;\n this.client_id = client_id;\n this.client_secret = client_secret;\n this.response_type = response_type;\n this.scope = scope;\n this.redirect_uri = redirect_uri;\n this.post_logout_redirect_uri = post_logout_redirect_uri;\n this.client_authentication = client_authentication;\n this.prompt = prompt;\n this.display = display;\n this.max_age = max_age;\n this.ui_locales = ui_locales;\n this.acr_values = acr_values;\n this.resource = resource;\n this.response_mode = response_mode;\n this.filterProtocolClaims = filterProtocolClaims != null ? filterProtocolClaims : true;\n this.loadUserInfo = !!loadUserInfo;\n this.staleStateAgeInSeconds = staleStateAgeInSeconds;\n this.clockSkewInSeconds = clockSkewInSeconds;\n this.userInfoJwtIssuer = userInfoJwtIssuer;\n this.mergeClaims = !!mergeClaims;\n this.disablePKCE = !!disablePKCE;\n this.revokeTokenAdditionalContentTypes = revokeTokenAdditionalContentTypes;\n if (fetchRequestCredentials && refreshTokenCredentials) {\n console.warn(\"Both fetchRequestCredentials and refreshTokenCredentials is set. Only fetchRequestCredentials will be used.\");\n }\n this.fetchRequestCredentials = fetchRequestCredentials ? fetchRequestCredentials : refreshTokenCredentials ? refreshTokenCredentials : \"same-origin\";\n if (stateStore) {\n this.stateStore = stateStore;\n } else {\n const store = typeof window !== \"undefined\" ? window.localStorage : new InMemoryWebStorage();\n this.stateStore = new WebStorageStateStore({ store });\n }\n this.refreshTokenAllowedScope = refreshTokenAllowedScope;\n this.extraQueryParams = extraQueryParams;\n this.extraTokenParams = extraTokenParams;\n this.extraHeaders = extraHeaders;\n }\n};\n\n// src/UserInfoService.ts\nvar UserInfoService = class {\n constructor(_settings, _metadataService) {\n this._settings = _settings;\n this._metadataService = _metadataService;\n this._logger = new Logger(\"UserInfoService\");\n this._getClaimsFromJwt = async (responseText) => {\n const logger2 = this._logger.create(\"_getClaimsFromJwt\");\n try {\n const payload = JwtUtils.decode(responseText);\n logger2.debug(\"JWT decoding successful\");\n return payload;\n } catch (err) {\n logger2.error(\"Error parsing JWT response\");\n throw err;\n }\n };\n this._jsonService = new JsonService(\n void 0,\n this._getClaimsFromJwt,\n this._settings.extraHeaders\n );\n }\n async getClaims(token) {\n const logger2 = this._logger.create(\"getClaims\");\n if (!token) {\n this._logger.throw(new Error(\"No token passed\"));\n }\n const url = await this._metadataService.getUserInfoEndpoint();\n logger2.debug(\"got userinfo url\", url);\n const claims = await this._jsonService.getJson(url, {\n token,\n credentials: this._settings.fetchRequestCredentials\n });\n logger2.debug(\"got claims\", claims);\n return claims;\n }\n};\n\n// src/TokenClient.ts\nvar TokenClient = class {\n constructor(_settings, _metadataService) {\n this._settings = _settings;\n this._metadataService = _metadataService;\n this._logger = new Logger(\"TokenClient\");\n this._jsonService = new JsonService(\n this._settings.revokeTokenAdditionalContentTypes,\n null,\n this._settings.extraHeaders\n );\n }\n /**\n * Exchange code.\n *\n * @see https://www.rfc-editor.org/rfc/rfc6749#section-4.1.3\n */\n async exchangeCode({\n grant_type = \"authorization_code\",\n redirect_uri = this._settings.redirect_uri,\n client_id = this._settings.client_id,\n client_secret = this._settings.client_secret,\n ...args\n }) {\n const logger2 = this._logger.create(\"exchangeCode\");\n if (!client_id) {\n logger2.throw(new Error(\"A client_id is required\"));\n }\n if (!redirect_uri) {\n logger2.throw(new Error(\"A redirect_uri is required\"));\n }\n if (!args.code) {\n logger2.throw(new Error(\"A code is required\"));\n }\n const params = new URLSearchParams({ grant_type, redirect_uri });\n for (const [key, value] of Object.entries(args)) {\n if (value != null) {\n params.set(key, value);\n }\n }\n let basicAuth;\n switch (this._settings.client_authentication) {\n case \"client_secret_basic\":\n if (!client_secret) {\n logger2.throw(new Error(\"A client_secret is required\"));\n throw null;\n }\n basicAuth = CryptoUtils.generateBasicAuth(client_id, client_secret);\n break;\n case \"client_secret_post\":\n params.append(\"client_id\", client_id);\n if (client_secret) {\n params.append(\"client_secret\", client_secret);\n }\n break;\n }\n const url = await this._metadataService.getTokenEndpoint(false);\n logger2.debug(\"got token endpoint\");\n const response = await this._jsonService.postForm(url, { body: params, basicAuth, initCredentials: this._settings.fetchRequestCredentials });\n logger2.debug(\"got response\");\n return response;\n }\n /**\n * Exchange credentials.\n *\n * @see https://www.rfc-editor.org/rfc/rfc6749#section-4.3.2\n */\n async exchangeCredentials({\n grant_type = \"password\",\n client_id = this._settings.client_id,\n client_secret = this._settings.client_secret,\n scope = this._settings.scope,\n ...args\n }) {\n const logger2 = this._logger.create(\"exchangeCredentials\");\n if (!client_id) {\n logger2.throw(new Error(\"A client_id is required\"));\n }\n const params = new URLSearchParams({ grant_type, scope });\n for (const [key, value] of Object.entries(args)) {\n if (value != null) {\n params.set(key, value);\n }\n }\n let basicAuth;\n switch (this._settings.client_authentication) {\n case \"client_secret_basic\":\n if (!client_secret) {\n logger2.throw(new Error(\"A client_secret is required\"));\n throw null;\n }\n basicAuth = CryptoUtils.generateBasicAuth(client_id, client_secret);\n break;\n case \"client_secret_post\":\n params.append(\"client_id\", client_id);\n if (client_secret) {\n params.append(\"client_secret\", client_secret);\n }\n break;\n }\n const url = await this._metadataService.getTokenEndpoint(false);\n logger2.debug(\"got token endpoint\");\n const response = await this._jsonService.postForm(url, { body: params, basicAuth, initCredentials: this._settings.fetchRequestCredentials });\n logger2.debug(\"got response\");\n return response;\n }\n /**\n * Exchange a refresh token.\n *\n * @see https://www.rfc-editor.org/rfc/rfc6749#section-6\n */\n async exchangeRefreshToken({\n grant_type = \"refresh_token\",\n client_id = this._settings.client_id,\n client_secret = this._settings.client_secret,\n timeoutInSeconds,\n ...args\n }) {\n const logger2 = this._logger.create(\"exchangeRefreshToken\");\n if (!client_id) {\n logger2.throw(new Error(\"A client_id is required\"));\n }\n if (!args.refresh_token) {\n logger2.throw(new Error(\"A refresh_token is required\"));\n }\n const params = new URLSearchParams({ grant_type });\n for (const [key, value] of Object.entries(args)) {\n if (value != null) {\n params.set(key, value);\n }\n }\n let basicAuth;\n switch (this._settings.client_authentication) {\n case \"client_secret_basic\":\n if (!client_secret) {\n logger2.throw(new Error(\"A client_secret is required\"));\n throw null;\n }\n basicAuth = CryptoUtils.generateBasicAuth(client_id, client_secret);\n break;\n case \"client_secret_post\":\n params.append(\"client_id\", client_id);\n if (client_secret) {\n params.append(\"client_secret\", client_secret);\n }\n break;\n }\n const url = await this._metadataService.getTokenEndpoint(false);\n logger2.debug(\"got token endpoint\");\n const response = await this._jsonService.postForm(url, { body: params, basicAuth, timeoutInSeconds, initCredentials: this._settings.fetchRequestCredentials });\n logger2.debug(\"got response\");\n return response;\n }\n /**\n * Revoke an access or refresh token.\n *\n * @see https://datatracker.ietf.org/doc/html/rfc7009#section-2.1\n */\n async revoke(args) {\n var _a;\n const logger2 = this._logger.create(\"revoke\");\n if (!args.token) {\n logger2.throw(new Error(\"A token is required\"));\n }\n const url = await this._metadataService.getRevocationEndpoint(false);\n logger2.debug(`got revocation endpoint, revoking ${(_a = args.token_type_hint) != null ? _a : \"default token type\"}`);\n const params = new URLSearchParams();\n for (const [key, value] of Object.entries(args)) {\n if (value != null) {\n params.set(key, value);\n }\n }\n params.set(\"client_id\", this._settings.client_id);\n if (this._settings.client_secret) {\n params.set(\"client_secret\", this._settings.client_secret);\n }\n await this._jsonService.postForm(url, { body: params });\n logger2.debug(\"got response\");\n }\n};\n\n// src/ResponseValidator.ts\nvar ResponseValidator = class {\n constructor(_settings, _metadataService, _claimsService) {\n this._settings = _settings;\n this._metadataService = _metadataService;\n this._claimsService = _claimsService;\n this._logger = new Logger(\"ResponseValidator\");\n this._userInfoService = new UserInfoService(this._settings, this._metadataService);\n this._tokenClient = new TokenClient(this._settings, this._metadataService);\n }\n async validateSigninResponse(response, state) {\n const logger2 = this._logger.create(\"validateSigninResponse\");\n this._processSigninState(response, state);\n logger2.debug(\"state processed\");\n await this._processCode(response, state);\n logger2.debug(\"code processed\");\n if (response.isOpenId) {\n this._validateIdTokenAttributes(response);\n }\n logger2.debug(\"tokens validated\");\n await this._processClaims(response, state == null ? void 0 : state.skipUserInfo, response.isOpenId);\n logger2.debug(\"claims processed\");\n }\n async validateCredentialsResponse(response, skipUserInfo) {\n const logger2 = this._logger.create(\"validateCredentialsResponse\");\n if (response.isOpenId) {\n this._validateIdTokenAttributes(response);\n }\n logger2.debug(\"tokens validated\");\n await this._processClaims(response, skipUserInfo, response.isOpenId);\n logger2.debug(\"claims processed\");\n }\n async validateRefreshResponse(response, state) {\n var _a, _b;\n const logger2 = this._logger.create(\"validateRefreshResponse\");\n response.userState = state.data;\n (_a = response.session_state) != null ? _a : response.session_state = state.session_state;\n (_b = response.scope) != null ? _b : response.scope = state.scope;\n if (response.isOpenId && !!response.id_token) {\n this._validateIdTokenAttributes(response, state.id_token);\n logger2.debug(\"ID Token validated\");\n }\n if (!response.id_token) {\n response.id_token = state.id_token;\n response.profile = state.profile;\n }\n const hasIdToken = response.isOpenId && !!response.id_token;\n await this._processClaims(response, false, hasIdToken);\n logger2.debug(\"claims processed\");\n }\n validateSignoutResponse(response, state) {\n const logger2 = this._logger.create(\"validateSignoutResponse\");\n if (state.id !== response.state) {\n logger2.throw(new Error(\"State does not match\"));\n }\n logger2.debug(\"state validated\");\n response.userState = state.data;\n if (response.error) {\n logger2.warn(\"Response was error\", response.error);\n throw new ErrorResponse(response);\n }\n }\n _processSigninState(response, state) {\n var _a;\n const logger2 = this._logger.create(\"_processSigninState\");\n if (state.id !== response.state) {\n logger2.throw(new Error(\"State does not match\"));\n }\n if (!state.client_id) {\n logger2.throw(new Error(\"No client_id on state\"));\n }\n if (!state.authority) {\n logger2.throw(new Error(\"No authority on state\"));\n }\n if (this._settings.authority !== state.authority) {\n logger2.throw(new Error(\"authority mismatch on settings vs. signin state\"));\n }\n if (this._settings.client_id && this._settings.client_id !== state.client_id) {\n logger2.throw(new Error(\"client_id mismatch on settings vs. signin state\"));\n }\n logger2.debug(\"state validated\");\n response.userState = state.data;\n (_a = response.scope) != null ? _a : response.scope = state.scope;\n if (response.error) {\n logger2.warn(\"Response was error\", response.error);\n throw new ErrorResponse(response);\n }\n if (state.code_verifier && !response.code) {\n logger2.throw(new Error(\"Expected code in response\"));\n }\n }\n async _processClaims(response, skipUserInfo = false, validateSub = true) {\n const logger2 = this._logger.create(\"_processClaims\");\n response.profile = this._claimsService.filterProtocolClaims(response.profile);\n if (skipUserInfo || !this._settings.loadUserInfo || !response.access_token) {\n logger2.debug(\"not loading user info\");\n return;\n }\n logger2.debug(\"loading user info\");\n const claims = await this._userInfoService.getClaims(response.access_token);\n logger2.debug(\"user info claims received from user info endpoint\");\n if (validateSub && claims.sub !== response.profile.sub) {\n logger2.throw(new Error(\"subject from UserInfo response does not match subject in ID Token\"));\n }\n response.profile = this._claimsService.mergeClaims(response.profile, this._claimsService.filterProtocolClaims(claims));\n logger2.debug(\"user info claims received, updated profile:\", response.profile);\n }\n async _processCode(response, state) {\n const logger2 = this._logger.create(\"_processCode\");\n if (response.code) {\n logger2.debug(\"Validating code\");\n const tokenResponse = await this._tokenClient.exchangeCode({\n client_id: state.client_id,\n client_secret: state.client_secret,\n code: response.code,\n redirect_uri: state.redirect_uri,\n code_verifier: state.code_verifier,\n ...state.extraTokenParams\n });\n Object.assign(response, tokenResponse);\n } else {\n logger2.debug(\"No code to process\");\n }\n }\n _validateIdTokenAttributes(response, existingToken) {\n var _a;\n const logger2 = this._logger.create(\"_validateIdTokenAttributes\");\n logger2.debug(\"decoding ID Token JWT\");\n const incoming = JwtUtils.decode((_a = response.id_token) != null ? _a : \"\");\n if (!incoming.sub) {\n logger2.throw(new Error(\"ID Token is missing a subject claim\"));\n }\n if (existingToken) {\n const existing = JwtUtils.decode(existingToken);\n if (incoming.sub !== existing.sub) {\n logger2.throw(new Error(\"sub in id_token does not match current sub\"));\n }\n if (incoming.auth_time && incoming.auth_time !== existing.auth_time) {\n logger2.throw(new Error(\"auth_time in id_token does not match original auth_time\"));\n }\n if (incoming.azp && incoming.azp !== existing.azp) {\n logger2.throw(new Error(\"azp in id_token does not match original azp\"));\n }\n if (!incoming.azp && existing.azp) {\n logger2.throw(new Error(\"azp not in id_token, but present in original id_token\"));\n }\n }\n response.profile = incoming;\n }\n};\n\n// src/State.ts\nvar State = class {\n constructor(args) {\n this.id = args.id || CryptoUtils.generateUUIDv4();\n this.data = args.data;\n if (args.created && args.created > 0) {\n this.created = args.created;\n } else {\n this.created = Timer.getEpochTime();\n }\n this.request_type = args.request_type;\n }\n toStorageString() {\n new Logger(\"State\").create(\"toStorageString\");\n return JSON.stringify({\n id: this.id,\n data: this.data,\n created: this.created,\n request_type: this.request_type\n });\n }\n static fromStorageString(storageString) {\n Logger.createStatic(\"State\", \"fromStorageString\");\n return new State(JSON.parse(storageString));\n }\n static async clearStaleState(storage, age) {\n const logger2 = Logger.createStatic(\"State\", \"clearStaleState\");\n const cutoff = Timer.getEpochTime() - age;\n const keys = await storage.getAllKeys();\n logger2.debug(\"got keys\", keys);\n for (let i = 0; i < keys.length; i++) {\n const key = keys[i];\n const item = await storage.get(key);\n let remove = false;\n if (item) {\n try {\n const state = State.fromStorageString(item);\n logger2.debug(\"got item from key:\", key, state.created);\n if (state.created <= cutoff) {\n remove = true;\n }\n } catch (err) {\n logger2.error(\"Error parsing state for key:\", key, err);\n remove = true;\n }\n } else {\n logger2.debug(\"no item in storage for key:\", key);\n remove = true;\n }\n if (remove) {\n logger2.debug(\"removed item for key:\", key);\n void storage.remove(key);\n }\n }\n }\n};\n\n// src/SigninState.ts\nvar SigninState = class extends State {\n constructor(args) {\n super(args);\n if (args.code_verifier === true) {\n this.code_verifier = CryptoUtils.generateCodeVerifier();\n } else if (args.code_verifier) {\n this.code_verifier = args.code_verifier;\n }\n if (this.code_verifier) {\n this.code_challenge = CryptoUtils.generateCodeChallenge(this.code_verifier);\n }\n this.authority = args.authority;\n this.client_id = args.client_id;\n this.redirect_uri = args.redirect_uri;\n this.scope = args.scope;\n this.client_secret = args.client_secret;\n this.extraTokenParams = args.extraTokenParams;\n this.response_mode = args.response_mode;\n this.skipUserInfo = args.skipUserInfo;\n }\n toStorageString() {\n new Logger(\"SigninState\").create(\"toStorageString\");\n return JSON.stringify({\n id: this.id,\n data: this.data,\n created: this.created,\n request_type: this.request_type,\n code_verifier: this.code_verifier,\n authority: this.authority,\n client_id: this.client_id,\n redirect_uri: this.redirect_uri,\n scope: this.scope,\n client_secret: this.client_secret,\n extraTokenParams: this.extraTokenParams,\n response_mode: this.response_mode,\n skipUserInfo: this.skipUserInfo\n });\n }\n static fromStorageString(storageString) {\n Logger.createStatic(\"SigninState\", \"fromStorageString\");\n const data = JSON.parse(storageString);\n return new SigninState(data);\n }\n};\n\n// src/SigninRequest.ts\nvar SigninRequest = class {\n constructor({\n // mandatory\n url,\n authority,\n client_id,\n redirect_uri,\n response_type,\n scope,\n // optional\n state_data,\n response_mode,\n request_type,\n client_secret,\n nonce,\n resource,\n skipUserInfo,\n extraQueryParams,\n extraTokenParams,\n disablePKCE,\n ...optionalParams\n }) {\n this._logger = new Logger(\"SigninRequest\");\n if (!url) {\n this._logger.error(\"ctor: No url passed\");\n throw new Error(\"url\");\n }\n if (!client_id) {\n this._logger.error(\"ctor: No client_id passed\");\n throw new Error(\"client_id\");\n }\n if (!redirect_uri) {\n this._logger.error(\"ctor: No redirect_uri passed\");\n throw new Error(\"redirect_uri\");\n }\n if (!response_type) {\n this._logger.error(\"ctor: No response_type passed\");\n throw new Error(\"response_type\");\n }\n if (!scope) {\n this._logger.error(\"ctor: No scope passed\");\n throw new Error(\"scope\");\n }\n if (!authority) {\n this._logger.error(\"ctor: No authority passed\");\n throw new Error(\"authority\");\n }\n this.state = new SigninState({\n data: state_data,\n request_type,\n code_verifier: !disablePKCE,\n client_id,\n authority,\n redirect_uri,\n response_mode,\n client_secret,\n scope,\n extraTokenParams,\n skipUserInfo\n });\n const parsedUrl = new URL(url);\n parsedUrl.searchParams.append(\"client_id\", client_id);\n parsedUrl.searchParams.append(\"redirect_uri\", redirect_uri);\n parsedUrl.searchParams.append(\"response_type\", response_type);\n parsedUrl.searchParams.append(\"scope\", scope);\n if (nonce) {\n parsedUrl.searchParams.append(\"nonce\", nonce);\n }\n parsedUrl.searchParams.append(\"state\", this.state.id);\n if (this.state.code_challenge) {\n parsedUrl.searchParams.append(\"code_challenge\", this.state.code_challenge);\n parsedUrl.searchParams.append(\"code_challenge_method\", \"S256\");\n }\n if (resource) {\n const resources = Array.isArray(resource) ? resource : [resource];\n resources.forEach((r) => parsedUrl.searchParams.append(\"resource\", r));\n }\n for (const [key, value] of Object.entries({ response_mode, ...optionalParams, ...extraQueryParams })) {\n if (value != null) {\n parsedUrl.searchParams.append(key, value.toString());\n }\n }\n this.url = parsedUrl.href;\n }\n};\n\n// src/SigninResponse.ts\nvar OidcScope = \"openid\";\nvar SigninResponse = class {\n constructor(params) {\n /** @see {@link User.access_token} */\n this.access_token = \"\";\n /** @see {@link User.token_type} */\n this.token_type = \"\";\n /** @see {@link User.profile} */\n this.profile = {};\n this.state = params.get(\"state\");\n this.session_state = params.get(\"session_state\");\n this.error = params.get(\"error\");\n this.error_description = params.get(\"error_description\");\n this.error_uri = params.get(\"error_uri\");\n this.code = params.get(\"code\");\n }\n get expires_in() {\n if (this.expires_at === void 0) {\n return void 0;\n }\n return this.expires_at - Timer.getEpochTime();\n }\n set expires_in(value) {\n if (typeof value === \"string\")\n value = Number(value);\n if (value !== void 0 && value >= 0) {\n this.expires_at = Math.floor(value) + Timer.getEpochTime();\n }\n }\n get isOpenId() {\n var _a;\n return ((_a = this.scope) == null ? void 0 : _a.split(\" \").includes(OidcScope)) || !!this.id_token;\n }\n};\n\n// src/SignoutRequest.ts\nvar SignoutRequest = class {\n constructor({\n url,\n state_data,\n id_token_hint,\n post_logout_redirect_uri,\n extraQueryParams,\n request_type\n }) {\n this._logger = new Logger(\"SignoutRequest\");\n if (!url) {\n this._logger.error(\"ctor: No url passed\");\n throw new Error(\"url\");\n }\n const parsedUrl = new URL(url);\n if (id_token_hint) {\n parsedUrl.searchParams.append(\"id_token_hint\", id_token_hint);\n }\n if (post_logout_redirect_uri) {\n parsedUrl.searchParams.append(\"post_logout_redirect_uri\", post_logout_redirect_uri);\n if (state_data) {\n this.state = new State({ data: state_data, request_type });\n parsedUrl.searchParams.append(\"state\", this.state.id);\n }\n }\n for (const [key, value] of Object.entries({ ...extraQueryParams })) {\n if (value != null) {\n parsedUrl.searchParams.append(key, value.toString());\n }\n }\n this.url = parsedUrl.href;\n }\n};\n\n// src/SignoutResponse.ts\nvar SignoutResponse = class {\n constructor(params) {\n this.state = params.get(\"state\");\n this.error = params.get(\"error\");\n this.error_description = params.get(\"error_description\");\n this.error_uri = params.get(\"error_uri\");\n }\n};\n\n// src/ClaimsService.ts\nvar DefaultProtocolClaims = [\n \"nbf\",\n \"jti\",\n \"auth_time\",\n \"nonce\",\n \"acr\",\n \"amr\",\n \"azp\",\n \"at_hash\"\n // https://openid.net/specs/openid-connect-core-1_0.html#CodeIDToken\n];\nvar InternalRequiredProtocolClaims = [\"sub\", \"iss\", \"aud\", \"exp\", \"iat\"];\nvar ClaimsService = class {\n constructor(_settings) {\n this._settings = _settings;\n this._logger = new Logger(\"ClaimsService\");\n }\n filterProtocolClaims(claims) {\n const result = { ...claims };\n if (this._settings.filterProtocolClaims) {\n let protocolClaims;\n if (Array.isArray(this._settings.filterProtocolClaims)) {\n protocolClaims = this._settings.filterProtocolClaims;\n } else {\n protocolClaims = DefaultProtocolClaims;\n }\n for (const claim of protocolClaims) {\n if (!InternalRequiredProtocolClaims.includes(claim)) {\n delete result[claim];\n }\n }\n }\n return result;\n }\n mergeClaims(claims1, claims2) {\n const result = { ...claims1 };\n for (const [claim, values] of Object.entries(claims2)) {\n for (const value of Array.isArray(values) ? values : [values]) {\n const previousValue = result[claim];\n if (!previousValue) {\n result[claim] = value;\n } else if (Array.isArray(previousValue)) {\n if (!previousValue.includes(value)) {\n previousValue.push(value);\n }\n } else if (result[claim] !== value) {\n if (typeof value === \"object\" && this._settings.mergeClaims) {\n result[claim] = this.mergeClaims(previousValue, value);\n } else {\n result[claim] = [previousValue, value];\n }\n }\n }\n }\n return result;\n }\n};\n\n// src/OidcClient.ts\nvar OidcClient = class {\n constructor(settings) {\n this._logger = new Logger(\"OidcClient\");\n this.settings = new OidcClientSettingsStore(settings);\n this.metadataService = new MetadataService(this.settings);\n this._claimsService = new ClaimsService(this.settings);\n this._validator = new ResponseValidator(this.settings, this.metadataService, this._claimsService);\n this._tokenClient = new TokenClient(this.settings, this.metadataService);\n }\n async createSigninRequest({\n state,\n request,\n request_uri,\n request_type,\n id_token_hint,\n login_hint,\n skipUserInfo,\n nonce,\n response_type = this.settings.response_type,\n scope = this.settings.scope,\n redirect_uri = this.settings.redirect_uri,\n prompt = this.settings.prompt,\n display = this.settings.display,\n max_age = this.settings.max_age,\n ui_locales = this.settings.ui_locales,\n acr_values = this.settings.acr_values,\n resource = this.settings.resource,\n response_mode = this.settings.response_mode,\n extraQueryParams = this.settings.extraQueryParams,\n extraTokenParams = this.settings.extraTokenParams\n }) {\n const logger2 = this._logger.create(\"createSigninRequest\");\n if (response_type !== \"code\") {\n throw new Error(\"Only the Authorization Code flow (with PKCE) is supported\");\n }\n const url = await this.metadataService.getAuthorizationEndpoint();\n logger2.debug(\"Received authorization endpoint\", url);\n const signinRequest = new SigninRequest({\n url,\n authority: this.settings.authority,\n client_id: this.settings.client_id,\n redirect_uri,\n response_type,\n scope,\n state_data: state,\n prompt,\n display,\n max_age,\n ui_locales,\n id_token_hint,\n login_hint,\n acr_values,\n resource,\n request,\n request_uri,\n extraQueryParams,\n extraTokenParams,\n request_type,\n response_mode,\n client_secret: this.settings.client_secret,\n skipUserInfo,\n nonce,\n disablePKCE: this.settings.disablePKCE\n });\n await this.clearStaleState();\n const signinState = signinRequest.state;\n await this.settings.stateStore.set(signinState.id, signinState.toStorageString());\n return signinRequest;\n }\n async readSigninResponseState(url, removeState = false) {\n const logger2 = this._logger.create(\"readSigninResponseState\");\n const response = new SigninResponse(UrlUtils.readParams(url, this.settings.response_mode));\n if (!response.state) {\n logger2.throw(new Error(\"No state in response\"));\n throw null;\n }\n const storedStateString = await this.settings.stateStore[removeState ? \"remove\" : \"get\"](response.state);\n if (!storedStateString) {\n logger2.throw(new Error(\"No matching state found in storage\"));\n throw null;\n }\n const state = SigninState.fromStorageString(storedStateString);\n return { state, response };\n }\n async processSigninResponse(url) {\n const logger2 = this._logger.create(\"processSigninResponse\");\n const { state, response } = await this.readSigninResponseState(url, true);\n logger2.debug(\"received state from storage; validating response\");\n await this._validator.validateSigninResponse(response, state);\n return response;\n }\n async processResourceOwnerPasswordCredentials({\n username,\n password,\n skipUserInfo = false,\n extraTokenParams = {}\n }) {\n const tokenResponse = await this._tokenClient.exchangeCredentials({ username, password, ...extraTokenParams });\n const signinResponse = new SigninResponse(new URLSearchParams());\n Object.assign(signinResponse, tokenResponse);\n await this._validator.validateCredentialsResponse(signinResponse, skipUserInfo);\n return signinResponse;\n }\n async useRefreshToken({\n state,\n timeoutInSeconds\n }) {\n var _a;\n const logger2 = this._logger.create(\"useRefreshToken\");\n let scope;\n if (this.settings.refreshTokenAllowedScope === void 0) {\n scope = state.scope;\n } else {\n const allowableScopes = this.settings.refreshTokenAllowedScope.split(\" \");\n const providedScopes = ((_a = state.scope) == null ? void 0 : _a.split(\" \")) || [];\n scope = providedScopes.filter((s) => allowableScopes.includes(s)).join(\" \");\n }\n const result = await this._tokenClient.exchangeRefreshToken({\n refresh_token: state.refresh_token,\n // provide the (possible filtered) scope list\n scope,\n timeoutInSeconds\n });\n const response = new SigninResponse(new URLSearchParams());\n Object.assign(response, result);\n logger2.debug(\"validating response\", response);\n await this._validator.validateRefreshResponse(response, {\n ...state,\n // overide the scope in the state handed over to the validator\n // so it can set the granted scope to the requested scope in case none is included in the response\n scope\n });\n return response;\n }\n async createSignoutRequest({\n state,\n id_token_hint,\n request_type,\n post_logout_redirect_uri = this.settings.post_logout_redirect_uri,\n extraQueryParams = this.settings.extraQueryParams\n } = {}) {\n const logger2 = this._logger.create(\"createSignoutRequest\");\n const url = await this.metadataService.getEndSessionEndpoint();\n if (!url) {\n logger2.throw(new Error(\"No end session endpoint\"));\n throw null;\n }\n logger2.debug(\"Received end session endpoint\", url);\n const request = new SignoutRequest({\n url,\n id_token_hint,\n post_logout_redirect_uri,\n state_data: state,\n extraQueryParams,\n request_type\n });\n await this.clearStaleState();\n const signoutState = request.state;\n if (signoutState) {\n logger2.debug(\"Signout request has state to persist\");\n await this.settings.stateStore.set(signoutState.id, signoutState.toStorageString());\n }\n return request;\n }\n async readSignoutResponseState(url, removeState = false) {\n const logger2 = this._logger.create(\"readSignoutResponseState\");\n const response = new SignoutResponse(UrlUtils.readParams(url, this.settings.response_mode));\n if (!response.state) {\n logger2.debug(\"No state in response\");\n if (response.error) {\n logger2.warn(\"Response was error:\", response.error);\n throw new ErrorResponse(response);\n }\n return { state: void 0, response };\n }\n const storedStateString = await this.settings.stateStore[removeState ? \"remove\" : \"get\"](response.state);\n if (!storedStateString) {\n logger2.throw(new Error(\"No matching state found in storage\"));\n throw null;\n }\n const state = State.fromStorageString(storedStateString);\n return { state, response };\n }\n async processSignoutResponse(url) {\n const logger2 = this._logger.create(\"processSignoutResponse\");\n const { state, response } = await this.readSignoutResponseState(url, true);\n if (state) {\n logger2.debug(\"Received state from storage; validating response\");\n this._validator.validateSignoutResponse(response, state);\n } else {\n logger2.debug(\"No state from storage; skipping response validation\");\n }\n return response;\n }\n clearStaleState() {\n this._logger.create(\"clearStaleState\");\n return State.clearStaleState(this.settings.stateStore, this.settings.staleStateAgeInSeconds);\n }\n async revokeToken(token, type) {\n this._logger.create(\"revokeToken\");\n return await this._tokenClient.revoke({\n token,\n token_type_hint: type\n });\n }\n};\n\n// src/SessionMonitor.ts\nvar SessionMonitor = class {\n constructor(_userManager) {\n this._userManager = _userManager;\n this._logger = new Logger(\"SessionMonitor\");\n this._start = async (user) => {\n const session_state = user.session_state;\n if (!session_state) {\n return;\n }\n const logger2 = this._logger.create(\"_start\");\n if (user.profile) {\n this._sub = user.profile.sub;\n this._sid = user.profile.sid;\n logger2.debug(\"session_state\", session_state, \", sub\", this._sub);\n } else {\n this._sub = void 0;\n this._sid = void 0;\n logger2.debug(\"session_state\", session_state, \", anonymous user\");\n }\n if (this._checkSessionIFrame) {\n this._checkSessionIFrame.start(session_state);\n return;\n }\n try {\n const url = await this._userManager.metadataService.getCheckSessionIframe();\n if (url) {\n logger2.debug(\"initializing check session iframe\");\n const client_id = this._userManager.settings.client_id;\n const intervalInSeconds = this._userManager.settings.checkSessionIntervalInSeconds;\n const stopOnError = this._userManager.settings.stopCheckSessionOnError;\n const checkSessionIFrame = new CheckSessionIFrame(this._callback, client_id, url, intervalInSeconds, stopOnError);\n await checkSessionIFrame.load();\n this._checkSessionIFrame = checkSessionIFrame;\n checkSessionIFrame.start(session_state);\n } else {\n logger2.warn(\"no check session iframe found in the metadata\");\n }\n } catch (err) {\n logger2.error(\"Error from getCheckSessionIframe:\", err instanceof Error ? err.message : err);\n }\n };\n this._stop = () => {\n const logger2 = this._logger.create(\"_stop\");\n this._sub = void 0;\n this._sid = void 0;\n if (this._checkSessionIFrame) {\n this._checkSessionIFrame.stop();\n }\n if (this._userManager.settings.monitorAnonymousSession) {\n const timerHandle = setInterval(async () => {\n clearInterval(timerHandle);\n try {\n const session = await this._userManager.querySessionStatus();\n if (session) {\n const tmpUser = {\n session_state: session.session_state,\n profile: session.sub && session.sid ? {\n sub: session.sub,\n sid: session.sid\n } : null\n };\n void this._start(tmpUser);\n }\n } catch (err) {\n logger2.error(\"error from querySessionStatus\", err instanceof Error ? err.message : err);\n }\n }, 1e3);\n }\n };\n this._callback = async () => {\n const logger2 = this._logger.create(\"_callback\");\n try {\n const session = await this._userManager.querySessionStatus();\n let raiseEvent = true;\n if (session && this._checkSessionIFrame) {\n if (session.sub === this._sub) {\n raiseEvent = false;\n this._checkSessionIFrame.start(session.session_state);\n if (session.sid === this._sid) {\n logger2.debug(\"same sub still logged in at OP, restarting check session iframe; session_state\", session.session_state);\n } else {\n logger2.debug(\"same sub still logged in at OP, session state has changed, restarting check session iframe; session_state\", session.session_state);\n this._userManager.events._raiseUserSessionChanged();\n }\n } else {\n logger2.debug(\"different subject signed into OP\", session.sub);\n }\n } else {\n logger2.debug(\"subject no longer signed into OP\");\n }\n if (raiseEvent) {\n if (this._sub) {\n this._userManager.events._raiseUserSignedOut();\n } else {\n this._userManager.events._raiseUserSignedIn();\n }\n } else {\n logger2.debug(\"no change in session detected, no event to raise\");\n }\n } catch (err) {\n if (this._sub) {\n logger2.debug(\"Error calling queryCurrentSigninSession; raising signed out event\", err);\n this._userManager.events._raiseUserSignedOut();\n }\n }\n };\n if (!_userManager) {\n this._logger.throw(new Error(\"No user manager passed\"));\n }\n this._userManager.events.addUserLoaded(this._start);\n this._userManager.events.addUserUnloaded(this._stop);\n this._init().catch((err) => {\n this._logger.error(err);\n });\n }\n async _init() {\n this._logger.create(\"_init\");\n const user = await this._userManager.getUser();\n if (user) {\n void this._start(user);\n } else if (this._userManager.settings.monitorAnonymousSession) {\n const session = await this._userManager.querySessionStatus();\n if (session) {\n const tmpUser = {\n session_state: session.session_state,\n profile: session.sub && session.sid ? {\n sub: session.sub,\n sid: session.sid\n } : null\n };\n void this._start(tmpUser);\n }\n }\n }\n};\n\n// src/User.ts\nvar User = class {\n constructor(args) {\n var _a;\n this.id_token = args.id_token;\n this.session_state = (_a = args.session_state) != null ? _a : null;\n this.access_token = args.access_token;\n this.refresh_token = args.refresh_token;\n this.token_type = args.token_type;\n this.scope = args.scope;\n this.profile = args.profile;\n this.expires_at = args.expires_at;\n this.state = args.userState;\n }\n /** Computed number of seconds the access token has remaining. */\n get expires_in() {\n if (this.expires_at === void 0) {\n return void 0;\n }\n return this.expires_at - Timer.getEpochTime();\n }\n set expires_in(value) {\n if (value !== void 0) {\n this.expires_at = Math.floor(value) + Timer.getEpochTime();\n }\n }\n /** Computed value indicating if the access token is expired. */\n get expired() {\n const expires_in = this.expires_in;\n if (expires_in === void 0) {\n return void 0;\n }\n return expires_in <= 0;\n }\n /** Array representing the parsed values from the `scope`. */\n get scopes() {\n var _a, _b;\n return (_b = (_a = this.scope) == null ? void 0 : _a.split(\" \")) != null ? _b : [];\n }\n toStorageString() {\n new Logger(\"User\").create(\"toStorageString\");\n return JSON.stringify({\n id_token: this.id_token,\n session_state: this.session_state,\n access_token: this.access_token,\n refresh_token: this.refresh_token,\n token_type: this.token_type,\n scope: this.scope,\n profile: this.profile,\n expires_at: this.expires_at\n });\n }\n static fromStorageString(storageString) {\n Logger.createStatic(\"User\", \"fromStorageString\");\n return new User(JSON.parse(storageString));\n }\n};\n\n// src/navigators/AbstractChildWindow.ts\nvar messageSource = \"oidc-client\";\nvar AbstractChildWindow = class {\n constructor() {\n this._abort = new Event(\"Window navigation aborted\");\n this._disposeHandlers = /* @__PURE__ */ new Set();\n this._window = null;\n }\n async navigate(params) {\n const logger2 = this._logger.create(\"navigate\");\n if (!this._window) {\n throw new Error(\"Attempted to navigate on a disposed window\");\n }\n logger2.debug(\"setting URL in window\");\n this._window.location.replace(params.url);\n const { url, keepOpen } = await new Promise((resolve, reject) => {\n const listener = (e) => {\n var _a;\n const data = e.data;\n const origin = (_a = params.scriptOrigin) != null ? _a : window.location.origin;\n if (e.origin !== origin || (data == null ? void 0 : data.source) !== messageSource) {\n return;\n }\n try {\n const state = UrlUtils.readParams(data.url, params.response_mode).get(\"state\");\n if (!state) {\n logger2.warn(\"no state found in response url\");\n }\n if (e.source !== this._window && state !== params.state) {\n return;\n }\n } catch (err) {\n this._dispose();\n reject(new Error(\"Invalid response from window\"));\n }\n resolve(data);\n };\n window.addEventListener(\"message\", listener, false);\n this._disposeHandlers.add(() => window.removeEventListener(\"message\", listener, false));\n this._disposeHandlers.add(this._abort.addHandler((reason) => {\n this._dispose();\n reject(reason);\n }));\n });\n logger2.debug(\"got response from window\");\n this._dispose();\n if (!keepOpen) {\n this.close();\n }\n return { url };\n }\n _dispose() {\n this._logger.create(\"_dispose\");\n for (const dispose of this._disposeHandlers) {\n dispose();\n }\n this._disposeHandlers.clear();\n }\n static _notifyParent(parent, url, keepOpen = false, targetOrigin = window.location.origin) {\n parent.postMessage({\n source: messageSource,\n url,\n keepOpen\n }, targetOrigin);\n }\n};\n\n// src/UserManagerSettings.ts\nvar DefaultPopupWindowFeatures = {\n location: false,\n toolbar: false,\n height: 640\n};\nvar DefaultPopupTarget = \"_blank\";\nvar DefaultAccessTokenExpiringNotificationTimeInSeconds = 60;\nvar DefaultCheckSessionIntervalInSeconds = 2;\nvar DefaultSilentRequestTimeoutInSeconds = 10;\nvar UserManagerSettingsStore = class extends OidcClientSettingsStore {\n constructor(args) {\n const {\n popup_redirect_uri = args.redirect_uri,\n popup_post_logout_redirect_uri = args.post_logout_redirect_uri,\n popupWindowFeatures = DefaultPopupWindowFeatures,\n popupWindowTarget = DefaultPopupTarget,\n redirectMethod = \"assign\",\n redirectTarget = \"self\",\n iframeNotifyParentOrigin = args.iframeNotifyParentOrigin,\n iframeScriptOrigin = args.iframeScriptOrigin,\n silent_redirect_uri = args.redirect_uri,\n silentRequestTimeoutInSeconds = DefaultSilentRequestTimeoutInSeconds,\n automaticSilentRenew = true,\n validateSubOnSilentRenew = true,\n includeIdTokenInSilentRenew = false,\n monitorSession = false,\n monitorAnonymousSession = false,\n checkSessionIntervalInSeconds = DefaultCheckSessionIntervalInSeconds,\n query_status_response_type = \"code\",\n stopCheckSessionOnError = true,\n revokeTokenTypes = [\"access_token\", \"refresh_token\"],\n revokeTokensOnSignout = false,\n includeIdTokenInSilentSignout = false,\n accessTokenExpiringNotificationTimeInSeconds = DefaultAccessTokenExpiringNotificationTimeInSeconds,\n userStore\n } = args;\n super(args);\n this.popup_redirect_uri = popup_redirect_uri;\n this.popup_post_logout_redirect_uri = popup_post_logout_redirect_uri;\n this.popupWindowFeatures = popupWindowFeatures;\n this.popupWindowTarget = popupWindowTarget;\n this.redirectMethod = redirectMethod;\n this.redirectTarget = redirectTarget;\n this.iframeNotifyParentOrigin = iframeNotifyParentOrigin;\n this.iframeScriptOrigin = iframeScriptOrigin;\n this.silent_redirect_uri = silent_redirect_uri;\n this.silentRequestTimeoutInSeconds = silentRequestTimeoutInSeconds;\n this.automaticSilentRenew = automaticSilentRenew;\n this.validateSubOnSilentRenew = validateSubOnSilentRenew;\n this.includeIdTokenInSilentRenew = includeIdTokenInSilentRenew;\n this.monitorSession = monitorSession;\n this.monitorAnonymousSession = monitorAnonymousSession;\n this.checkSessionIntervalInSeconds = checkSessionIntervalInSeconds;\n this.stopCheckSessionOnError = stopCheckSessionOnError;\n this.query_status_response_type = query_status_response_type;\n this.revokeTokenTypes = revokeTokenTypes;\n this.revokeTokensOnSignout = revokeTokensOnSignout;\n this.includeIdTokenInSilentSignout = includeIdTokenInSilentSignout;\n this.accessTokenExpiringNotificationTimeInSeconds = accessTokenExpiringNotificationTimeInSeconds;\n if (userStore) {\n this.userStore = userStore;\n } else {\n const store = typeof window !== \"undefined\" ? window.sessionStorage : new InMemoryWebStorage();\n this.userStore = new WebStorageStateStore({ store });\n }\n }\n};\n\n// src/navigators/IFrameWindow.ts\nvar IFrameWindow = class extends AbstractChildWindow {\n constructor({\n silentRequestTimeoutInSeconds = DefaultSilentRequestTimeoutInSeconds\n }) {\n super();\n this._logger = new Logger(\"IFrameWindow\");\n this._timeoutInSeconds = silentRequestTimeoutInSeconds;\n this._frame = IFrameWindow.createHiddenIframe();\n this._window = this._frame.contentWindow;\n }\n static createHiddenIframe() {\n const iframe = window.document.createElement(\"iframe\");\n iframe.style.visibility = \"hidden\";\n iframe.style.position = \"fixed\";\n iframe.style.left = \"-1000px\";\n iframe.style.top = \"0\";\n iframe.width = \"0\";\n iframe.height = \"0\";\n iframe.setAttribute(\"sandbox\", \"allow-scripts allow-same-origin allow-forms\");\n window.document.body.appendChild(iframe);\n return iframe;\n }\n async navigate(params) {\n this._logger.debug(\"navigate: Using timeout of:\", this._timeoutInSeconds);\n const timer = setTimeout(() => this._abort.raise(new ErrorTimeout(\"IFrame timed out without a response\")), this._timeoutInSeconds * 1e3);\n this._disposeHandlers.add(() => clearTimeout(timer));\n return await super.navigate(params);\n }\n close() {\n var _a;\n if (this._frame) {\n if (this._frame.parentNode) {\n this._frame.addEventListener(\"load\", (ev) => {\n var _a2;\n const frame = ev.target;\n (_a2 = frame.parentNode) == null ? void 0 : _a2.removeChild(frame);\n this._abort.raise(new Error(\"IFrame removed from DOM\"));\n }, true);\n (_a = this._frame.contentWindow) == null ? void 0 : _a.location.replace(\"about:blank\");\n }\n this._frame = null;\n }\n this._window = null;\n }\n static notifyParent(url, targetOrigin) {\n return super._notifyParent(window.parent, url, false, targetOrigin);\n }\n};\n\n// src/navigators/IFrameNavigator.ts\nvar IFrameNavigator = class {\n constructor(_settings) {\n this._settings = _settings;\n this._logger = new Logger(\"IFrameNavigator\");\n }\n async prepare({\n silentRequestTimeoutInSeconds = this._settings.silentRequestTimeoutInSeconds\n }) {\n return new IFrameWindow({ silentRequestTimeoutInSeconds });\n }\n async callback(url) {\n this._logger.create(\"callback\");\n IFrameWindow.notifyParent(url, this._settings.iframeNotifyParentOrigin);\n }\n};\n\n// src/navigators/PopupWindow.ts\nvar checkForPopupClosedInterval = 500;\nvar PopupWindow = class extends AbstractChildWindow {\n constructor({\n popupWindowTarget = DefaultPopupTarget,\n popupWindowFeatures = {}\n }) {\n super();\n this._logger = new Logger(\"PopupWindow\");\n const centeredPopup = PopupUtils.center({ ...DefaultPopupWindowFeatures, ...popupWindowFeatures });\n this._window = window.open(void 0, popupWindowTarget, PopupUtils.serialize(centeredPopup));\n }\n async navigate(params) {\n var _a;\n (_a = this._window) == null ? void 0 : _a.focus();\n const popupClosedInterval = setInterval(() => {\n if (!this._window || this._window.closed) {\n this._abort.raise(new Error(\"Popup closed by user\"));\n }\n }, checkForPopupClosedInterval);\n this._disposeHandlers.add(() => clearInterval(popupClosedInterval));\n return await super.navigate(params);\n }\n close() {\n if (this._window) {\n if (!this._window.closed) {\n this._window.close();\n this._abort.raise(new Error(\"Popup closed\"));\n }\n }\n this._window = null;\n }\n static notifyOpener(url, keepOpen) {\n if (!window.opener) {\n throw new Error(\"No window.opener. Can't complete notification.\");\n }\n return super._notifyParent(window.opener, url, keepOpen);\n }\n};\n\n// src/navigators/PopupNavigator.ts\nvar PopupNavigator = class {\n constructor(_settings) {\n this._settings = _settings;\n this._logger = new Logger(\"PopupNavigator\");\n }\n async prepare({\n popupWindowFeatures = this._settings.popupWindowFeatures,\n popupWindowTarget = this._settings.popupWindowTarget\n }) {\n return new PopupWindow({ popupWindowFeatures, popupWindowTarget });\n }\n async callback(url, keepOpen = false) {\n this._logger.create(\"callback\");\n PopupWindow.notifyOpener(url, keepOpen);\n }\n};\n\n// src/navigators/RedirectNavigator.ts\nvar RedirectNavigator = class {\n constructor(_settings) {\n this._settings = _settings;\n this._logger = new Logger(\"RedirectNavigator\");\n }\n async prepare({\n redirectMethod = this._settings.redirectMethod,\n redirectTarget = this._settings.redirectTarget\n }) {\n var _a;\n this._logger.create(\"prepare\");\n let targetWindow = window.self;\n if (redirectTarget === \"top\") {\n targetWindow = (_a = window.top) != null ? _a : window.self;\n }\n const redirect = targetWindow.location[redirectMethod].bind(targetWindow.location);\n let abort;\n return {\n navigate: async (params) => {\n this._logger.create(\"navigate\");\n const promise = new Promise((resolve, reject) => {\n abort = reject;\n });\n redirect(params.url);\n return await promise;\n },\n close: () => {\n this._logger.create(\"close\");\n abort == null ? void 0 : abort(new Error(\"Redirect aborted\"));\n targetWindow.stop();\n }\n };\n }\n};\n\n// src/UserManagerEvents.ts\nvar UserManagerEvents = class extends AccessTokenEvents {\n constructor(settings) {\n super({ expiringNotificationTimeInSeconds: settings.accessTokenExpiringNotificationTimeInSeconds });\n this._logger = new Logger(\"UserManagerEvents\");\n this._userLoaded = new Event(\"User loaded\");\n this._userUnloaded = new Event(\"User unloaded\");\n this._silentRenewError = new Event(\"Silent renew error\");\n this._userSignedIn = new Event(\"User signed in\");\n this._userSignedOut = new Event(\"User signed out\");\n this._userSessionChanged = new Event(\"User session changed\");\n }\n load(user, raiseEvent = true) {\n super.load(user);\n if (raiseEvent) {\n this._userLoaded.raise(user);\n }\n }\n unload() {\n super.unload();\n this._userUnloaded.raise();\n }\n /**\n * Add callback: Raised when a user session has been established (or re-established).\n */\n addUserLoaded(cb) {\n return this._userLoaded.addHandler(cb);\n }\n /**\n * Remove callback: Raised when a user session has been established (or re-established).\n */\n removeUserLoaded(cb) {\n return this._userLoaded.removeHandler(cb);\n }\n /**\n * Add callback: Raised when a user session has been terminated.\n */\n addUserUnloaded(cb) {\n return this._userUnloaded.addHandler(cb);\n }\n /**\n * Remove callback: Raised when a user session has been terminated.\n */\n removeUserUnloaded(cb) {\n return this._userUnloaded.removeHandler(cb);\n }\n /**\n * Add callback: Raised when the automatic silent renew has failed.\n */\n addSilentRenewError(cb) {\n return this._silentRenewError.addHandler(cb);\n }\n /**\n * Remove callback: Raised when the automatic silent renew has failed.\n */\n removeSilentRenewError(cb) {\n return this._silentRenewError.removeHandler(cb);\n }\n /**\n * @internal\n */\n _raiseSilentRenewError(e) {\n this._silentRenewError.raise(e);\n }\n /**\n * Add callback: Raised when the user is signed in (when `monitorSession` is set).\n * @see {@link UserManagerSettings.monitorSession}\n */\n addUserSignedIn(cb) {\n return this._userSignedIn.addHandler(cb);\n }\n /**\n * Remove callback: Raised when the user is signed in (when `monitorSession` is set).\n */\n removeUserSignedIn(cb) {\n this._userSignedIn.removeHandler(cb);\n }\n /**\n * @internal\n */\n _raiseUserSignedIn() {\n this._userSignedIn.raise();\n }\n /**\n * Add callback: Raised when the user's sign-in status at the OP has changed (when `monitorSession` is set).\n * @see {@link UserManagerSettings.monitorSession}\n */\n addUserSignedOut(cb) {\n return this._userSignedOut.addHandler(cb);\n }\n /**\n * Remove callback: Raised when the user's sign-in status at the OP has changed (when `monitorSession` is set).\n */\n removeUserSignedOut(cb) {\n this._userSignedOut.removeHandler(cb);\n }\n /**\n * @internal\n */\n _raiseUserSignedOut() {\n this._userSignedOut.raise();\n }\n /**\n * Add callback: Raised when the user session changed (when `monitorSession` is set).\n * @see {@link UserManagerSettings.monitorSession}\n */\n addUserSessionChanged(cb) {\n return this._userSessionChanged.addHandler(cb);\n }\n /**\n * Remove callback: Raised when the user session changed (when `monitorSession` is set).\n */\n removeUserSessionChanged(cb) {\n this._userSessionChanged.removeHandler(cb);\n }\n /**\n * @internal\n */\n _raiseUserSessionChanged() {\n this._userSessionChanged.raise();\n }\n};\n\n// src/SilentRenewService.ts\nvar SilentRenewService = class {\n constructor(_userManager) {\n this._userManager = _userManager;\n this._logger = new Logger(\"SilentRenewService\");\n this._isStarted = false;\n this._retryTimer = new Timer(\"Retry Silent Renew\");\n this._tokenExpiring = async () => {\n const logger2 = this._logger.create(\"_tokenExpiring\");\n try {\n await this._userManager.signinSilent();\n logger2.debug(\"silent token renewal successful\");\n } catch (err) {\n if (err instanceof ErrorTimeout) {\n logger2.warn(\"ErrorTimeout from signinSilent:\", err, \"retry in 5s\");\n this._retryTimer.init(5);\n return;\n }\n logger2.error(\"Error from signinSilent:\", err);\n this._userManager.events._raiseSilentRenewError(err);\n }\n };\n }\n async start() {\n const logger2 = this._logger.create(\"start\");\n if (!this._isStarted) {\n this._isStarted = true;\n this._userManager.events.addAccessTokenExpiring(this._tokenExpiring);\n this._retryTimer.addHandler(this._tokenExpiring);\n try {\n await this._userManager.getUser();\n } catch (err) {\n logger2.error(\"getUser error\", err);\n }\n }\n }\n stop() {\n if (this._isStarted) {\n this._retryTimer.cancel();\n this._retryTimer.removeHandler(this._tokenExpiring);\n this._userManager.events.removeAccessTokenExpiring(this._tokenExpiring);\n this._isStarted = false;\n }\n }\n};\n\n// src/RefreshState.ts\nvar RefreshState = class {\n constructor(args) {\n this.refresh_token = args.refresh_token;\n this.id_token = args.id_token;\n this.session_state = args.session_state;\n this.scope = args.scope;\n this.profile = args.profile;\n this.data = args.state;\n }\n};\n\n// src/UserManager.ts\nvar UserManager = class {\n constructor(settings) {\n this._logger = new Logger(\"UserManager\");\n this.settings = new UserManagerSettingsStore(settings);\n this._client = new OidcClient(settings);\n this._redirectNavigator = new RedirectNavigator(this.settings);\n this._popupNavigator = new PopupNavigator(this.settings);\n this._iframeNavigator = new IFrameNavigator(this.settings);\n this._events = new UserManagerEvents(this.settings);\n this._silentRenewService = new SilentRenewService(this);\n if (this.settings.automaticSilentRenew) {\n this.startSilentRenew();\n }\n this._sessionMonitor = null;\n if (this.settings.monitorSession) {\n this._sessionMonitor = new SessionMonitor(this);\n }\n }\n /** Returns an object used to register for events raised by the `UserManager`. */\n get events() {\n return this._events;\n }\n /** Returns an object used to access the metadata configuration of the OIDC provider. */\n get metadataService() {\n return this._client.metadataService;\n }\n /**\n * Returns promise to load the `User` object for the currently authenticated user.\n */\n async getUser() {\n const logger2 = this._logger.create(\"getUser\");\n const user = await this._loadUser();\n if (user) {\n logger2.info(\"user loaded\");\n this._events.load(user, false);\n return user;\n }\n logger2.info(\"user not found in storage\");\n return null;\n }\n /**\n * Returns promise to remove from any storage the currently authenticated user.\n */\n async removeUser() {\n const logger2 = this._logger.create(\"removeUser\");\n await this.storeUser(null);\n logger2.info(\"user removed from storage\");\n this._events.unload();\n }\n /**\n * Returns promise to trigger a redirect of the current window to the authorization endpoint.\n */\n async signinRedirect(args = {}) {\n this._logger.create(\"signinRedirect\");\n const {\n redirectMethod,\n ...requestArgs\n } = args;\n const handle = await this._redirectNavigator.prepare({ redirectMethod });\n await this._signinStart({\n request_type: \"si:r\",\n ...requestArgs\n }, handle);\n }\n /**\n * Returns promise to process response from the authorization endpoint. The result of the promise is the authenticated `User`.\n */\n async signinRedirectCallback(url = window.location.href) {\n const logger2 = this._logger.create(\"signinRedirectCallback\");\n const user = await this._signinEnd(url);\n if (user.profile && user.profile.sub) {\n logger2.info(\"success, signed in subject\", user.profile.sub);\n } else {\n logger2.info(\"no subject\");\n }\n return user;\n }\n /**\n * Returns promise to process the signin with user/password. The result of the promise is the authenticated `User`.\n *\n * Throws an ErrorResponse in case of wrong authentication.\n */\n async signinResourceOwnerCredentials({\n username,\n password,\n skipUserInfo = false\n }) {\n const logger2 = this._logger.create(\"signinResourceOwnerCredential\");\n const signinResponse = await this._client.processResourceOwnerPasswordCredentials({ username, password, skipUserInfo, extraTokenParams: this.settings.extraTokenParams });\n logger2.debug(\"got signin response\");\n const user = await this._buildUser(signinResponse);\n if (user.profile && user.profile.sub) {\n logger2.info(\"success, signed in subject\", user.profile.sub);\n } else {\n logger2.info(\"no subject\");\n }\n return user;\n }\n /**\n * Returns promise to trigger a request (via a popup window) to the authorization endpoint. The result of the promise is the authenticated `User`.\n */\n async signinPopup(args = {}) {\n const logger2 = this._logger.create(\"signinPopup\");\n const {\n popupWindowFeatures,\n popupWindowTarget,\n ...requestArgs\n } = args;\n const url = this.settings.popup_redirect_uri;\n if (!url) {\n logger2.throw(new Error(\"No popup_redirect_uri configured\"));\n }\n const handle = await this._popupNavigator.prepare({ popupWindowFeatures, popupWindowTarget });\n const user = await this._signin({\n request_type: \"si:p\",\n redirect_uri: url,\n display: \"popup\",\n ...requestArgs\n }, handle);\n if (user) {\n if (user.profile && user.profile.sub) {\n logger2.info(\"success, signed in subject\", user.profile.sub);\n } else {\n logger2.info(\"no subject\");\n }\n }\n return user;\n }\n /**\n * Returns promise to notify the opening window of response from the authorization endpoint.\n */\n async signinPopupCallback(url = window.location.href, keepOpen = false) {\n const logger2 = this._logger.create(\"signinPopupCallback\");\n await this._popupNavigator.callback(url, keepOpen);\n logger2.info(\"success\");\n }\n /**\n * Returns promise to trigger a silent request (via an iframe) to the authorization endpoint.\n * The result of the promise is the authenticated `User`.\n */\n async signinSilent(args = {}) {\n var _a;\n const logger2 = this._logger.create(\"signinSilent\");\n const {\n silentRequestTimeoutInSeconds,\n ...requestArgs\n } = args;\n let user = await this._loadUser();\n if (user == null ? void 0 : user.refresh_token) {\n logger2.debug(\"using refresh token\");\n const state = new RefreshState(user);\n return await this._useRefreshToken(state);\n }\n const url = this.settings.silent_redirect_uri;\n if (!url) {\n logger2.throw(new Error(\"No silent_redirect_uri configured\"));\n }\n let verifySub;\n if (user && this.settings.validateSubOnSilentRenew) {\n logger2.debug(\"subject prior to silent renew:\", user.profile.sub);\n verifySub = user.profile.sub;\n }\n const handle = await this._iframeNavigator.prepare({ silentRequestTimeoutInSeconds });\n user = await this._signin({\n request_type: \"si:s\",\n redirect_uri: url,\n prompt: \"none\",\n id_token_hint: this.settings.includeIdTokenInSilentRenew ? user == null ? void 0 : user.id_token : void 0,\n ...requestArgs\n }, handle, verifySub);\n if (user) {\n if ((_a = user.profile) == null ? void 0 : _a.sub) {\n logger2.info(\"success, signed in subject\", user.profile.sub);\n } else {\n logger2.info(\"no subject\");\n }\n }\n return user;\n }\n async _useRefreshToken(state) {\n const response = await this._client.useRefreshToken({\n state,\n timeoutInSeconds: this.settings.silentRequestTimeoutInSeconds\n });\n const user = new User({ ...state, ...response });\n await this.storeUser(user);\n this._events.load(user);\n return user;\n }\n /**\n * Returns promise to notify the parent window of response from the authorization endpoint.\n */\n async signinSilentCallback(url = window.location.href) {\n const logger2 = this._logger.create(\"signinSilentCallback\");\n await this._iframeNavigator.callback(url);\n logger2.info(\"success\");\n }\n async signinCallback(url = window.location.href) {\n const { state } = await this._client.readSigninResponseState(url);\n switch (state.request_type) {\n case \"si:r\":\n return await this.signinRedirectCallback(url);\n case \"si:p\":\n return await this.signinPopupCallback(url);\n case \"si:s\":\n return await this.signinSilentCallback(url);\n default:\n throw new Error(\"invalid response_type in state\");\n }\n }\n async signoutCallback(url = window.location.href, keepOpen = false) {\n const { state } = await this._client.readSignoutResponseState(url);\n if (!state) {\n return;\n }\n switch (state.request_type) {\n case \"so:r\":\n await this.signoutRedirectCallback(url);\n break;\n case \"so:p\":\n await this.signoutPopupCallback(url, keepOpen);\n break;\n case \"so:s\":\n await this.signoutSilentCallback(url);\n break;\n default:\n throw new Error(\"invalid response_type in state\");\n }\n }\n /**\n * Returns promise to query OP for user's current signin status. Returns object with session_state and subject identifier.\n */\n async querySessionStatus(args = {}) {\n const logger2 = this._logger.create(\"querySessionStatus\");\n const {\n silentRequestTimeoutInSeconds,\n ...requestArgs\n } = args;\n const url = this.settings.silent_redirect_uri;\n if (!url) {\n logger2.throw(new Error(\"No silent_redirect_uri configured\"));\n }\n const user = await this._loadUser();\n const handle = await this._iframeNavigator.prepare({ silentRequestTimeoutInSeconds });\n const navResponse = await this._signinStart({\n request_type: \"si:s\",\n // this acts like a signin silent\n redirect_uri: url,\n prompt: \"none\",\n id_token_hint: this.settings.includeIdTokenInSilentRenew ? user == null ? void 0 : user.id_token : void 0,\n response_type: this.settings.query_status_response_type,\n scope: \"openid\",\n skipUserInfo: true,\n ...requestArgs\n }, handle);\n try {\n const signinResponse = await this._client.processSigninResponse(navResponse.url);\n logger2.debug(\"got signin response\");\n if (signinResponse.session_state && signinResponse.profile.sub) {\n logger2.info(\"success for subject\", signinResponse.profile.sub);\n return {\n session_state: signinResponse.session_state,\n sub: signinResponse.profile.sub,\n sid: signinResponse.profile.sid\n };\n }\n logger2.info(\"success, user not authenticated\");\n return null;\n } catch (err) {\n if (this.settings.monitorAnonymousSession && err instanceof ErrorResponse) {\n switch (err.error) {\n case \"login_required\":\n case \"consent_required\":\n case \"interaction_required\":\n case \"account_selection_required\":\n logger2.info(\"success for anonymous user\");\n return {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n session_state: err.session_state\n };\n }\n }\n throw err;\n }\n }\n async _signin(args, handle, verifySub) {\n const navResponse = await this._signinStart(args, handle);\n return await this._signinEnd(navResponse.url, verifySub);\n }\n async _signinStart(args, handle) {\n const logger2 = this._logger.create(\"_signinStart\");\n try {\n const signinRequest = await this._client.createSigninRequest(args);\n logger2.debug(\"got signin request\");\n return await handle.navigate({\n url: signinRequest.url,\n state: signinRequest.state.id,\n response_mode: signinRequest.state.response_mode,\n scriptOrigin: this.settings.iframeScriptOrigin\n });\n } catch (err) {\n logger2.debug(\"error after preparing navigator, closing navigator window\");\n handle.close();\n throw err;\n }\n }\n async _signinEnd(url, verifySub) {\n const logger2 = this._logger.create(\"_signinEnd\");\n const signinResponse = await this._client.processSigninResponse(url);\n logger2.debug(\"got signin response\");\n const user = await this._buildUser(signinResponse, verifySub);\n return user;\n }\n async _buildUser(signinResponse, verifySub) {\n const logger2 = this._logger.create(\"_buildUser\");\n const user = new User(signinResponse);\n if (verifySub) {\n if (verifySub !== user.profile.sub) {\n logger2.debug(\"current user does not match user returned from signin. sub from signin:\", user.profile.sub);\n throw new ErrorResponse({ ...signinResponse, error: \"login_required\" });\n }\n logger2.debug(\"current user matches user returned from signin\");\n }\n await this.storeUser(user);\n logger2.debug(\"user stored\");\n this._events.load(user);\n return user;\n }\n /**\n * Returns promise to trigger a redirect of the current window to the end session endpoint.\n */\n async signoutRedirect(args = {}) {\n const logger2 = this._logger.create(\"signoutRedirect\");\n const {\n redirectMethod,\n ...requestArgs\n } = args;\n const handle = await this._redirectNavigator.prepare({ redirectMethod });\n await this._signoutStart({\n request_type: \"so:r\",\n post_logout_redirect_uri: this.settings.post_logout_redirect_uri,\n ...requestArgs\n }, handle);\n logger2.info(\"success\");\n }\n /**\n * Returns promise to process response from the end session endpoint.\n */\n async signoutRedirectCallback(url = window.location.href) {\n const logger2 = this._logger.create(\"signoutRedirectCallback\");\n const response = await this._signoutEnd(url);\n logger2.info(\"success\");\n return response;\n }\n /**\n * Returns promise to trigger a redirect of a popup window window to the end session endpoint.\n */\n async signoutPopup(args = {}) {\n const logger2 = this._logger.create(\"signoutPopup\");\n const {\n popupWindowFeatures,\n popupWindowTarget,\n ...requestArgs\n } = args;\n const url = this.settings.popup_post_logout_redirect_uri;\n const handle = await this._popupNavigator.prepare({ popupWindowFeatures, popupWindowTarget });\n await this._signout({\n request_type: \"so:p\",\n post_logout_redirect_uri: url,\n // we're putting a dummy entry in here because we\n // need a unique id from the state for notification\n // to the parent window, which is necessary if we\n // plan to return back to the client after signout\n // and so we can close the popup after signout\n state: url == null ? void 0 : {},\n ...requestArgs\n }, handle);\n logger2.info(\"success\");\n }\n /**\n * Returns promise to process response from the end session endpoint from a popup window.\n */\n async signoutPopupCallback(url = window.location.href, keepOpen = false) {\n const logger2 = this._logger.create(\"signoutPopupCallback\");\n await this._popupNavigator.callback(url, keepOpen);\n logger2.info(\"success\");\n }\n async _signout(args, handle) {\n const navResponse = await this._signoutStart(args, handle);\n return await this._signoutEnd(navResponse.url);\n }\n async _signoutStart(args = {}, handle) {\n var _a;\n const logger2 = this._logger.create(\"_signoutStart\");\n try {\n const user = await this._loadUser();\n logger2.debug(\"loaded current user from storage\");\n if (this.settings.revokeTokensOnSignout) {\n await this._revokeInternal(user);\n }\n const id_token = args.id_token_hint || user && user.id_token;\n if (id_token) {\n logger2.debug(\"setting id_token_hint in signout request\");\n args.id_token_hint = id_token;\n }\n await this.removeUser();\n logger2.debug(\"user removed, creating signout request\");\n const signoutRequest = await this._client.createSignoutRequest(args);\n logger2.debug(\"got signout request\");\n return await handle.navigate({\n url: signoutRequest.url,\n state: (_a = signoutRequest.state) == null ? void 0 : _a.id\n });\n } catch (err) {\n logger2.debug(\"error after preparing navigator, closing navigator window\");\n handle.close();\n throw err;\n }\n }\n async _signoutEnd(url) {\n const logger2 = this._logger.create(\"_signoutEnd\");\n const signoutResponse = await this._client.processSignoutResponse(url);\n logger2.debug(\"got signout response\");\n return signoutResponse;\n }\n /**\n * Returns promise to trigger a silent request (via an iframe) to the end session endpoint.\n */\n async signoutSilent(args = {}) {\n var _a;\n const logger2 = this._logger.create(\"signoutSilent\");\n const {\n silentRequestTimeoutInSeconds,\n ...requestArgs\n } = args;\n const id_token_hint = this.settings.includeIdTokenInSilentSignout ? (_a = await this._loadUser()) == null ? void 0 : _a.id_token : void 0;\n const url = this.settings.popup_post_logout_redirect_uri;\n const handle = await this._iframeNavigator.prepare({ silentRequestTimeoutInSeconds });\n await this._signout({\n request_type: \"so:s\",\n post_logout_redirect_uri: url,\n id_token_hint,\n ...requestArgs\n }, handle);\n logger2.info(\"success\");\n }\n /**\n * Returns promise to notify the parent window of response from the end session endpoint.\n */\n async signoutSilentCallback(url = window.location.href) {\n const logger2 = this._logger.create(\"signoutSilentCallback\");\n await this._iframeNavigator.callback(url);\n logger2.info(\"success\");\n }\n async revokeTokens(types) {\n const user = await this._loadUser();\n await this._revokeInternal(user, types);\n }\n async _revokeInternal(user, types = this.settings.revokeTokenTypes) {\n const logger2 = this._logger.create(\"_revokeInternal\");\n if (!user)\n return;\n const typesPresent = types.filter((type) => typeof user[type] === \"string\");\n if (!typesPresent.length) {\n logger2.debug(\"no need to revoke due to no token(s)\");\n return;\n }\n for (const type of typesPresent) {\n await this._client.revokeToken(\n user[type],\n // eslint-disable-line @typescript-eslint/no-non-null-assertion\n type\n );\n logger2.info(`${type} revoked successfully`);\n if (type !== \"access_token\") {\n user[type] = null;\n }\n }\n await this.storeUser(user);\n logger2.debug(\"user stored\");\n this._events.load(user);\n }\n /**\n * Enables silent renew for the `UserManager`.\n */\n startSilentRenew() {\n this._logger.create(\"startSilentRenew\");\n void this._silentRenewService.start();\n }\n /**\n * Disables silent renew for the `UserManager`.\n */\n stopSilentRenew() {\n this._silentRenewService.stop();\n }\n get _userStoreKey() {\n return `user:${this.settings.authority}:${this.settings.client_id}`;\n }\n async _loadUser() {\n const logger2 = this._logger.create(\"_loadUser\");\n const storageString = await this.settings.userStore.get(this._userStoreKey);\n if (storageString) {\n logger2.debug(\"user storageString loaded\");\n return User.fromStorageString(storageString);\n }\n logger2.debug(\"no user storageString\");\n return null;\n }\n async storeUser(user) {\n const logger2 = this._logger.create(\"storeUser\");\n if (user) {\n logger2.debug(\"storing user\");\n const storageString = user.toStorageString();\n await this.settings.userStore.set(this._userStoreKey, storageString);\n } else {\n this._logger.debug(\"removing user\");\n await this.settings.userStore.remove(this._userStoreKey);\n }\n }\n /**\n * Removes stale state entries in storage for incomplete authorize requests.\n */\n async clearStaleState() {\n await this._client.clearStaleState();\n }\n};\n\n// package.json\nvar version = \"2.2.4\";\n\n// src/Version.ts\nvar Version = version;\nexport {\n AccessTokenEvents,\n CheckSessionIFrame,\n ErrorResponse,\n ErrorTimeout,\n InMemoryWebStorage,\n Log,\n Logger,\n MetadataService,\n OidcClient,\n OidcClientSettingsStore,\n SessionMonitor,\n SigninResponse,\n SigninState,\n SignoutResponse,\n State,\n User,\n UserManager,\n UserManagerSettingsStore,\n Version,\n WebStorageStateStore\n};\n//# sourceMappingURL=oidc-client-ts.js.map\n","import { UserManager, WebStorageStateStore, User } from \"oidc-client-ts\";\nimport axios from \"axios\";\nimport { appSettings } from \"../appSettings\";\n\nconst IMP_USER = \"IMP_user\";\nconst IMP_SLUG = \"IMP_slug\";\nconst DEL_CA = \"DELEGATE_childAcct\";\n\nclass AuthService {\n private userManager: UserManager;\n private storage: Storage;\n\n private http = axios.create({\n baseURL: appSettings.backEnd.baseUrlAndPath,\n });\n\n constructor() {\n const origin = window.location.origin;\n const settings = {\n client_id: \"app-builder.client\",\n response_type: \"code\",\n scope: \"openid profile app-builder.backend.api\",\n authority: appSettings.openId.baseUrl,\n redirect_uri: `${origin}/auth/callback.html`,\n silent_redirect_uri: `${origin}/auth/silent-renew.html`,\n post_logout_redirect_uri: origin,\n automaticSilentRenew: true,\n filterProtocolClaims: true,\n loadUserInfo: true,\n monitorSession: true,\n userStore: new WebStorageStateStore({ store: window.localStorage }),\n };\n\n this.userManager = new UserManager(settings);\n this.userManager.clearStaleState();\n\n this.userManager.events.addSilentRenewError(() => {\n const host = window.location.host;\n if (!host.includes(\"localhost\")) this.login(window.location.href);\n });\n\n this.userManager.events.addUserSignedOut(() => {\n sessionStorage.removeItem(DEL_CA);\n });\n\n this.storage = window.localStorage;\n\n window.addEventListener(\"storage\", async (e) =>\n this.checkDelegateSession(e.key)\n );\n }\n\n // Used to redirect a open tab that is still pointing to a different child account\n // delegate session. This will redirect to the home page and reload the current child account\n // session.\n private async checkDelegateSession(key: string | null) {\n if (!key || !key.includes(\"oidc.user\")) return;\n\n const sessionDelegateAcct = sessionStorage.getItem(DEL_CA);\n if (sessionDelegateAcct) {\n const childAcct = await this.getDelgateForAccountId();\n if (childAcct) {\n if (Number(sessionDelegateAcct) !== childAcct) {\n sessionStorage.setItem(DEL_CA, childAcct.toString());\n document.location.href = \"/\";\n }\n }\n }\n }\n\n public async delegate() {\n // force a re-fetch/re-signin of the user in order to get any updated claims\n // required for switching between child accounts when delegating\n await this.userManager.removeUser();\n }\n\n public impersonate(slug: string | (string | null)[]): void {\n if (typeof slug !== \"string\") return;\n\n this.storage.removeItem(IMP_USER);\n this.storage.setItem(IMP_SLUG, slug);\n }\n\n public getUser(): Promise {\n const storedUser = this.storage.getItem(IMP_USER);\n if (storedUser) {\n return Promise.resolve(JSON.parse(storedUser));\n }\n\n return this.userManager.getUser();\n }\n\n public login(returnUrl: string): Promise {\n return this.userManager.signinRedirect({ state: returnUrl || \"/\" });\n }\n\n public logout(): Promise {\n this.storage.removeItem(IMP_SLUG);\n this.storage.removeItem(IMP_USER);\n return this.userManager.signoutRedirect();\n }\n\n public isImpersonating(): boolean {\n const storedUser = this.storage.getItem(IMP_USER);\n return storedUser ? true : false;\n }\n\n public async getAccessToken(): Promise {\n const storedSlug = this.storage.getItem(IMP_SLUG);\n if (storedSlug) return `imp:${storedSlug}`;\n\n const user = await this.userManager.getUser();\n return user == null || user.expired || !user.access_token\n ? null\n : user.access_token;\n }\n\n public async getOrRenewAccessToken(): Promise {\n const storedSlug = this.storage.getItem(IMP_SLUG);\n if (storedSlug) return `imp:${storedSlug}`;\n\n let user = await this.userManager.getUser();\n if (user == null || user.expired || !user.access_token)\n user = await this.userManager.signinSilent();\n\n return user == null || user.expired || !user.access_token\n ? null\n : user.access_token;\n }\n\n public async isLoggedIn(): Promise {\n let user = null;\n\n const storedUser = this.storage.getItem(IMP_USER);\n if (storedUser) {\n user = JSON.parse(storedUser);\n return true;\n }\n\n const storedSlug = this.storage.getItem(IMP_SLUG);\n if (storedSlug) {\n try {\n const resp = await this.http.get(\"users/impersonated\", {\n headers: { Authorization: `Bearer imp:${storedSlug}` },\n });\n const d = resp.data.data;\n user = new User({\n id_token: storedSlug,\n session_state: \"\",\n access_token: storedSlug,\n refresh_token: storedSlug,\n token_type: \"\",\n scope: \"openid profile app-builder.backend.api\",\n expires_at: 0,\n profile: d.profile,\n userState: null,\n });\n this.storage.setItem(IMP_USER, JSON.stringify(user));\n return true;\n } catch (error) {\n this.storage.removeItem(IMP_SLUG);\n return false;\n }\n }\n\n user = await this.userManager.getUser();\n\n const delegateForAcctId = await this.getDelgateForAccountId();\n if (delegateForAcctId)\n sessionStorage.setItem(DEL_CA, delegateForAcctId.toString());\n\n return user != null && !user.expired;\n }\n\n public async getAccountId(): Promise {\n return (\n Number(await this.getCustomClaims(\"accountId\")) || null\n );\n }\n\n private async getDelgateForAccountId(): Promise {\n return (\n Number(await this.getCustomClaims(\"childAccountId\")) ||\n null\n );\n }\n\n public async getActiveAccountId(): Promise {\n const [childAcctId, acctId] = await Promise.all([\n this.getDelgateForAccountId(),\n this.getAccountId(),\n ]);\n return childAcctId ?? acctId;\n }\n\n private superAdminId = 1;\n private templateAdminId = 10305;\n public async isAdmin(): Promise {\n const accountId = await this.getAccountId();\n return accountId === this.superAdminId;\n }\n\n public async isTemplateAdmin(): Promise {\n const accountId = await this.getAccountId();\n return (\n accountId === this.superAdminId || accountId === this.templateAdminId\n );\n }\n\n private async getCustomClaims(claimName: string): Promise {\n const user = await this.getUser();\n return (\n (user?.profile[\n `https://schema.screenfeed.com/claims/${claimName}`\n ] as T) || null\n );\n }\n\n public handleCallback() {\n this.userManager\n .signinRedirectCallback()\n .then((user) => {\n location.href = (user.state as string) || \"/\";\n })\n .catch(() => {\n location.href = \"/\";\n });\n }\n\n public handleSilentRenew() {\n this.userManager.signinSilentCallback();\n }\n}\n\nexport const auth = new AuthService();\n","import axios, { AxiosInstance, AxiosRequestConfig, AxiosResponse } from \"axios\";\nimport { appSettings } from \"../appSettings\";\nimport { auth } from \"./authService\";\n\nexport interface BackendRequestConfig extends AxiosRequestConfig {\n anonymous?: boolean;\n}\n\nexport interface PagedResult {\n page: number;\n pageSize?: number;\n totalItems: number;\n}\n\nexport interface BackendResponseMeta {\n paging?: PagedResult;\n}\n\nexport interface BackendApiResponse {\n data: T;\n meta: BackendResponseMeta;\n errors: BackendError[];\n}\n\nexport interface BackendError {\n code: string;\n message: string;\n}\n\nclass ApiClient {\n private instance = axios.create({\n baseURL: appSettings.backEnd.baseUrlAndPath,\n });\n\n constructor() {\n this.instance.interceptors.response.use(\n (response) => {\n return response;\n },\n async (error) => {\n let originalRequest = error.config;\n\n if (error.response.status === 401 && originalRequest._rety)\n return auth.logout();\n\n if (error.response.status === 401) {\n originalRequest._rety = true;\n\n if (auth.isImpersonating()) auth.logout();\n\n return (await auth.getOrRenewAccessToken())\n ? this.instance.request(originalRequest)\n : auth.logout();\n }\n\n return Promise.reject(error);\n }\n );\n }\n\n delete(\n url: string,\n config?: BackendRequestConfig | undefined\n ): Promise {\n return this.callWithAuth>(config, (a, c) =>\n a.delete(url, c)\n )\n .then((response) => {\n return response.data.data;\n })\n .catch((error) => {\n return Promise.reject(error?.response?.data?.errors ?? error?.message);\n });\n }\n\n get(url: string, config?: BackendRequestConfig | undefined): Promise {\n return this.callWithAuth>(config, (a, c) =>\n a.get(url, c)\n )\n .then((response) => {\n return response.data.data;\n })\n .catch((error) => {\n return Promise.reject(error?.response?.data?.errors ?? error?.message);\n });\n }\n\n post(\n url: string,\n data?: any,\n config?: BackendRequestConfig | undefined\n ): Promise {\n return this.callWithAuth>(config, (a, c) =>\n a.post(url, data, c)\n )\n .then((response) => {\n return response.data.data;\n })\n .catch((error) => {\n return Promise.reject(error?.response?.data?.errors ?? error?.message);\n });\n }\n\n put(\n url: string,\n data?: any,\n config?: BackendRequestConfig | undefined\n ): Promise {\n return this.callWithAuth>(config, (a, c) =>\n a.put(url, data, c)\n )\n .then((response) => {\n return response.data.data;\n })\n .catch((error) => {\n return Promise.reject(error?.response?.data?.errors ?? error?.message);\n });\n }\n\n private async callWithAuth(\n config: BackendRequestConfig | undefined,\n fn: (\n a: AxiosInstance,\n c: AxiosRequestConfig | undefined\n ) => Promise>\n ): Promise> {\n if (\n (!config || !config.anonymous) &&\n (!config || !config.headers || !config.headers[\"Authorization\"])\n ) {\n try {\n const token = await auth.getOrRenewAccessToken();\n if (token != null) {\n config = config || {};\n config.headers = config.headers || {};\n config.headers[\"Authorization\"] = `Bearer ${token}`;\n }\n } catch (error) {\n return Promise.reject(error);\n }\n }\n\n return await fn(this.instance, config);\n }\n}\n\nexport const api = new ApiClient();\n","import Vue from \"vue\";\nimport { defineStore } from \"pinia\";\n\nimport {\n Condition,\n ConditionGroup,\n ConditionGroupWidgetRef,\n Logic,\n} from \"@/types/logic\";\nimport { DEFAULT_CONDITION_ID } from \"@/constants\";\nimport { api } from \"@/api/backend\";\nimport { useAppEditorStore } from \"./appEditor\";\n\n/**\n * We will update the Widget interface to embed all condition-scope-able properties under a conditionVersions dictionary.\n *\n * \"addWidget\" will be updated, so that all new widgets are created in this new shape.\n * And all methods for reading/updating widget properties will be updated to use this new shape.\n *\n * To ensure backwards compatibility, we should loop through all widgets when an app loads, and put them into this new shape, if they are not already in it.\n */\n\nexport interface ConditionGroupsState {\n /**\n * Stores all condition groups that are attached to any widget within the app.\n */\n conditionGroups: ConditionGroup[];\n\n /**\n * Stores record of which Condition is actively being edited, for each widget (keyed by widgetId).\n */\n activeWidgetConditionsMap: Record;\n\n /**\n * Stores most recently \"copied\" condition group uuid, for pasting\n */\n copiedConditionGroupUuid: string | null;\n\n /**\n * A flag that indicates whether the frontend is awaiting the backend to refresh widget data under a condition.\n */\n refreshingConditionUuid: string | null;\n}\n\nexport interface CopiedConditionGroupPayload {\n conditionGroup: ConditionGroup;\n // copyFromToConditionUuids: Record;\n}\n\nexport const useConditionGroupsStore = defineStore(\"conditionGroups\", {\n state: (): ConditionGroupsState => {\n return {\n conditionGroups: [],\n activeWidgetConditionsMap: {},\n copiedConditionGroupUuid: null,\n refreshingConditionUuid: null,\n };\n },\n\n getters: {\n getActiveConditionId: (state) => (wid: string) =>\n state.activeWidgetConditionsMap[wid] ?? DEFAULT_CONDITION_ID,\n },\n\n actions: {\n getConditionGroups(appUuid: string) {\n return api.get(`apps/${appUuid}/conditiongroups`).then((data) => {\n this.conditionGroups = data as ConditionGroup[];\n });\n },\n\n getConditionGroup(payload: {\n appUuid: string;\n conditionGroupUuid: string;\n }) {\n const { appUuid, conditionGroupUuid } = payload;\n return api.get(`apps/${appUuid}/conditiongroups/${conditionGroupUuid}`);\n },\n\n removeConditionGroup(payload: {\n conditionGroupUuid: string;\n widgetId: string;\n appUuid: string;\n }) {\n const appEditor = useAppEditorStore();\n const { appUuid, widgetId, conditionGroupUuid } = payload;\n return api\n .delete(\n `apps/${appUuid}/conditiongroups/${conditionGroupUuid}?widgetId=${widgetId}`\n )\n .then(() => {\n const conditionUuids = this.conditionGroups\n .find((cg) => cg.uuid === conditionGroupUuid)\n ?.conditions.map((c) => c.uuid);\n\n // Remove conditional data bindings associated with the widget from which the condition group was removed\n if (conditionUuids !== undefined && conditionUuids.length > 0) {\n appEditor.removeDataBindingsForConditions({\n widgetId,\n conditionUuids,\n });\n }\n\n // Remove conditions from Widget Conditions array\n if (\n typeof conditionUuids !== \"undefined\" &&\n conditionUuids?.length > 0\n ) {\n appEditor.removeWidgetConditions({\n widgetIds: [widgetId],\n conditionUuids,\n });\n }\n })\n .then(() => {\n // Ensure \"publish\" button updates to show \"republish\"\n appEditor.updateApp();\n });\n },\n\n async setActiveCondition(payload: {\n conditionGroupUuid: string;\n conditionUuid: string;\n }): Promise {\n const { conditionGroupUuid, conditionUuid } = payload;\n\n const appEditor = useAppEditorStore();\n\n /**\n * Update all widgets that share a reference to the selected widget's conditionGroup\n */\n const conditionGroup = this.conditionGroups.find(\n (group) => group.uuid === conditionGroupUuid\n );\n\n const widgetRefs = conditionGroup?.widgets || [];\n\n widgetRefs.forEach((widget) => {\n Vue.set(this.activeWidgetConditionsMap, widget.widgetId, conditionUuid);\n });\n\n /**\n * Reset undo/redo history when user enters new conditional editing context.\n * Must call \"clearStack\" from the Header, to refer to same stack.\n * Do this regardless of whether there was an error refreshing widget data.\n */\n appEditor.resetUndoRedoState();\n },\n\n copyConditionGroup(payload: {\n appUuid: string;\n conditionGroupUuid?: string;\n widgetId: string;\n parentWidgetId?: string;\n }): Promise {\n const { appUuid, widgetId, parentWidgetId, conditionGroupUuid } = payload;\n const cgUuid = conditionGroupUuid ?? this.copiedConditionGroupUuid;\n return api.post(`apps/${appUuid}/conditiongroups/${cgUuid}/copy`, {\n widgetId,\n parentWidgetId,\n });\n },\n\n // Note, there is also updateConditionGroup, for ordering\n\n getCondition(payload: { appUuid: string; conditionUuid: string }) {\n const { appUuid, conditionUuid } = payload;\n return api.get(`apps/${appUuid}/conditions/${conditionUuid}`);\n },\n\n createCondition(payload: {\n appUuid: string;\n condition: Logic;\n widgetId: string;\n parentWidgetId?: string;\n }): Promise {\n const { appUuid, condition, widgetId, parentWidgetId } = payload;\n const data: any = { condition };\n const widgetRef: ConditionGroupWidgetRef = {\n widgetId,\n };\n if (parentWidgetId) {\n widgetRef.parentRepeaterWidgetId = parentWidgetId;\n }\n data.widgets = [widgetRef];\n console.log(\"Posting data\", data);\n return api.post(`apps/${appUuid}/conditions`, data);\n },\n\n /**\n * Removes a condition from a condition group.\n * This affects all widgets that reference the condition group.\n * If the condition being deleted is the only condition in the group,\n * the group is deleted as well.\n * @param conditionUuid\n * @param conditionGroupUuid\n * @returns\n */\n async deleteCondition(conditionUuid: string, conditionGroupUuid: string) {\n const appEditor = useAppEditorStore();\n const appUuid = appEditor.uuid;\n\n const group = this.conditionGroups.find(\n (cg) => cg.uuid === conditionGroupUuid\n );\n\n if (typeof group === \"undefined\") {\n return Promise.reject(\n `Condition group ${conditionGroupUuid} not found`\n );\n }\n\n const condition = group.conditions.find((c) => c.uuid === conditionUuid);\n if (typeof condition === \"undefined\") {\n return Promise.reject(`Condition ${conditionUuid} not found`);\n }\n\n // Delete condition on the backend\n await api.delete(`apps/${appUuid}/conditions/${conditionUuid}`);\n\n // Get all widgetIds associated with this condition\n const widgetIds = this.conditionGroups.reduce((acc, cg) => {\n cg.conditions.forEach((c) => {\n if (c.uuid === conditionUuid) {\n acc.push(...cg.widgets.map((w) => w.widgetId));\n }\n });\n return acc;\n }, [] as string[]);\n\n // Remove condition from Widget Conditions array\n appEditor.removeWidgetConditions({\n widgetIds: widgetIds,\n conditionUuids: [conditionUuid],\n });\n\n // Remove data bindings associated with this conditionUuid, across all widgets attached to the condition group\n appEditor.removeDataBindingsForConditions({\n conditionUuids: [conditionUuid],\n });\n\n // If the condition being deleted is active for any widget,\n // reset that widget's active conditon to default.\n for (const key in this.activeWidgetConditionsMap) {\n if (this.activeWidgetConditionsMap[key] === conditionUuid) {\n console.info(\"Resetting active condition for widget to default\", key);\n this.activeWidgetConditionsMap[key] = DEFAULT_CONDITION_ID;\n }\n }\n\n // Remove the condition from the condition group\n const index = group.conditions.findIndex((c) => c.uuid === conditionUuid);\n if (index > -1) {\n group.conditions.splice(index, 1);\n }\n\n // Check if the condition we just deleted was the LAST condition in the group.\n // If so, delete the group.\n if (group.conditions.length === 0) {\n // Delete condition group from local state\n this.conditionGroups = this.conditionGroups.filter(\n (cg) => cg.uuid !== conditionGroupUuid\n );\n }\n },\n\n updateSortOrder(\n appUuid: string,\n conditionGroupUuid: string,\n orderedIds: string[]\n ) {\n const group = this.conditionGroups.find(\n (cg) => cg.uuid === conditionGroupUuid\n );\n if (typeof group === \"undefined\") {\n return Promise.resolve();\n }\n\n const conditionOrder: Record = {};\n orderedIds.forEach((uuid, index) => {\n const c = group.conditions.find((c) => c.uuid === uuid);\n if (c) {\n c.order = index;\n conditionOrder[uuid] = index;\n }\n });\n const endpoint = `apps/${appUuid}/conditiongroups/${conditionGroupUuid}`;\n return api.put(endpoint, { conditionOrder }).then(() => {\n const appEditor = useAppEditorStore();\n // Ensure \"publish\" button updates to show \"republish\"\n appEditor.updateApp();\n });\n },\n },\n});\n","import {\n Widget,\n WidgetProperties,\n WidgetWithConditions,\n} from \"@/components/widgets/Widget\";\nimport { DEFAULT_CONDITION_ID } from \"@/constants\";\nimport { DataBindingType } from \"@/types/data\";\nimport { ConditionsData } from \"@/types/rendererData\";\nimport omit from \"lodash.omit\";\n\nexport const isConditionalBindingType = (type: DataBindingType): boolean => {\n return [\"Asset\", \"DataSetNode\", \"Scalar\"].includes(type);\n};\n\n/**\n * Takes a `WidgetWithCondtions` and returns a `Widget` which\n * will have the props from the active condition.\n *\n * The props returned here do NOT have their dynamic values resolved.\n *\n * You'll need to merge this with `widgetData` in a separate step.\n *\n * @param widget The stored `WidgetWithConditions`\n * @param conditions This can be either the `ConditionsData` object (from RenderData) or the `Record` object from (conditionGroupStore)\n * @param groupUuid The groupUuid of the current row of the repeater data set (if it has one)\n */\nexport const getActiveWidget = (\n widget: WidgetWithConditions,\n conditions: ConditionsData | Record | undefined,\n groupUuid: string | undefined = undefined\n): Widget => {\n if (!widget) return {} as Widget;\n\n if (widget.conditionalVersions === undefined) {\n return widget as unknown as Widget;\n }\n\n if (conditions === undefined) {\n console.error(\"getActiveWidget called with undefined conditions\");\n return widget as unknown as Widget;\n }\n\n const activeConditionId = getActiveConditionId(\n widget.wid,\n conditions,\n groupUuid\n );\n\n const source = omit(widget, \"conditionalVersions\");\n const props =\n widget.conditionalVersions[activeConditionId] ??\n widget.conditionalVersions[DEFAULT_CONDITION_ID];\n\n return {\n ...source,\n ...props,\n };\n};\n\nexport const getActiveConditionId = (\n widgetId: string,\n conditions: ConditionsData | Record | undefined,\n groupUuid?: string\n): string => {\n if (conditions === undefined) {\n console.error(\"getActiveWidget called with undefined conditions\");\n return DEFAULT_CONDITION_ID;\n }\n\n // If conditions object is provided, it means we are in the renderer\n if (conditions?.widgets !== undefined) {\n const widgets = (conditions as ConditionsData).widgets;\n const conditionInfo = widgets[widgetId];\n\n if (typeof conditionInfo?.conditionUuid === \"string\") {\n return conditionInfo.conditionUuid;\n }\n\n if (typeof conditionInfo?.recordCondRef === \"string\") {\n const recordCondRef = (conditions as ConditionsData)\n .recordCondRefToRecordCondition[conditionInfo.recordCondRef];\n\n return recordCondRef[groupUuid as string] ?? DEFAULT_CONDITION_ID;\n }\n } else if (widgetId in conditions) {\n return (conditions as Record)[widgetId];\n }\n\n return DEFAULT_CONDITION_ID;\n};\n\n/**\n * Converts a `Widget` into a `WidgetWithCondtions`.\n *\n * This modifies the widget by reference. It does NOT return a new object.\n */\nexport const addConditionalVersions = (wg: Widget) => {\n if (\"conditionalVersions\" in wg) {\n return;\n }\n\n const wc = wg as unknown as WidgetWithConditions;\n const props: WidgetProperties = {};\n\n for (const prop in wg) {\n if (\n prop === \"wid\" ||\n prop === \"parentId\" ||\n prop === \"locked\" ||\n prop === \"type\"\n ) {\n continue;\n }\n const val = wg[prop as keyof Widget];\n if (typeof val === \"object\") {\n props[prop] = JSON.parse(JSON.stringify(val));\n } else {\n props[prop] = val;\n }\n delete (wg as any)[prop];\n }\n wc.conditionalVersions = {\n [DEFAULT_CONDITION_ID]: props,\n };\n};\n\n/**\n * Serves similar purpose to getActiveConditionId, from the pinia store.\n * However, this is used in renderer, not in the app editor.\n * Uses info from backend about which condition is active at a given time, for each widget.\n */\nexport const getRenderedConditionId = (\n widgetId: string,\n conditions: ConditionsData | undefined,\n groupUuid?: string\n) => {\n let activeConditionId = DEFAULT_CONDITION_ID;\n\n if (\n !conditions ||\n typeof conditions !== \"object\" ||\n typeof conditions.widgets === \"undefined\"\n ) {\n return activeConditionId;\n }\n\n const conditionInfo = conditions.widgets[widgetId];\n\n if (typeof conditionInfo?.conditionUuid === \"string\") {\n activeConditionId = conditionInfo.conditionUuid;\n }\n\n if (typeof conditionInfo?.recordCondRef === \"string\") {\n const recordCondRef =\n conditions.recordCondRefToRecordCondition[conditionInfo.recordCondRef];\n\n activeConditionId =\n recordCondRef[groupUuid as string] ?? DEFAULT_CONDITION_ID;\n }\n\n return activeConditionId;\n};\n","import { BASE_PARENT_ID } from \"@/constants\";\nimport { WidgetType } from \"@/types\";\n\nexport interface ComponentOptions {\n wid: string;\n parentId: string;\n locked: boolean;\n type: WidgetType;\n name?: string;\n}\n\nexport const DefaultComponentOptions = {\n parentId: BASE_PARENT_ID,\n locked: false,\n};\n","/*!\n * vue-i18n v8.27.2 \n * (c) 2022 kazuya kawaguchi\n * Released under the MIT License.\n */\n/* */\n\n/**\n * constants\n */\n\nvar numberFormatKeys = [\n 'compactDisplay',\n 'currency',\n 'currencyDisplay',\n 'currencySign',\n 'localeMatcher',\n 'notation',\n 'numberingSystem',\n 'signDisplay',\n 'style',\n 'unit',\n 'unitDisplay',\n 'useGrouping',\n 'minimumIntegerDigits',\n 'minimumFractionDigits',\n 'maximumFractionDigits',\n 'minimumSignificantDigits',\n 'maximumSignificantDigits'\n];\n\n/**\n * utilities\n */\n\nfunction warn (msg, err) {\n if (typeof console !== 'undefined') {\n console.warn('[vue-i18n] ' + msg);\n /* istanbul ignore if */\n if (err) {\n console.warn(err.stack);\n }\n }\n}\n\nfunction error (msg, err) {\n if (typeof console !== 'undefined') {\n console.error('[vue-i18n] ' + msg);\n /* istanbul ignore if */\n if (err) {\n console.error(err.stack);\n }\n }\n}\n\nvar isArray = Array.isArray;\n\nfunction isObject (obj) {\n return obj !== null && typeof obj === 'object'\n}\n\nfunction isBoolean (val) {\n return typeof val === 'boolean'\n}\n\nfunction isString (val) {\n return typeof val === 'string'\n}\n\nvar toString = Object.prototype.toString;\nvar OBJECT_STRING = '[object Object]';\nfunction isPlainObject (obj) {\n return toString.call(obj) === OBJECT_STRING\n}\n\nfunction isNull (val) {\n return val === null || val === undefined\n}\n\nfunction isFunction (val) {\n return typeof val === 'function'\n}\n\nfunction parseArgs () {\n var args = [], len = arguments.length;\n while ( len-- ) args[ len ] = arguments[ len ];\n\n var locale = null;\n var params = null;\n if (args.length === 1) {\n if (isObject(args[0]) || isArray(args[0])) {\n params = args[0];\n } else if (typeof args[0] === 'string') {\n locale = args[0];\n }\n } else if (args.length === 2) {\n if (typeof args[0] === 'string') {\n locale = args[0];\n }\n /* istanbul ignore if */\n if (isObject(args[1]) || isArray(args[1])) {\n params = args[1];\n }\n }\n\n return { locale: locale, params: params }\n}\n\nfunction looseClone (obj) {\n return JSON.parse(JSON.stringify(obj))\n}\n\nfunction remove (arr, item) {\n if (arr.delete(item)) {\n return arr\n }\n}\n\nfunction arrayFrom (arr) {\n var ret = [];\n arr.forEach(function (a) { return ret.push(a); });\n return ret\n}\n\nfunction includes (arr, item) {\n return !!~arr.indexOf(item)\n}\n\nvar hasOwnProperty = Object.prototype.hasOwnProperty;\nfunction hasOwn (obj, key) {\n return hasOwnProperty.call(obj, key)\n}\n\nfunction merge (target) {\n var arguments$1 = arguments;\n\n var output = Object(target);\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments$1[i];\n if (source !== undefined && source !== null) {\n var key = (void 0);\n for (key in source) {\n if (hasOwn(source, key)) {\n if (isObject(source[key])) {\n output[key] = merge(output[key], source[key]);\n } else {\n output[key] = source[key];\n }\n }\n }\n }\n }\n return output\n}\n\nfunction looseEqual (a, b) {\n if (a === b) { return true }\n var isObjectA = isObject(a);\n var isObjectB = isObject(b);\n if (isObjectA && isObjectB) {\n try {\n var isArrayA = isArray(a);\n var isArrayB = isArray(b);\n if (isArrayA && isArrayB) {\n return a.length === b.length && a.every(function (e, i) {\n return looseEqual(e, b[i])\n })\n } else if (!isArrayA && !isArrayB) {\n var keysA = Object.keys(a);\n var keysB = Object.keys(b);\n return keysA.length === keysB.length && keysA.every(function (key) {\n return looseEqual(a[key], b[key])\n })\n } else {\n /* istanbul ignore next */\n return false\n }\n } catch (e) {\n /* istanbul ignore next */\n return false\n }\n } else if (!isObjectA && !isObjectB) {\n return String(a) === String(b)\n } else {\n return false\n }\n}\n\n/**\n * Sanitizes html special characters from input strings. For mitigating risk of XSS attacks.\n * @param rawText The raw input from the user that should be escaped.\n */\nfunction escapeHtml(rawText) {\n return rawText\n .replace(//g, '>')\n .replace(/\"/g, '"')\n .replace(/'/g, ''')\n}\n\n/**\n * Escapes html tags and special symbols from all provided params which were returned from parseArgs().params.\n * This method performs an in-place operation on the params object.\n *\n * @param {any} params Parameters as provided from `parseArgs().params`.\n * May be either an array of strings or a string->any map.\n *\n * @returns The manipulated `params` object.\n */\nfunction escapeParams(params) {\n if(params != null) {\n Object.keys(params).forEach(function (key) {\n if(typeof(params[key]) == 'string') {\n params[key] = escapeHtml(params[key]);\n }\n });\n }\n return params\n}\n\n/* */\n\nfunction extend (Vue) {\n if (!Vue.prototype.hasOwnProperty('$i18n')) {\n // $FlowFixMe\n Object.defineProperty(Vue.prototype, '$i18n', {\n get: function get () { return this._i18n }\n });\n }\n\n Vue.prototype.$t = function (key) {\n var values = [], len = arguments.length - 1;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 1 ];\n\n var i18n = this.$i18n;\n return i18n._t.apply(i18n, [ key, i18n.locale, i18n._getMessages(), this ].concat( values ))\n };\n\n Vue.prototype.$tc = function (key, choice) {\n var values = [], len = arguments.length - 2;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 2 ];\n\n var i18n = this.$i18n;\n return i18n._tc.apply(i18n, [ key, i18n.locale, i18n._getMessages(), this, choice ].concat( values ))\n };\n\n Vue.prototype.$te = function (key, locale) {\n var i18n = this.$i18n;\n return i18n._te(key, i18n.locale, i18n._getMessages(), locale)\n };\n\n Vue.prototype.$d = function (value) {\n var ref;\n\n var args = [], len = arguments.length - 1;\n while ( len-- > 0 ) args[ len ] = arguments[ len + 1 ];\n return (ref = this.$i18n).d.apply(ref, [ value ].concat( args ))\n };\n\n Vue.prototype.$n = function (value) {\n var ref;\n\n var args = [], len = arguments.length - 1;\n while ( len-- > 0 ) args[ len ] = arguments[ len + 1 ];\n return (ref = this.$i18n).n.apply(ref, [ value ].concat( args ))\n };\n}\n\n/* */\n\n/**\n * Mixin\n * \n * If `bridge` mode, empty mixin is returned,\n * else regulary mixin implementation is returned.\n */\nfunction defineMixin (bridge) {\n if ( bridge === void 0 ) bridge = false;\n\n function mounted () {\n if (this !== this.$root && this.$options.__INTLIFY_META__ && this.$el) {\n this.$el.setAttribute('data-intlify', this.$options.__INTLIFY_META__);\n }\n }\n\n return bridge\n ? { mounted: mounted } // delegate `vue-i18n-bridge` mixin implementation\n : { // regulary \n beforeCreate: function beforeCreate () {\n var options = this.$options;\n options.i18n = options.i18n || ((options.__i18nBridge || options.__i18n) ? {} : null);\n\n if (options.i18n) {\n if (options.i18n instanceof VueI18n) {\n // init locale messages via custom blocks\n if ((options.__i18nBridge || options.__i18n)) {\n try {\n var localeMessages = options.i18n && options.i18n.messages ? options.i18n.messages : {};\n var _i18n = options.__i18nBridge || options.__i18n;\n _i18n.forEach(function (resource) {\n localeMessages = merge(localeMessages, JSON.parse(resource));\n });\n Object.keys(localeMessages).forEach(function (locale) {\n options.i18n.mergeLocaleMessage(locale, localeMessages[locale]);\n });\n } catch (e) {\n if (process.env.NODE_ENV !== 'production') {\n error(\"Cannot parse locale messages via custom blocks.\", e);\n }\n }\n }\n this._i18n = options.i18n;\n this._i18nWatcher = this._i18n.watchI18nData();\n } else if (isPlainObject(options.i18n)) {\n var rootI18n = this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n\n ? this.$root.$i18n\n : null;\n // component local i18n\n if (rootI18n) {\n options.i18n.root = this.$root;\n options.i18n.formatter = rootI18n.formatter;\n options.i18n.fallbackLocale = rootI18n.fallbackLocale;\n options.i18n.formatFallbackMessages = rootI18n.formatFallbackMessages;\n options.i18n.silentTranslationWarn = rootI18n.silentTranslationWarn;\n options.i18n.silentFallbackWarn = rootI18n.silentFallbackWarn;\n options.i18n.pluralizationRules = rootI18n.pluralizationRules;\n options.i18n.preserveDirectiveContent = rootI18n.preserveDirectiveContent;\n }\n\n // init locale messages via custom blocks\n if ((options.__i18nBridge || options.__i18n)) {\n try {\n var localeMessages$1 = options.i18n && options.i18n.messages ? options.i18n.messages : {};\n var _i18n$1 = options.__i18nBridge || options.__i18n;\n _i18n$1.forEach(function (resource) {\n localeMessages$1 = merge(localeMessages$1, JSON.parse(resource));\n });\n options.i18n.messages = localeMessages$1;\n } catch (e) {\n if (process.env.NODE_ENV !== 'production') {\n warn(\"Cannot parse locale messages via custom blocks.\", e);\n }\n }\n }\n\n var ref = options.i18n;\n var sharedMessages = ref.sharedMessages;\n if (sharedMessages && isPlainObject(sharedMessages)) {\n options.i18n.messages = merge(options.i18n.messages, sharedMessages);\n }\n\n this._i18n = new VueI18n(options.i18n);\n this._i18nWatcher = this._i18n.watchI18nData();\n\n if (options.i18n.sync === undefined || !!options.i18n.sync) {\n this._localeWatcher = this.$i18n.watchLocale();\n }\n\n if (rootI18n) {\n rootI18n.onComponentInstanceCreated(this._i18n);\n }\n } else {\n if (process.env.NODE_ENV !== 'production') {\n warn(\"Cannot be interpreted 'i18n' option.\");\n }\n }\n } else if (this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n) {\n // root i18n\n this._i18n = this.$root.$i18n;\n } else if (options.parent && options.parent.$i18n && options.parent.$i18n instanceof VueI18n) {\n // parent i18n\n this._i18n = options.parent.$i18n;\n }\n },\n\n beforeMount: function beforeMount () {\n var options = this.$options;\n options.i18n = options.i18n || ((options.__i18nBridge || options.__i18n) ? {} : null);\n\n if (options.i18n) {\n if (options.i18n instanceof VueI18n) {\n // init locale messages via custom blocks\n this._i18n.subscribeDataChanging(this);\n this._subscribing = true;\n } else if (isPlainObject(options.i18n)) {\n this._i18n.subscribeDataChanging(this);\n this._subscribing = true;\n } else {\n if (process.env.NODE_ENV !== 'production') {\n warn(\"Cannot be interpreted 'i18n' option.\");\n }\n }\n } else if (this.$root && this.$root.$i18n && this.$root.$i18n instanceof VueI18n) {\n this._i18n.subscribeDataChanging(this);\n this._subscribing = true;\n } else if (options.parent && options.parent.$i18n && options.parent.$i18n instanceof VueI18n) {\n this._i18n.subscribeDataChanging(this);\n this._subscribing = true;\n }\n },\n\n mounted: mounted,\n\n beforeDestroy: function beforeDestroy () {\n if (!this._i18n) { return }\n\n var self = this;\n this.$nextTick(function () {\n if (self._subscribing) {\n self._i18n.unsubscribeDataChanging(self);\n delete self._subscribing;\n }\n\n if (self._i18nWatcher) {\n self._i18nWatcher();\n self._i18n.destroyVM();\n delete self._i18nWatcher;\n }\n\n if (self._localeWatcher) {\n self._localeWatcher();\n delete self._localeWatcher;\n }\n });\n }\n }\n}\n\n/* */\n\nvar interpolationComponent = {\n name: 'i18n',\n functional: true,\n props: {\n tag: {\n type: [String, Boolean, Object],\n default: 'span'\n },\n path: {\n type: String,\n required: true\n },\n locale: {\n type: String\n },\n places: {\n type: [Array, Object]\n }\n },\n render: function render (h, ref) {\n var data = ref.data;\n var parent = ref.parent;\n var props = ref.props;\n var slots = ref.slots;\n\n var $i18n = parent.$i18n;\n if (!$i18n) {\n if (process.env.NODE_ENV !== 'production') {\n warn('Cannot find VueI18n instance!');\n }\n return\n }\n\n var path = props.path;\n var locale = props.locale;\n var places = props.places;\n var params = slots();\n var children = $i18n.i(\n path,\n locale,\n onlyHasDefaultPlace(params) || places\n ? useLegacyPlaces(params.default, places)\n : params\n );\n\n var tag = (!!props.tag && props.tag !== true) || props.tag === false ? props.tag : 'span';\n return tag ? h(tag, data, children) : children\n }\n};\n\nfunction onlyHasDefaultPlace (params) {\n var prop;\n for (prop in params) {\n if (prop !== 'default') { return false }\n }\n return Boolean(prop)\n}\n\nfunction useLegacyPlaces (children, places) {\n var params = places ? createParamsFromPlaces(places) : {};\n\n if (!children) { return params }\n\n // Filter empty text nodes\n children = children.filter(function (child) {\n return child.tag || child.text.trim() !== ''\n });\n\n var everyPlace = children.every(vnodeHasPlaceAttribute);\n if (process.env.NODE_ENV !== 'production' && everyPlace) {\n warn('`place` attribute is deprecated in next major version. Please switch to Vue slots.');\n }\n\n return children.reduce(\n everyPlace ? assignChildPlace : assignChildIndex,\n params\n )\n}\n\nfunction createParamsFromPlaces (places) {\n if (process.env.NODE_ENV !== 'production') {\n warn('`places` prop is deprecated in next major version. Please switch to Vue slots.');\n }\n\n return Array.isArray(places)\n ? places.reduce(assignChildIndex, {})\n : Object.assign({}, places)\n}\n\nfunction assignChildPlace (params, child) {\n if (child.data && child.data.attrs && child.data.attrs.place) {\n params[child.data.attrs.place] = child;\n }\n return params\n}\n\nfunction assignChildIndex (params, child, index) {\n params[index] = child;\n return params\n}\n\nfunction vnodeHasPlaceAttribute (vnode) {\n return Boolean(vnode.data && vnode.data.attrs && vnode.data.attrs.place)\n}\n\n/* */\n\nvar numberComponent = {\n name: 'i18n-n',\n functional: true,\n props: {\n tag: {\n type: [String, Boolean, Object],\n default: 'span'\n },\n value: {\n type: Number,\n required: true\n },\n format: {\n type: [String, Object]\n },\n locale: {\n type: String\n }\n },\n render: function render (h, ref) {\n var props = ref.props;\n var parent = ref.parent;\n var data = ref.data;\n\n var i18n = parent.$i18n;\n\n if (!i18n) {\n if (process.env.NODE_ENV !== 'production') {\n warn('Cannot find VueI18n instance!');\n }\n return null\n }\n\n var key = null;\n var options = null;\n\n if (isString(props.format)) {\n key = props.format;\n } else if (isObject(props.format)) {\n if (props.format.key) {\n key = props.format.key;\n }\n\n // Filter out number format options only\n options = Object.keys(props.format).reduce(function (acc, prop) {\n var obj;\n\n if (includes(numberFormatKeys, prop)) {\n return Object.assign({}, acc, ( obj = {}, obj[prop] = props.format[prop], obj ))\n }\n return acc\n }, null);\n }\n\n var locale = props.locale || i18n.locale;\n var parts = i18n._ntp(props.value, locale, key, options);\n\n var values = parts.map(function (part, index) {\n var obj;\n\n var slot = data.scopedSlots && data.scopedSlots[part.type];\n return slot ? slot(( obj = {}, obj[part.type] = part.value, obj.index = index, obj.parts = parts, obj )) : part.value\n });\n\n var tag = (!!props.tag && props.tag !== true) || props.tag === false ? props.tag : 'span';\n return tag\n ? h(tag, {\n attrs: data.attrs,\n 'class': data['class'],\n staticClass: data.staticClass\n }, values)\n : values\n }\n};\n\n/* */\n\nfunction bind (el, binding, vnode) {\n if (!assert(el, vnode)) { return }\n\n t(el, binding, vnode);\n}\n\nfunction update (el, binding, vnode, oldVNode) {\n if (!assert(el, vnode)) { return }\n\n var i18n = vnode.context.$i18n;\n if (localeEqual(el, vnode) &&\n (looseEqual(binding.value, binding.oldValue) &&\n looseEqual(el._localeMessage, i18n.getLocaleMessage(i18n.locale)))) { return }\n\n t(el, binding, vnode);\n}\n\nfunction unbind (el, binding, vnode, oldVNode) {\n var vm = vnode.context;\n if (!vm) {\n warn('Vue instance does not exists in VNode context');\n return\n }\n\n var i18n = vnode.context.$i18n || {};\n if (!binding.modifiers.preserve && !i18n.preserveDirectiveContent) {\n el.textContent = '';\n }\n el._vt = undefined;\n delete el['_vt'];\n el._locale = undefined;\n delete el['_locale'];\n el._localeMessage = undefined;\n delete el['_localeMessage'];\n}\n\nfunction assert (el, vnode) {\n var vm = vnode.context;\n if (!vm) {\n warn('Vue instance does not exists in VNode context');\n return false\n }\n\n if (!vm.$i18n) {\n warn('VueI18n instance does not exists in Vue instance');\n return false\n }\n\n return true\n}\n\nfunction localeEqual (el, vnode) {\n var vm = vnode.context;\n return el._locale === vm.$i18n.locale\n}\n\nfunction t (el, binding, vnode) {\n var ref$1, ref$2;\n\n var value = binding.value;\n\n var ref = parseValue(value);\n var path = ref.path;\n var locale = ref.locale;\n var args = ref.args;\n var choice = ref.choice;\n if (!path && !locale && !args) {\n warn('value type not supported');\n return\n }\n\n if (!path) {\n warn('`path` is required in v-t directive');\n return\n }\n\n var vm = vnode.context;\n if (choice != null) {\n el._vt = el.textContent = (ref$1 = vm.$i18n).tc.apply(ref$1, [ path, choice ].concat( makeParams(locale, args) ));\n } else {\n el._vt = el.textContent = (ref$2 = vm.$i18n).t.apply(ref$2, [ path ].concat( makeParams(locale, args) ));\n }\n el._locale = vm.$i18n.locale;\n el._localeMessage = vm.$i18n.getLocaleMessage(vm.$i18n.locale);\n}\n\nfunction parseValue (value) {\n var path;\n var locale;\n var args;\n var choice;\n\n if (isString(value)) {\n path = value;\n } else if (isPlainObject(value)) {\n path = value.path;\n locale = value.locale;\n args = value.args;\n choice = value.choice;\n }\n\n return { path: path, locale: locale, args: args, choice: choice }\n}\n\nfunction makeParams (locale, args) {\n var params = [];\n\n locale && params.push(locale);\n if (args && (Array.isArray(args) || isPlainObject(args))) {\n params.push(args);\n }\n\n return params\n}\n\nvar Vue;\n\nfunction install (_Vue, options) {\n if ( options === void 0 ) options = { bridge: false };\n\n /* istanbul ignore if */\n if (process.env.NODE_ENV !== 'production' && install.installed && _Vue === Vue) {\n warn('already installed.');\n return\n }\n install.installed = true;\n\n Vue = _Vue;\n\n var version = (Vue.version && Number(Vue.version.split('.')[0])) || -1;\n /* istanbul ignore if */\n if (process.env.NODE_ENV !== 'production' && version < 2) {\n warn((\"vue-i18n (\" + (install.version) + \") need to use Vue 2.0 or later (Vue: \" + (Vue.version) + \").\"));\n return\n }\n\n extend(Vue);\n Vue.mixin(defineMixin(options.bridge));\n Vue.directive('t', { bind: bind, update: update, unbind: unbind });\n Vue.component(interpolationComponent.name, interpolationComponent);\n Vue.component(numberComponent.name, numberComponent);\n\n // use simple mergeStrategies to prevent i18n instance lose '__proto__'\n var strats = Vue.config.optionMergeStrategies;\n strats.i18n = function (parentVal, childVal) {\n return childVal === undefined\n ? parentVal\n : childVal\n };\n}\n\n/* */\n\nvar BaseFormatter = function BaseFormatter () {\n this._caches = Object.create(null);\n};\n\nBaseFormatter.prototype.interpolate = function interpolate (message, values) {\n if (!values) {\n return [message]\n }\n var tokens = this._caches[message];\n if (!tokens) {\n tokens = parse(message);\n this._caches[message] = tokens;\n }\n return compile(tokens, values)\n};\n\n\n\nvar RE_TOKEN_LIST_VALUE = /^(?:\\d)+/;\nvar RE_TOKEN_NAMED_VALUE = /^(?:\\w)+/;\n\nfunction parse (format) {\n var tokens = [];\n var position = 0;\n\n var text = '';\n while (position < format.length) {\n var char = format[position++];\n if (char === '{') {\n if (text) {\n tokens.push({ type: 'text', value: text });\n }\n\n text = '';\n var sub = '';\n char = format[position++];\n while (char !== undefined && char !== '}') {\n sub += char;\n char = format[position++];\n }\n var isClosed = char === '}';\n\n var type = RE_TOKEN_LIST_VALUE.test(sub)\n ? 'list'\n : isClosed && RE_TOKEN_NAMED_VALUE.test(sub)\n ? 'named'\n : 'unknown';\n tokens.push({ value: sub, type: type });\n } else if (char === '%') {\n // when found rails i18n syntax, skip text capture\n if (format[(position)] !== '{') {\n text += char;\n }\n } else {\n text += char;\n }\n }\n\n text && tokens.push({ type: 'text', value: text });\n\n return tokens\n}\n\nfunction compile (tokens, values) {\n var compiled = [];\n var index = 0;\n\n var mode = Array.isArray(values)\n ? 'list'\n : isObject(values)\n ? 'named'\n : 'unknown';\n if (mode === 'unknown') { return compiled }\n\n while (index < tokens.length) {\n var token = tokens[index];\n switch (token.type) {\n case 'text':\n compiled.push(token.value);\n break\n case 'list':\n compiled.push(values[parseInt(token.value, 10)]);\n break\n case 'named':\n if (mode === 'named') {\n compiled.push((values)[token.value]);\n } else {\n if (process.env.NODE_ENV !== 'production') {\n warn((\"Type of token '\" + (token.type) + \"' and format of value '\" + mode + \"' don't match!\"));\n }\n }\n break\n case 'unknown':\n if (process.env.NODE_ENV !== 'production') {\n warn(\"Detect 'unknown' type of token!\");\n }\n break\n }\n index++;\n }\n\n return compiled\n}\n\n/* */\n\n/**\n * Path parser\n * - Inspired:\n * Vue.js Path parser\n */\n\n// actions\nvar APPEND = 0;\nvar PUSH = 1;\nvar INC_SUB_PATH_DEPTH = 2;\nvar PUSH_SUB_PATH = 3;\n\n// states\nvar BEFORE_PATH = 0;\nvar IN_PATH = 1;\nvar BEFORE_IDENT = 2;\nvar IN_IDENT = 3;\nvar IN_SUB_PATH = 4;\nvar IN_SINGLE_QUOTE = 5;\nvar IN_DOUBLE_QUOTE = 6;\nvar AFTER_PATH = 7;\nvar ERROR = 8;\n\nvar pathStateMachine = [];\n\npathStateMachine[BEFORE_PATH] = {\n 'ws': [BEFORE_PATH],\n 'ident': [IN_IDENT, APPEND],\n '[': [IN_SUB_PATH],\n 'eof': [AFTER_PATH]\n};\n\npathStateMachine[IN_PATH] = {\n 'ws': [IN_PATH],\n '.': [BEFORE_IDENT],\n '[': [IN_SUB_PATH],\n 'eof': [AFTER_PATH]\n};\n\npathStateMachine[BEFORE_IDENT] = {\n 'ws': [BEFORE_IDENT],\n 'ident': [IN_IDENT, APPEND],\n '0': [IN_IDENT, APPEND],\n 'number': [IN_IDENT, APPEND]\n};\n\npathStateMachine[IN_IDENT] = {\n 'ident': [IN_IDENT, APPEND],\n '0': [IN_IDENT, APPEND],\n 'number': [IN_IDENT, APPEND],\n 'ws': [IN_PATH, PUSH],\n '.': [BEFORE_IDENT, PUSH],\n '[': [IN_SUB_PATH, PUSH],\n 'eof': [AFTER_PATH, PUSH]\n};\n\npathStateMachine[IN_SUB_PATH] = {\n \"'\": [IN_SINGLE_QUOTE, APPEND],\n '\"': [IN_DOUBLE_QUOTE, APPEND],\n '[': [IN_SUB_PATH, INC_SUB_PATH_DEPTH],\n ']': [IN_PATH, PUSH_SUB_PATH],\n 'eof': ERROR,\n 'else': [IN_SUB_PATH, APPEND]\n};\n\npathStateMachine[IN_SINGLE_QUOTE] = {\n \"'\": [IN_SUB_PATH, APPEND],\n 'eof': ERROR,\n 'else': [IN_SINGLE_QUOTE, APPEND]\n};\n\npathStateMachine[IN_DOUBLE_QUOTE] = {\n '\"': [IN_SUB_PATH, APPEND],\n 'eof': ERROR,\n 'else': [IN_DOUBLE_QUOTE, APPEND]\n};\n\n/**\n * Check if an expression is a literal value.\n */\n\nvar literalValueRE = /^\\s?(?:true|false|-?[\\d.]+|'[^']*'|\"[^\"]*\")\\s?$/;\nfunction isLiteral (exp) {\n return literalValueRE.test(exp)\n}\n\n/**\n * Strip quotes from a string\n */\n\nfunction stripQuotes (str) {\n var a = str.charCodeAt(0);\n var b = str.charCodeAt(str.length - 1);\n return a === b && (a === 0x22 || a === 0x27)\n ? str.slice(1, -1)\n : str\n}\n\n/**\n * Determine the type of a character in a keypath.\n */\n\nfunction getPathCharType (ch) {\n if (ch === undefined || ch === null) { return 'eof' }\n\n var code = ch.charCodeAt(0);\n\n switch (code) {\n case 0x5B: // [\n case 0x5D: // ]\n case 0x2E: // .\n case 0x22: // \"\n case 0x27: // '\n return ch\n\n case 0x5F: // _\n case 0x24: // $\n case 0x2D: // -\n return 'ident'\n\n case 0x09: // Tab\n case 0x0A: // Newline\n case 0x0D: // Return\n case 0xA0: // No-break space\n case 0xFEFF: // Byte Order Mark\n case 0x2028: // Line Separator\n case 0x2029: // Paragraph Separator\n return 'ws'\n }\n\n return 'ident'\n}\n\n/**\n * Format a subPath, return its plain form if it is\n * a literal string or number. Otherwise prepend the\n * dynamic indicator (*).\n */\n\nfunction formatSubPath (path) {\n var trimmed = path.trim();\n // invalid leading 0\n if (path.charAt(0) === '0' && isNaN(path)) { return false }\n\n return isLiteral(trimmed) ? stripQuotes(trimmed) : '*' + trimmed\n}\n\n/**\n * Parse a string path into an array of segments\n */\n\nfunction parse$1 (path) {\n var keys = [];\n var index = -1;\n var mode = BEFORE_PATH;\n var subPathDepth = 0;\n var c;\n var key;\n var newChar;\n var type;\n var transition;\n var action;\n var typeMap;\n var actions = [];\n\n actions[PUSH] = function () {\n if (key !== undefined) {\n keys.push(key);\n key = undefined;\n }\n };\n\n actions[APPEND] = function () {\n if (key === undefined) {\n key = newChar;\n } else {\n key += newChar;\n }\n };\n\n actions[INC_SUB_PATH_DEPTH] = function () {\n actions[APPEND]();\n subPathDepth++;\n };\n\n actions[PUSH_SUB_PATH] = function () {\n if (subPathDepth > 0) {\n subPathDepth--;\n mode = IN_SUB_PATH;\n actions[APPEND]();\n } else {\n subPathDepth = 0;\n if (key === undefined) { return false }\n key = formatSubPath(key);\n if (key === false) {\n return false\n } else {\n actions[PUSH]();\n }\n }\n };\n\n function maybeUnescapeQuote () {\n var nextChar = path[index + 1];\n if ((mode === IN_SINGLE_QUOTE && nextChar === \"'\") ||\n (mode === IN_DOUBLE_QUOTE && nextChar === '\"')) {\n index++;\n newChar = '\\\\' + nextChar;\n actions[APPEND]();\n return true\n }\n }\n\n while (mode !== null) {\n index++;\n c = path[index];\n\n if (c === '\\\\' && maybeUnescapeQuote()) {\n continue\n }\n\n type = getPathCharType(c);\n typeMap = pathStateMachine[mode];\n transition = typeMap[type] || typeMap['else'] || ERROR;\n\n if (transition === ERROR) {\n return // parse error\n }\n\n mode = transition[0];\n action = actions[transition[1]];\n if (action) {\n newChar = transition[2];\n newChar = newChar === undefined\n ? c\n : newChar;\n if (action() === false) {\n return\n }\n }\n\n if (mode === AFTER_PATH) {\n return keys\n }\n }\n}\n\n\n\n\n\nvar I18nPath = function I18nPath () {\n this._cache = Object.create(null);\n};\n\n/**\n * External parse that check for a cache hit first\n */\nI18nPath.prototype.parsePath = function parsePath (path) {\n var hit = this._cache[path];\n if (!hit) {\n hit = parse$1(path);\n if (hit) {\n this._cache[path] = hit;\n }\n }\n return hit || []\n};\n\n/**\n * Get path value from path string\n */\nI18nPath.prototype.getPathValue = function getPathValue (obj, path) {\n if (!isObject(obj)) { return null }\n\n var paths = this.parsePath(path);\n if (paths.length === 0) {\n return null\n } else {\n var length = paths.length;\n var last = obj;\n var i = 0;\n while (i < length) {\n var value = last[paths[i]];\n if (value === undefined || value === null) {\n return null\n }\n last = value;\n i++;\n }\n\n return last\n }\n};\n\n/* */\n\n\n\nvar htmlTagMatcher = /<\\/?[\\w\\s=\"/.':;#-\\/]+>/;\nvar linkKeyMatcher = /(?:@(?:\\.[a-zA-Z]+)?:(?:[\\w\\-_|./]+|\\([\\w\\-_:|./]+\\)))/g;\nvar linkKeyPrefixMatcher = /^@(?:\\.([a-zA-Z]+))?:/;\nvar bracketsMatcher = /[()]/g;\nvar defaultModifiers = {\n 'upper': function (str) { return str.toLocaleUpperCase(); },\n 'lower': function (str) { return str.toLocaleLowerCase(); },\n 'capitalize': function (str) { return (\"\" + (str.charAt(0).toLocaleUpperCase()) + (str.substr(1))); }\n};\n\nvar defaultFormatter = new BaseFormatter();\n\nvar VueI18n = function VueI18n (options) {\n var this$1 = this;\n if ( options === void 0 ) options = {};\n\n // Auto install if it is not done yet and `window` has `Vue`.\n // To allow users to avoid auto-installation in some cases,\n // this code should be placed here. See #290\n /* istanbul ignore if */\n if (!Vue && typeof window !== 'undefined' && window.Vue) {\n install(window.Vue);\n }\n\n var locale = options.locale || 'en-US';\n var fallbackLocale = options.fallbackLocale === false\n ? false\n : options.fallbackLocale || 'en-US';\n var messages = options.messages || {};\n var dateTimeFormats = options.dateTimeFormats || options.datetimeFormats || {};\n var numberFormats = options.numberFormats || {};\n\n this._vm = null;\n this._formatter = options.formatter || defaultFormatter;\n this._modifiers = options.modifiers || {};\n this._missing = options.missing || null;\n this._root = options.root || null;\n this._sync = options.sync === undefined ? true : !!options.sync;\n this._fallbackRoot = options.fallbackRoot === undefined\n ? true\n : !!options.fallbackRoot;\n this._fallbackRootWithEmptyString = options.fallbackRootWithEmptyString === undefined\n ? true\n : !!options.fallbackRootWithEmptyString;\n this._formatFallbackMessages = options.formatFallbackMessages === undefined\n ? false\n : !!options.formatFallbackMessages;\n this._silentTranslationWarn = options.silentTranslationWarn === undefined\n ? false\n : options.silentTranslationWarn;\n this._silentFallbackWarn = options.silentFallbackWarn === undefined\n ? false\n : !!options.silentFallbackWarn;\n this._dateTimeFormatters = {};\n this._numberFormatters = {};\n this._path = new I18nPath();\n this._dataListeners = new Set();\n this._componentInstanceCreatedListener = options.componentInstanceCreatedListener || null;\n this._preserveDirectiveContent = options.preserveDirectiveContent === undefined\n ? false\n : !!options.preserveDirectiveContent;\n this.pluralizationRules = options.pluralizationRules || {};\n this._warnHtmlInMessage = options.warnHtmlInMessage || 'off';\n this._postTranslation = options.postTranslation || null;\n this._escapeParameterHtml = options.escapeParameterHtml || false;\n\n if ('__VUE_I18N_BRIDGE__' in options) {\n this.__VUE_I18N_BRIDGE__ = options.__VUE_I18N_BRIDGE__;\n }\n\n /**\n * @param choice {number} a choice index given by the input to $tc: `$tc('path.to.rule', choiceIndex)`\n * @param choicesLength {number} an overall amount of available choices\n * @returns a final choice index\n */\n this.getChoiceIndex = function (choice, choicesLength) {\n var thisPrototype = Object.getPrototypeOf(this$1);\n if (thisPrototype && thisPrototype.getChoiceIndex) {\n var prototypeGetChoiceIndex = (thisPrototype.getChoiceIndex);\n return (prototypeGetChoiceIndex).call(this$1, choice, choicesLength)\n }\n\n // Default (old) getChoiceIndex implementation - english-compatible\n var defaultImpl = function (_choice, _choicesLength) {\n _choice = Math.abs(_choice);\n\n if (_choicesLength === 2) {\n return _choice\n ? _choice > 1\n ? 1\n : 0\n : 1\n }\n\n return _choice ? Math.min(_choice, 2) : 0\n };\n\n if (this$1.locale in this$1.pluralizationRules) {\n return this$1.pluralizationRules[this$1.locale].apply(this$1, [choice, choicesLength])\n } else {\n return defaultImpl(choice, choicesLength)\n }\n };\n\n\n this._exist = function (message, key) {\n if (!message || !key) { return false }\n if (!isNull(this$1._path.getPathValue(message, key))) { return true }\n // fallback for flat key\n if (message[key]) { return true }\n return false\n };\n\n if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {\n Object.keys(messages).forEach(function (locale) {\n this$1._checkLocaleMessage(locale, this$1._warnHtmlInMessage, messages[locale]);\n });\n }\n\n this._initVM({\n locale: locale,\n fallbackLocale: fallbackLocale,\n messages: messages,\n dateTimeFormats: dateTimeFormats,\n numberFormats: numberFormats\n });\n};\n\nvar prototypeAccessors = { vm: { configurable: true },messages: { configurable: true },dateTimeFormats: { configurable: true },numberFormats: { configurable: true },availableLocales: { configurable: true },locale: { configurable: true },fallbackLocale: { configurable: true },formatFallbackMessages: { configurable: true },missing: { configurable: true },formatter: { configurable: true },silentTranslationWarn: { configurable: true },silentFallbackWarn: { configurable: true },preserveDirectiveContent: { configurable: true },warnHtmlInMessage: { configurable: true },postTranslation: { configurable: true },sync: { configurable: true } };\n\nVueI18n.prototype._checkLocaleMessage = function _checkLocaleMessage (locale, level, message) {\n var paths = [];\n\n var fn = function (level, locale, message, paths) {\n if (isPlainObject(message)) {\n Object.keys(message).forEach(function (key) {\n var val = message[key];\n if (isPlainObject(val)) {\n paths.push(key);\n paths.push('.');\n fn(level, locale, val, paths);\n paths.pop();\n paths.pop();\n } else {\n paths.push(key);\n fn(level, locale, val, paths);\n paths.pop();\n }\n });\n } else if (isArray(message)) {\n message.forEach(function (item, index) {\n if (isPlainObject(item)) {\n paths.push((\"[\" + index + \"]\"));\n paths.push('.');\n fn(level, locale, item, paths);\n paths.pop();\n paths.pop();\n } else {\n paths.push((\"[\" + index + \"]\"));\n fn(level, locale, item, paths);\n paths.pop();\n }\n });\n } else if (isString(message)) {\n var ret = htmlTagMatcher.test(message);\n if (ret) {\n var msg = \"Detected HTML in message '\" + message + \"' of keypath '\" + (paths.join('')) + \"' at '\" + locale + \"'. Consider component interpolation with '' to avoid XSS. See https://bit.ly/2ZqJzkp\";\n if (level === 'warn') {\n warn(msg);\n } else if (level === 'error') {\n error(msg);\n }\n }\n }\n };\n\n fn(level, locale, message, paths);\n};\n\nVueI18n.prototype._initVM = function _initVM (data) {\n var silent = Vue.config.silent;\n Vue.config.silent = true;\n this._vm = new Vue({ data: data, __VUE18N__INSTANCE__: true });\n Vue.config.silent = silent;\n};\n\nVueI18n.prototype.destroyVM = function destroyVM () {\n this._vm.$destroy();\n};\n\nVueI18n.prototype.subscribeDataChanging = function subscribeDataChanging (vm) {\n this._dataListeners.add(vm);\n};\n\nVueI18n.prototype.unsubscribeDataChanging = function unsubscribeDataChanging (vm) {\n remove(this._dataListeners, vm);\n};\n\nVueI18n.prototype.watchI18nData = function watchI18nData () {\n var this$1 = this;\n return this._vm.$watch('$data', function () {\n var listeners = arrayFrom(this$1._dataListeners);\n var i = listeners.length;\n while(i--) {\n Vue.nextTick(function () {\n listeners[i] && listeners[i].$forceUpdate();\n });\n }\n }, { deep: true })\n};\n\nVueI18n.prototype.watchLocale = function watchLocale (composer) {\n if (!composer) {\n /* istanbul ignore if */\n if (!this._sync || !this._root) { return null }\n var target = this._vm;\n return this._root.$i18n.vm.$watch('locale', function (val) {\n target.$set(target, 'locale', val);\n target.$forceUpdate();\n }, { immediate: true })\n } else {\n // deal with vue-i18n-bridge\n if (!this.__VUE_I18N_BRIDGE__) { return null }\n var self = this;\n var target$1 = this._vm;\n return this.vm.$watch('locale', function (val) {\n target$1.$set(target$1, 'locale', val);\n if (self.__VUE_I18N_BRIDGE__ && composer) {\n composer.locale.value = val;\n }\n target$1.$forceUpdate();\n }, { immediate: true })\n }\n};\n\nVueI18n.prototype.onComponentInstanceCreated = function onComponentInstanceCreated (newI18n) {\n if (this._componentInstanceCreatedListener) {\n this._componentInstanceCreatedListener(newI18n, this);\n }\n};\n\nprototypeAccessors.vm.get = function () { return this._vm };\n\nprototypeAccessors.messages.get = function () { return looseClone(this._getMessages()) };\nprototypeAccessors.dateTimeFormats.get = function () { return looseClone(this._getDateTimeFormats()) };\nprototypeAccessors.numberFormats.get = function () { return looseClone(this._getNumberFormats()) };\nprototypeAccessors.availableLocales.get = function () { return Object.keys(this.messages).sort() };\n\nprototypeAccessors.locale.get = function () { return this._vm.locale };\nprototypeAccessors.locale.set = function (locale) {\n this._vm.$set(this._vm, 'locale', locale);\n};\n\nprototypeAccessors.fallbackLocale.get = function () { return this._vm.fallbackLocale };\nprototypeAccessors.fallbackLocale.set = function (locale) {\n this._localeChainCache = {};\n this._vm.$set(this._vm, 'fallbackLocale', locale);\n};\n\nprototypeAccessors.formatFallbackMessages.get = function () { return this._formatFallbackMessages };\nprototypeAccessors.formatFallbackMessages.set = function (fallback) { this._formatFallbackMessages = fallback; };\n\nprototypeAccessors.missing.get = function () { return this._missing };\nprototypeAccessors.missing.set = function (handler) { this._missing = handler; };\n\nprototypeAccessors.formatter.get = function () { return this._formatter };\nprototypeAccessors.formatter.set = function (formatter) { this._formatter = formatter; };\n\nprototypeAccessors.silentTranslationWarn.get = function () { return this._silentTranslationWarn };\nprototypeAccessors.silentTranslationWarn.set = function (silent) { this._silentTranslationWarn = silent; };\n\nprototypeAccessors.silentFallbackWarn.get = function () { return this._silentFallbackWarn };\nprototypeAccessors.silentFallbackWarn.set = function (silent) { this._silentFallbackWarn = silent; };\n\nprototypeAccessors.preserveDirectiveContent.get = function () { return this._preserveDirectiveContent };\nprototypeAccessors.preserveDirectiveContent.set = function (preserve) { this._preserveDirectiveContent = preserve; };\n\nprototypeAccessors.warnHtmlInMessage.get = function () { return this._warnHtmlInMessage };\nprototypeAccessors.warnHtmlInMessage.set = function (level) {\n var this$1 = this;\n\n var orgLevel = this._warnHtmlInMessage;\n this._warnHtmlInMessage = level;\n if (orgLevel !== level && (level === 'warn' || level === 'error')) {\n var messages = this._getMessages();\n Object.keys(messages).forEach(function (locale) {\n this$1._checkLocaleMessage(locale, this$1._warnHtmlInMessage, messages[locale]);\n });\n }\n};\n\nprototypeAccessors.postTranslation.get = function () { return this._postTranslation };\nprototypeAccessors.postTranslation.set = function (handler) { this._postTranslation = handler; };\n\nprototypeAccessors.sync.get = function () { return this._sync };\nprototypeAccessors.sync.set = function (val) { this._sync = val; };\n\nVueI18n.prototype._getMessages = function _getMessages () { return this._vm.messages };\nVueI18n.prototype._getDateTimeFormats = function _getDateTimeFormats () { return this._vm.dateTimeFormats };\nVueI18n.prototype._getNumberFormats = function _getNumberFormats () { return this._vm.numberFormats };\n\nVueI18n.prototype._warnDefault = function _warnDefault (locale, key, result, vm, values, interpolateMode) {\n if (!isNull(result)) { return result }\n if (this._missing) {\n var missingRet = this._missing.apply(null, [locale, key, vm, values]);\n if (isString(missingRet)) {\n return missingRet\n }\n } else {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key)) {\n warn(\n \"Cannot translate the value of keypath '\" + key + \"'. \" +\n 'Use the value of keypath as default.'\n );\n }\n }\n\n if (this._formatFallbackMessages) {\n var parsedArgs = parseArgs.apply(void 0, values);\n return this._render(key, interpolateMode, parsedArgs.params, key)\n } else {\n return key\n }\n};\n\nVueI18n.prototype._isFallbackRoot = function _isFallbackRoot (val) {\n return (this._fallbackRootWithEmptyString? !val : isNull(val)) && !isNull(this._root) && this._fallbackRoot\n};\n\nVueI18n.prototype._isSilentFallbackWarn = function _isSilentFallbackWarn (key) {\n return this._silentFallbackWarn instanceof RegExp\n ? this._silentFallbackWarn.test(key)\n : this._silentFallbackWarn\n};\n\nVueI18n.prototype._isSilentFallback = function _isSilentFallback (locale, key) {\n return this._isSilentFallbackWarn(key) && (this._isFallbackRoot() || locale !== this.fallbackLocale)\n};\n\nVueI18n.prototype._isSilentTranslationWarn = function _isSilentTranslationWarn (key) {\n return this._silentTranslationWarn instanceof RegExp\n ? this._silentTranslationWarn.test(key)\n : this._silentTranslationWarn\n};\n\nVueI18n.prototype._interpolate = function _interpolate (\n locale,\n message,\n key,\n host,\n interpolateMode,\n values,\n visitedLinkStack\n) {\n if (!message) { return null }\n\n var pathRet = this._path.getPathValue(message, key);\n if (isArray(pathRet) || isPlainObject(pathRet)) { return pathRet }\n\n var ret;\n if (isNull(pathRet)) {\n /* istanbul ignore else */\n if (isPlainObject(message)) {\n ret = message[key];\n if (!(isString(ret) || isFunction(ret))) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {\n warn((\"Value of key '\" + key + \"' is not a string or function !\"));\n }\n return null\n }\n } else {\n return null\n }\n } else {\n /* istanbul ignore else */\n if (isString(pathRet) || isFunction(pathRet)) {\n ret = pathRet;\n } else {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallback(locale, key)) {\n warn((\"Value of key '\" + key + \"' is not a string or function!\"));\n }\n return null\n }\n }\n\n // Check for the existence of links within the translated string\n if (isString(ret) && (ret.indexOf('@:') >= 0 || ret.indexOf('@.') >= 0)) {\n ret = this._link(locale, message, ret, host, 'raw', values, visitedLinkStack);\n }\n\n return this._render(ret, interpolateMode, values, key)\n};\n\nVueI18n.prototype._link = function _link (\n locale,\n message,\n str,\n host,\n interpolateMode,\n values,\n visitedLinkStack\n) {\n var ret = str;\n\n // Match all the links within the local\n // We are going to replace each of\n // them with its translation\n var matches = ret.match(linkKeyMatcher);\n\n // eslint-disable-next-line no-autofix/prefer-const\n for (var idx in matches) {\n // ie compatible: filter custom array\n // prototype method\n if (!matches.hasOwnProperty(idx)) {\n continue\n }\n var link = matches[idx];\n var linkKeyPrefixMatches = link.match(linkKeyPrefixMatcher);\n var linkPrefix = linkKeyPrefixMatches[0];\n var formatterName = linkKeyPrefixMatches[1];\n\n // Remove the leading @:, @.case: and the brackets\n var linkPlaceholder = link.replace(linkPrefix, '').replace(bracketsMatcher, '');\n\n if (includes(visitedLinkStack, linkPlaceholder)) {\n if (process.env.NODE_ENV !== 'production') {\n warn((\"Circular reference found. \\\"\" + link + \"\\\" is already visited in the chain of \" + (visitedLinkStack.reverse().join(' <- '))));\n }\n return ret\n }\n visitedLinkStack.push(linkPlaceholder);\n\n // Translate the link\n var translated = this._interpolate(\n locale, message, linkPlaceholder, host,\n interpolateMode === 'raw' ? 'string' : interpolateMode,\n interpolateMode === 'raw' ? undefined : values,\n visitedLinkStack\n );\n\n if (this._isFallbackRoot(translated)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(linkPlaceholder)) {\n warn((\"Fall back to translate the link placeholder '\" + linkPlaceholder + \"' with root locale.\"));\n }\n /* istanbul ignore if */\n if (!this._root) { throw Error('unexpected error') }\n var root = this._root.$i18n;\n translated = root._translate(\n root._getMessages(), root.locale, root.fallbackLocale,\n linkPlaceholder, host, interpolateMode, values\n );\n }\n translated = this._warnDefault(\n locale, linkPlaceholder, translated, host,\n isArray(values) ? values : [values],\n interpolateMode\n );\n\n if (this._modifiers.hasOwnProperty(formatterName)) {\n translated = this._modifiers[formatterName](translated);\n } else if (defaultModifiers.hasOwnProperty(formatterName)) {\n translated = defaultModifiers[formatterName](translated);\n }\n\n visitedLinkStack.pop();\n\n // Replace the link with the translated\n ret = !translated ? ret : ret.replace(link, translated);\n }\n\n return ret\n};\n\nVueI18n.prototype._createMessageContext = function _createMessageContext (values, formatter, path, interpolateMode) {\n var this$1 = this;\n\n var _list = isArray(values) ? values : [];\n var _named = isObject(values) ? values : {};\n var list = function (index) { return _list[index]; };\n var named = function (key) { return _named[key]; };\n var messages = this._getMessages();\n var locale = this.locale;\n\n return {\n list: list,\n named: named,\n values: values,\n formatter: formatter,\n path: path,\n messages: messages,\n locale: locale,\n linked: function (linkedKey) { return this$1._interpolate(locale, messages[locale] || {}, linkedKey, null, interpolateMode, undefined, [linkedKey]); }\n }\n};\n\nVueI18n.prototype._render = function _render (message, interpolateMode, values, path) {\n if (isFunction(message)) {\n return message(\n this._createMessageContext(values, this._formatter || defaultFormatter, path, interpolateMode)\n )\n }\n\n var ret = this._formatter.interpolate(message, values, path);\n\n // If the custom formatter refuses to work - apply the default one\n if (!ret) {\n ret = defaultFormatter.interpolate(message, values, path);\n }\n\n // if interpolateMode is **not** 'string' ('row'),\n // return the compiled data (e.g. ['foo', VNode, 'bar']) with formatter\n return interpolateMode === 'string' && !isString(ret) ? ret.join('') : ret\n};\n\nVueI18n.prototype._appendItemToChain = function _appendItemToChain (chain, item, blocks) {\n var follow = false;\n if (!includes(chain, item)) {\n follow = true;\n if (item) {\n follow = item[item.length - 1] !== '!';\n item = item.replace(/!/g, '');\n chain.push(item);\n if (blocks && blocks[item]) {\n follow = blocks[item];\n }\n }\n }\n return follow\n};\n\nVueI18n.prototype._appendLocaleToChain = function _appendLocaleToChain (chain, locale, blocks) {\n var follow;\n var tokens = locale.split('-');\n do {\n var item = tokens.join('-');\n follow = this._appendItemToChain(chain, item, blocks);\n tokens.splice(-1, 1);\n } while (tokens.length && (follow === true))\n return follow\n};\n\nVueI18n.prototype._appendBlockToChain = function _appendBlockToChain (chain, block, blocks) {\n var follow = true;\n for (var i = 0; (i < block.length) && (isBoolean(follow)); i++) {\n var locale = block[i];\n if (isString(locale)) {\n follow = this._appendLocaleToChain(chain, locale, blocks);\n }\n }\n return follow\n};\n\nVueI18n.prototype._getLocaleChain = function _getLocaleChain (start, fallbackLocale) {\n if (start === '') { return [] }\n\n if (!this._localeChainCache) {\n this._localeChainCache = {};\n }\n\n var chain = this._localeChainCache[start];\n if (!chain) {\n if (!fallbackLocale) {\n fallbackLocale = this.fallbackLocale;\n }\n chain = [];\n\n // first block defined by start\n var block = [start];\n\n // while any intervening block found\n while (isArray(block)) {\n block = this._appendBlockToChain(\n chain,\n block,\n fallbackLocale\n );\n }\n\n // last block defined by default\n var defaults;\n if (isArray(fallbackLocale)) {\n defaults = fallbackLocale;\n } else if (isObject(fallbackLocale)) {\n /* $FlowFixMe */\n if (fallbackLocale['default']) {\n defaults = fallbackLocale['default'];\n } else {\n defaults = null;\n }\n } else {\n defaults = fallbackLocale;\n }\n\n // convert defaults to array\n if (isString(defaults)) {\n block = [defaults];\n } else {\n block = defaults;\n }\n if (block) {\n this._appendBlockToChain(\n chain,\n block,\n null\n );\n }\n this._localeChainCache[start] = chain;\n }\n return chain\n};\n\nVueI18n.prototype._translate = function _translate (\n messages,\n locale,\n fallback,\n key,\n host,\n interpolateMode,\n args\n) {\n var chain = this._getLocaleChain(locale, fallback);\n var res;\n for (var i = 0; i < chain.length; i++) {\n var step = chain[i];\n res =\n this._interpolate(step, messages[step], key, host, interpolateMode, args, [key]);\n if (!isNull(res)) {\n if (step !== locale && process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to translate the keypath '\" + key + \"' with '\" + step + \"' locale.\"));\n }\n return res\n }\n }\n return null\n};\n\nVueI18n.prototype._t = function _t (key, _locale, messages, host) {\n var ref;\n\n var values = [], len = arguments.length - 4;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 4 ];\n if (!key) { return '' }\n\n var parsedArgs = parseArgs.apply(void 0, values);\n if(this._escapeParameterHtml) {\n parsedArgs.params = escapeParams(parsedArgs.params);\n }\n\n var locale = parsedArgs.locale || _locale;\n\n var ret = this._translate(\n messages, locale, this.fallbackLocale, key,\n host, 'string', parsedArgs.params\n );\n if (this._isFallbackRoot(ret)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to translate the keypath '\" + key + \"' with root locale.\"));\n }\n /* istanbul ignore if */\n if (!this._root) { throw Error('unexpected error') }\n return (ref = this._root).$t.apply(ref, [ key ].concat( values ))\n } else {\n ret = this._warnDefault(locale, key, ret, host, values, 'string');\n if (this._postTranslation && ret !== null && ret !== undefined) {\n ret = this._postTranslation(ret, key);\n }\n return ret\n }\n};\n\nVueI18n.prototype.t = function t (key) {\n var ref;\n\n var values = [], len = arguments.length - 1;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 1 ];\n return (ref = this)._t.apply(ref, [ key, this.locale, this._getMessages(), null ].concat( values ))\n};\n\nVueI18n.prototype._i = function _i (key, locale, messages, host, values) {\n var ret =\n this._translate(messages, locale, this.fallbackLocale, key, host, 'raw', values);\n if (this._isFallbackRoot(ret)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key)) {\n warn((\"Fall back to interpolate the keypath '\" + key + \"' with root locale.\"));\n }\n if (!this._root) { throw Error('unexpected error') }\n return this._root.$i18n.i(key, locale, values)\n } else {\n return this._warnDefault(locale, key, ret, host, [values], 'raw')\n }\n};\n\nVueI18n.prototype.i = function i (key, locale, values) {\n /* istanbul ignore if */\n if (!key) { return '' }\n\n if (!isString(locale)) {\n locale = this.locale;\n }\n\n return this._i(key, locale, this._getMessages(), null, values)\n};\n\nVueI18n.prototype._tc = function _tc (\n key,\n _locale,\n messages,\n host,\n choice\n) {\n var ref;\n\n var values = [], len = arguments.length - 5;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 5 ];\n if (!key) { return '' }\n if (choice === undefined) {\n choice = 1;\n }\n\n var predefined = { 'count': choice, 'n': choice };\n var parsedArgs = parseArgs.apply(void 0, values);\n parsedArgs.params = Object.assign(predefined, parsedArgs.params);\n values = parsedArgs.locale === null ? [parsedArgs.params] : [parsedArgs.locale, parsedArgs.params];\n return this.fetchChoice((ref = this)._t.apply(ref, [ key, _locale, messages, host ].concat( values )), choice)\n};\n\nVueI18n.prototype.fetchChoice = function fetchChoice (message, choice) {\n /* istanbul ignore if */\n if (!message || !isString(message)) { return null }\n var choices = message.split('|');\n\n choice = this.getChoiceIndex(choice, choices.length);\n if (!choices[choice]) { return message }\n return choices[choice].trim()\n};\n\nVueI18n.prototype.tc = function tc (key, choice) {\n var ref;\n\n var values = [], len = arguments.length - 2;\n while ( len-- > 0 ) values[ len ] = arguments[ len + 2 ];\n return (ref = this)._tc.apply(ref, [ key, this.locale, this._getMessages(), null, choice ].concat( values ))\n};\n\nVueI18n.prototype._te = function _te (key, locale, messages) {\n var args = [], len = arguments.length - 3;\n while ( len-- > 0 ) args[ len ] = arguments[ len + 3 ];\n\n var _locale = parseArgs.apply(void 0, args).locale || locale;\n return this._exist(messages[_locale], key)\n};\n\nVueI18n.prototype.te = function te (key, locale) {\n return this._te(key, this.locale, this._getMessages(), locale)\n};\n\nVueI18n.prototype.getLocaleMessage = function getLocaleMessage (locale) {\n return looseClone(this._vm.messages[locale] || {})\n};\n\nVueI18n.prototype.setLocaleMessage = function setLocaleMessage (locale, message) {\n if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {\n this._checkLocaleMessage(locale, this._warnHtmlInMessage, message);\n }\n this._vm.$set(this._vm.messages, locale, message);\n};\n\nVueI18n.prototype.mergeLocaleMessage = function mergeLocaleMessage (locale, message) {\n if (this._warnHtmlInMessage === 'warn' || this._warnHtmlInMessage === 'error') {\n this._checkLocaleMessage(locale, this._warnHtmlInMessage, message);\n }\n this._vm.$set(this._vm.messages, locale, merge(\n typeof this._vm.messages[locale] !== 'undefined' && Object.keys(this._vm.messages[locale]).length\n ? Object.assign({}, this._vm.messages[locale])\n : {},\n message\n ));\n};\n\nVueI18n.prototype.getDateTimeFormat = function getDateTimeFormat (locale) {\n return looseClone(this._vm.dateTimeFormats[locale] || {})\n};\n\nVueI18n.prototype.setDateTimeFormat = function setDateTimeFormat (locale, format) {\n this._vm.$set(this._vm.dateTimeFormats, locale, format);\n this._clearDateTimeFormat(locale, format);\n};\n\nVueI18n.prototype.mergeDateTimeFormat = function mergeDateTimeFormat (locale, format) {\n this._vm.$set(this._vm.dateTimeFormats, locale, merge(this._vm.dateTimeFormats[locale] || {}, format));\n this._clearDateTimeFormat(locale, format);\n};\n\nVueI18n.prototype._clearDateTimeFormat = function _clearDateTimeFormat (locale, format) {\n // eslint-disable-next-line no-autofix/prefer-const\n for (var key in format) {\n var id = locale + \"__\" + key;\n\n if (!this._dateTimeFormatters.hasOwnProperty(id)) {\n continue\n }\n\n delete this._dateTimeFormatters[id];\n }\n};\n\nVueI18n.prototype._localizeDateTime = function _localizeDateTime (\n value,\n locale,\n fallback,\n dateTimeFormats,\n key\n) {\n var _locale = locale;\n var formats = dateTimeFormats[_locale];\n\n var chain = this._getLocaleChain(locale, fallback);\n for (var i = 0; i < chain.length; i++) {\n var current = _locale;\n var step = chain[i];\n formats = dateTimeFormats[step];\n _locale = step;\n // fallback locale\n if (isNull(formats) || isNull(formats[key])) {\n if (step !== locale && process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to '\" + step + \"' datetime formats from '\" + current + \"' datetime formats.\"));\n }\n } else {\n break\n }\n }\n\n if (isNull(formats) || isNull(formats[key])) {\n return null\n } else {\n var format = formats[key];\n var id = _locale + \"__\" + key;\n var formatter = this._dateTimeFormatters[id];\n if (!formatter) {\n formatter = this._dateTimeFormatters[id] = new Intl.DateTimeFormat(_locale, format);\n }\n return formatter.format(value)\n }\n};\n\nVueI18n.prototype._d = function _d (value, locale, key) {\n /* istanbul ignore if */\n if (process.env.NODE_ENV !== 'production' && !VueI18n.availabilities.dateTimeFormat) {\n warn('Cannot format a Date value due to not supported Intl.DateTimeFormat.');\n return ''\n }\n\n if (!key) {\n return new Intl.DateTimeFormat(locale).format(value)\n }\n\n var ret =\n this._localizeDateTime(value, locale, this.fallbackLocale, this._getDateTimeFormats(), key);\n if (this._isFallbackRoot(ret)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to datetime localization of root: key '\" + key + \"'.\"));\n }\n /* istanbul ignore if */\n if (!this._root) { throw Error('unexpected error') }\n return this._root.$i18n.d(value, key, locale)\n } else {\n return ret || ''\n }\n};\n\nVueI18n.prototype.d = function d (value) {\n var args = [], len = arguments.length - 1;\n while ( len-- > 0 ) args[ len ] = arguments[ len + 1 ];\n\n var locale = this.locale;\n var key = null;\n\n if (args.length === 1) {\n if (isString(args[0])) {\n key = args[0];\n } else if (isObject(args[0])) {\n if (args[0].locale) {\n locale = args[0].locale;\n }\n if (args[0].key) {\n key = args[0].key;\n }\n }\n } else if (args.length === 2) {\n if (isString(args[0])) {\n key = args[0];\n }\n if (isString(args[1])) {\n locale = args[1];\n }\n }\n\n return this._d(value, locale, key)\n};\n\nVueI18n.prototype.getNumberFormat = function getNumberFormat (locale) {\n return looseClone(this._vm.numberFormats[locale] || {})\n};\n\nVueI18n.prototype.setNumberFormat = function setNumberFormat (locale, format) {\n this._vm.$set(this._vm.numberFormats, locale, format);\n this._clearNumberFormat(locale, format);\n};\n\nVueI18n.prototype.mergeNumberFormat = function mergeNumberFormat (locale, format) {\n this._vm.$set(this._vm.numberFormats, locale, merge(this._vm.numberFormats[locale] || {}, format));\n this._clearNumberFormat(locale, format);\n};\n\nVueI18n.prototype._clearNumberFormat = function _clearNumberFormat (locale, format) {\n // eslint-disable-next-line no-autofix/prefer-const\n for (var key in format) {\n var id = locale + \"__\" + key;\n\n if (!this._numberFormatters.hasOwnProperty(id)) {\n continue\n }\n\n delete this._numberFormatters[id];\n }\n};\n\nVueI18n.prototype._getNumberFormatter = function _getNumberFormatter (\n value,\n locale,\n fallback,\n numberFormats,\n key,\n options\n) {\n var _locale = locale;\n var formats = numberFormats[_locale];\n\n var chain = this._getLocaleChain(locale, fallback);\n for (var i = 0; i < chain.length; i++) {\n var current = _locale;\n var step = chain[i];\n formats = numberFormats[step];\n _locale = step;\n // fallback locale\n if (isNull(formats) || isNull(formats[key])) {\n if (step !== locale && process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to '\" + step + \"' number formats from '\" + current + \"' number formats.\"));\n }\n } else {\n break\n }\n }\n\n if (isNull(formats) || isNull(formats[key])) {\n return null\n } else {\n var format = formats[key];\n\n var formatter;\n if (options) {\n // If options specified - create one time number formatter\n formatter = new Intl.NumberFormat(_locale, Object.assign({}, format, options));\n } else {\n var id = _locale + \"__\" + key;\n formatter = this._numberFormatters[id];\n if (!formatter) {\n formatter = this._numberFormatters[id] = new Intl.NumberFormat(_locale, format);\n }\n }\n return formatter\n }\n};\n\nVueI18n.prototype._n = function _n (value, locale, key, options) {\n /* istanbul ignore if */\n if (!VueI18n.availabilities.numberFormat) {\n if (process.env.NODE_ENV !== 'production') {\n warn('Cannot format a Number value due to not supported Intl.NumberFormat.');\n }\n return ''\n }\n\n if (!key) {\n var nf = !options ? new Intl.NumberFormat(locale) : new Intl.NumberFormat(locale, options);\n return nf.format(value)\n }\n\n var formatter = this._getNumberFormatter(value, locale, this.fallbackLocale, this._getNumberFormats(), key, options);\n var ret = formatter && formatter.format(value);\n if (this._isFallbackRoot(ret)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key) && !this._isSilentFallbackWarn(key)) {\n warn((\"Fall back to number localization of root: key '\" + key + \"'.\"));\n }\n /* istanbul ignore if */\n if (!this._root) { throw Error('unexpected error') }\n return this._root.$i18n.n(value, Object.assign({}, { key: key, locale: locale }, options))\n } else {\n return ret || ''\n }\n};\n\nVueI18n.prototype.n = function n (value) {\n var args = [], len = arguments.length - 1;\n while ( len-- > 0 ) args[ len ] = arguments[ len + 1 ];\n\n var locale = this.locale;\n var key = null;\n var options = null;\n\n if (args.length === 1) {\n if (isString(args[0])) {\n key = args[0];\n } else if (isObject(args[0])) {\n if (args[0].locale) {\n locale = args[0].locale;\n }\n if (args[0].key) {\n key = args[0].key;\n }\n\n // Filter out number format options only\n options = Object.keys(args[0]).reduce(function (acc, key) {\n var obj;\n\n if (includes(numberFormatKeys, key)) {\n return Object.assign({}, acc, ( obj = {}, obj[key] = args[0][key], obj ))\n }\n return acc\n }, null);\n }\n } else if (args.length === 2) {\n if (isString(args[0])) {\n key = args[0];\n }\n if (isString(args[1])) {\n locale = args[1];\n }\n }\n\n return this._n(value, locale, key, options)\n};\n\nVueI18n.prototype._ntp = function _ntp (value, locale, key, options) {\n /* istanbul ignore if */\n if (!VueI18n.availabilities.numberFormat) {\n if (process.env.NODE_ENV !== 'production') {\n warn('Cannot format to parts a Number value due to not supported Intl.NumberFormat.');\n }\n return []\n }\n\n if (!key) {\n var nf = !options ? new Intl.NumberFormat(locale) : new Intl.NumberFormat(locale, options);\n return nf.formatToParts(value)\n }\n\n var formatter = this._getNumberFormatter(value, locale, this.fallbackLocale, this._getNumberFormats(), key, options);\n var ret = formatter && formatter.formatToParts(value);\n if (this._isFallbackRoot(ret)) {\n if (process.env.NODE_ENV !== 'production' && !this._isSilentTranslationWarn(key)) {\n warn((\"Fall back to format number to parts of root: key '\" + key + \"' .\"));\n }\n /* istanbul ignore if */\n if (!this._root) { throw Error('unexpected error') }\n return this._root.$i18n._ntp(value, locale, key, options)\n } else {\n return ret || []\n }\n};\n\nObject.defineProperties( VueI18n.prototype, prototypeAccessors );\n\nvar availabilities;\n// $FlowFixMe\nObject.defineProperty(VueI18n, 'availabilities', {\n get: function get () {\n if (!availabilities) {\n var intlDefined = typeof Intl !== 'undefined';\n availabilities = {\n dateTimeFormat: intlDefined && typeof Intl.DateTimeFormat !== 'undefined',\n numberFormat: intlDefined && typeof Intl.NumberFormat !== 'undefined'\n };\n }\n\n return availabilities\n }\n});\n\nVueI18n.install = install;\nVueI18n.version = '8.27.2';\n\nexport default VueI18n;\n","// src/index.ts\nimport Vue, { computed, getCurrentInstance } from \"vue\";\nimport VueI18n from \"vue-i18n\";\nvar i18nInstance;\nfunction createI18n(options) {\n i18nInstance = new VueI18n(options);\n return i18nInstance;\n}\nfunction useI18n() {\n if (!i18nInstance)\n throw new Error(\"vue-i18n not initialized\");\n const i18n = i18nInstance;\n const instance = getCurrentInstance();\n const vm = (instance == null ? void 0 : instance.proxy) || instance || new Vue({});\n const locale = computed({\n get() {\n return i18n.locale;\n },\n set(v) {\n i18n.locale = v;\n }\n });\n return {\n locale,\n t: vm.$t.bind(vm),\n tc: vm.$tc.bind(vm),\n d: vm.$d.bind(vm),\n te: vm.$te.bind(vm),\n n: vm.$n.bind(vm)\n };\n}\nexport {\n createI18n,\n useI18n\n};\n","import { DateTimeFormats } from \"vue-i18n\";\n\nconst dateTimeFormats: DateTimeFormats = {\n en: {\n short: {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n },\n long: {\n year: \"numeric\",\n month: \"long\",\n day: \"numeric\",\n weekday: \"long\",\n hour: \"numeric\",\n minute: \"numeric\",\n },\n },\n};\n\nexport default dateTimeFormats;\n","import Vue from \"vue\";\nimport VueI18n from \"vue-i18n\";\nimport { createI18n } from \"vue-i18n-composable\";\nimport dateTimeFormats from \"@/locales/dateTimeFormats\";\nimport messages from \"./locales/en.json\";\nimport appSettings from \"./appSettings\";\n\nVue.use(VueI18n);\n\nconst options = {\n locale: appSettings.i18n.locale,\n fallbackLocale: appSettings.i18n.fallbackLocale,\n messages: { en: messages },\n dateTimeFormats,\n};\n\nexport default createI18n(options);\n","\"use strict\";\nvar _extends =\n Object.assign ||\n function (target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i];\n for (var key in source) {\n if (Object.prototype.hasOwnProperty.call(source, key)) {\n target[key] = source[key];\n }\n }\n }\n return target;\n };\n\n/**\n * Find the actual point position of a transformed point\n *\n * @param {Object} payload an object holding required information to find actual point\n * @param {number} payload.x position of x\n * @param {number} payload.y position of y\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x,y}} the center of element\n * @param {number} payload.rad the a computed radians of a provided angle\n *\n * @returns {{x: number, y: number}} an object holding the position\n */\nvar findPoint = function findPoint(_ref) {\n var x = _ref.x,\n y = _ref.y,\n angle = _ref.angle,\n center = _ref.center,\n _ref$rad = _ref.rad,\n rad = _ref$rad === undefined ? angle * (Math.PI / 180) : _ref$rad;\n return {\n x:\n (x - center.x) * Math.cos(rad) -\n (y - center.y) * Math.sin(rad) +\n center.x,\n y:\n (x - center.x) * Math.sin(rad) +\n (y - center.y) * Math.cos(rad) +\n center.y,\n };\n};\n\n/**\n * Get the Center point of a box\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n *\n * @returns {{x: *, y: *}} the center of point of element\n */\nexport const getCenter = function getCenter(_ref2) {\n var x = _ref2.x,\n y = _ref2.y,\n scaleX = _ref2.scaleX,\n scaleY = _ref2.scaleY,\n width = _ref2.width,\n height = _ref2.height;\n\n var changedWidth = width * scaleX;\n var changedHeight = height * scaleY;\n\n var changedWidthDiff = changedWidth - width;\n var changedHeightDiff = changedHeight - height;\n\n return {\n x: x - changedWidthDiff + changedWidth / 2,\n y: y - changedHeightDiff + changedHeight / 2,\n };\n};\n\n/**\n * get the TopLeft point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getTL = function getTL(_ref3) {\n var x = _ref3.x,\n y = _ref3.y,\n scaleX = _ref3.scaleX,\n scaleY = _ref3.scaleY,\n width = _ref3.width,\n height = _ref3.height,\n angle = _ref3.angle,\n _ref3$center = _ref3.center,\n center =\n _ref3$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref3$center;\n return findPoint({\n x: x,\n y: y,\n angle: angle,\n center: center,\n });\n};\n\n/**\n * get the LeftBottom point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getBL = function getBL(_ref4) {\n var x = _ref4.x,\n y = _ref4.y,\n scaleX = _ref4.scaleX,\n scaleY = _ref4.scaleY,\n width = _ref4.width,\n height = _ref4.height,\n angle = _ref4.angle,\n _ref4$center = _ref4.center,\n center =\n _ref4$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref4$center;\n\n return findPoint({\n angle: angle,\n center: center,\n x: x,\n y: y + height * scaleY,\n });\n};\n\n/**\n * Get TopRight point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getTR = function getTR(_ref5) {\n var x = _ref5.x,\n y = _ref5.y,\n scaleX = _ref5.scaleX,\n scaleY = _ref5.scaleY,\n width = _ref5.width,\n height = _ref5.height,\n angle = _ref5.angle,\n _ref5$center = _ref5.center,\n center =\n _ref5$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref5$center;\n return findPoint({\n angle: angle,\n center: center,\n x: x + width * scaleX,\n y: y,\n });\n};\n\n/**\n * Get BottomRight point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getBR = function getBR(_ref6) {\n var x = _ref6.x,\n y = _ref6.y,\n scaleX = _ref6.scaleX,\n scaleY = _ref6.scaleY,\n width = _ref6.width,\n height = _ref6.height,\n angle = _ref6.angle,\n _ref6$center = _ref6.center,\n center =\n _ref6$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref6$center;\n\n return findPoint({\n angle: angle,\n center: center,\n x: x + width * scaleX,\n y: y + height * scaleY,\n });\n};\n\n/**\n * get MiddleRight point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getMR = function getMR(_ref7) {\n var x = _ref7.x,\n y = _ref7.y,\n scaleX = _ref7.scaleX,\n scaleY = _ref7.scaleY,\n width = _ref7.width,\n height = _ref7.height,\n angle = _ref7.angle,\n _ref7$center = _ref7.center,\n center =\n _ref7$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref7$center;\n return findPoint({\n x: x + width * scaleX,\n y: y + (height * scaleY) / 2,\n center: center,\n angle: angle,\n });\n};\n\n/**\n * get MiddleBottom point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getBM = function getBM(_ref8) {\n var x = _ref8.x,\n y = _ref8.y,\n scaleX = _ref8.scaleX,\n scaleY = _ref8.scaleY,\n width = _ref8.width,\n height = _ref8.height,\n angle = _ref8.angle,\n _ref8$center = _ref8.center,\n center =\n _ref8$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref8$center;\n return findPoint({\n x: x + (width * scaleX) / 2,\n y: y + height * scaleY,\n center: center,\n angle: angle,\n });\n};\n\n/**\n * get MiddleTop point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nexport const getTM = function getTM(_ref9) {\n var x = _ref9.x,\n y = _ref9.y,\n scaleX = _ref9.scaleX,\n scaleY = _ref9.scaleY,\n width = _ref9.width,\n height = _ref9.height,\n angle = _ref9.angle,\n _ref9$center = _ref9.center,\n center =\n _ref9$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref9$center;\n return findPoint({\n x: x + (width * scaleX) / 2,\n y: y,\n center: center,\n angle: angle,\n });\n};\n\n/**\n * get MiddleLeft point position\n *\n * @param {Object} payload element information\n * @param {number} payload.x the position of x\n * @param {number} payload.y the position of y\n * @param {number} payload.scaleX the scaleX of element\n * @param {number} payload.scaleY the scaleY of element\n * @param {number} payload.width the original width of element\n * @param {number} payload.height the original height of element\n * @param {number} payload.angle the rotation angle\n * @param {Object} payload.center {{x:number, y:number}}\n *\n * @returns {{x: number, y: number}} the position\n */\nconst getML = function getML(_ref10) {\n var x = _ref10.x,\n y = _ref10.y,\n scaleX = _ref10.scaleX,\n scaleY = _ref10.scaleY,\n width = _ref10.width,\n height = _ref10.height,\n angle = _ref10.angle,\n _ref10$center = _ref10.center,\n center =\n _ref10$center === undefined\n ? getCenter({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n })\n : _ref10$center;\n return findPoint({\n x: x,\n y: y + (height * scaleY) / 2,\n center: center,\n angle: angle,\n });\n};\n\n/**\n * given a point, get it's opposite point\n *\n * @param {string} scaleType scale point position name\n * @param {Object} props element information\n * @param {number} props.x the position of x\n * @param {number} props.y the position of y\n * @param {number} props.scaleX the scaleX of element\n * @param {number} props.scaleY the scaleY of element\n * @param {number} props.width the original width of element\n * @param {number} props.height the original height of element\n * @param {number} props.angle the rotation angle\n * @param {Object} props.center {{x:number, y:number}}\n *\n * @returns {{x:number, y:number}} point position\n */\nexport const getOppositePoint = function getOppositePoint(scaleType, props) {\n var caller = void 0;\n\n var center = getCenter({\n x: props.x,\n y: props.y,\n width: props.width,\n height: props.height,\n scaleX: props.scaleX,\n scaleY: props.scaleY,\n });\n\n props = _extends(\n {\n center: center,\n },\n props,\n {\n x: getOriginalPositionFromScale(props.x, props.width, props.scaleX),\n y: getOriginalPositionFromScale(props.y, props.height, props.scaleY),\n }\n );\n\n switch (scaleType) {\n case \"tl\":\n caller = getBR;\n break;\n\n case \"ml\":\n caller = getMR;\n break;\n\n case \"tr\":\n caller = getBL;\n break;\n\n case \"tm\":\n caller = getBM;\n break;\n\n case \"bl\":\n caller = getTR;\n break;\n\n case \"bm\":\n caller = getTM;\n break;\n\n case \"br\":\n caller = getTL;\n break;\n\n case \"mr\":\n caller = getML;\n break;\n }\n return caller(props);\n};\n\n/**\n * given a point position by it's string name\n *\n * @param {string} scaleType scale point position name\n * @param {Object} props element information\n * @param {number} props.x the position of x\n * @param {number} props.y the position of y\n * @param {number} props.scaleX the scaleX of element\n * @param {number} props.scaleY the scaleY of element\n * @param {number} props.width the original width of element\n * @param {number} props.height the original height of element\n * @param {number} props.angle the rotation angle\n * @param {boolean} props.scaleFromCenter scaling performed from center\n * @param {Object} props.center {{x:number, y:number}}\n *\n * @returns {{x:number, y:number}} point position\n */\nexport const getPoint = function getPoint(scaleType, props) {\n var center = getCenter({\n x: props.x,\n y: props.y,\n width: props.width,\n height: props.height,\n scaleX: props.scaleX,\n scaleY: props.scaleY,\n });\n\n if (props.scaleFromCenter) {\n return center;\n }\n\n props = _extends(\n {\n center: center,\n },\n props,\n {\n x: getOriginalPositionFromScale(props.x, props.width, props.scaleX),\n y: getOriginalPositionFromScale(props.y, props.height, props.scaleY),\n }\n );\n\n var caller = void 0;\n switch (scaleType) {\n case \"tl\":\n caller = getTL;\n break;\n\n case \"ml\":\n caller = getML;\n break;\n\n case \"tr\":\n caller = getTR;\n break;\n\n case \"tm\":\n caller = getTM;\n break;\n\n case \"bl\":\n caller = getBL;\n break;\n\n case \"bm\":\n caller = getBM;\n break;\n\n case \"br\":\n caller = getBR;\n break;\n\n case \"mr\":\n caller = getMR;\n break;\n }\n\n return caller(props);\n};\n\n/**\n * get sine and cosine for a point based on angle and point name\n *\n * @param {string} scaleType scale point position name\n * @param {number} angle the rotation angle\n *\n * @returns {{sin: number, cos: number}} the sine and cosine of scale type\n */\nexport const getSineCosine = function getSineCosine(scaleType, angle) {\n switch (scaleType) {\n case \"tr\":\n case \"tm\":\n case \"bl\":\n case \"bm\":\n return {\n cos: Math.cos(-angle * (Math.PI / 180)),\n sin: Math.sin(-angle * (Math.PI / 180)),\n };\n default:\n return {\n sin: Math.sin(angle * (Math.PI / 180)),\n cos: Math.cos(angle * (Math.PI / 180)),\n };\n }\n};\n\n/**\n * get the amount of movement for a point\n *\n * @param {string} scaleType scale point position name\n * @param {object} oppositePoint the opposite point position {x: number,y: number}\n * @param {object} point the point position {x: number,y: number}\n * @param {object} moveDiff the the amount of pixels that element moved {x: number,y: number}\n *\n * @returns {{x: number, y:number}} the new position of moved element\n */\nexport const getMovePoint = function getMovePoint(\n scaleType,\n oppositePoint,\n point,\n moveDiff\n) {\n switch (scaleType) {\n case \"tl\":\n return {\n x: oppositePoint.x - (moveDiff.x + point.x),\n y: oppositePoint.y - (moveDiff.y + point.y),\n };\n case \"ml\":\n return {\n x: oppositePoint.x - moveDiff.x - point.x,\n y: oppositePoint.y - moveDiff.y - point.y,\n };\n\n case \"tr\":\n case \"tm\":\n return {\n x: point.x + (moveDiff.x - oppositePoint.x),\n y: oppositePoint.y - (moveDiff.y + point.y),\n };\n case \"mr\":\n case \"br\":\n return {\n x: point.x + (moveDiff.x - oppositePoint.x),\n y: point.y + (moveDiff.y - oppositePoint.y),\n };\n case \"bl\":\n case \"bm\":\n return {\n x: oppositePoint.x - (moveDiff.x + point.x),\n y: point.y + (moveDiff.y - oppositePoint.y),\n };\n }\n};\n\n/**\n * guess the original point position based on scale and the position after scaling\n *\n * @param {number} position the position of x or y\n * @param {number} size the size of element (width for x, height for y)\n * @param {number} scale the amount of scaled element (scaleX for x, scaleY for y)\n *\n * @returns {number} the original point position\n */\nvar getOriginalPositionFromScale = function getOriginalPositionFromScale(\n position,\n size,\n scale\n) {\n var changed = size * scale;\n\n var diff = changed - size;\n\n return position - diff;\n};\n\n/**\n * Find the real position of lowest and highest handle\n *\n * @param point\n * @returns {{x: number, y: number}|*}\n */\nexport const minMax = function minMax(point) {\n var points = [getTL(point), getTR(point), getBL(point), getBR(point)];\n\n var bounds = points.reduce(function (bounds, point, c) {\n if (c === 0) {\n bounds.xmin = point.x;\n bounds.xmax = point.x;\n bounds.ymin = point.y;\n bounds.ymax = point.y;\n } else {\n bounds.xmin = Math.min(bounds.xmin, point.x);\n bounds.xmax = Math.max(bounds.xmax, point.x);\n bounds.ymin = Math.min(bounds.ymin, point.y);\n bounds.ymax = Math.max(bounds.ymax, point.y);\n }\n return bounds;\n }, {});\n\n return bounds;\n};\n","\"use strict\";\nvar _extends =\n Object.assign ||\n function (target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i];\n for (var key in source) {\n if (Object.prototype.hasOwnProperty.call(source, key)) {\n target[key] = source[key];\n }\n }\n }\n return target;\n };\n\nimport * as _pointFinder from \"./point-finder\";\n\n/**\n * Perform Scaling based on a positioned handle\n *\n * @param {string} scaleType scale point position name\n * @param {Object} payload an object holding element information\n * @param {number} payload.startX mouse down position on X axis\n * @param {number} payload.startY mouse down position on Y axis\n * @param {number} payload.x position of x\n * @param {number} payload.y position of y\n * @param {number} payload.scaleX amount of scale for x (width)\n * @param {number} payload.scaleY amount of scale for y (height)\n * @param {number} payload.canvasScale scale of canvas\n * @param {number} payload.width original width\n * @param {number} payload.height original height\n * @param {number} payload.angle the angle of rotation\n * @param {number} payload.scaleLimit minimum scale limit\n * @param {boolean} payload.scaleFromCenter is scale from center\n * @param {boolean} payload.aspectRatio is scale on aspect ration\n * @param {boolean} payload.lockAspect whether aspect ratio is locked on widget\n * @param {Function} onUpdate a callback on mouse up\n */\nexport default function (scaleType, _ref, onUpdate) {\n var startX = _ref.startX,\n startY = _ref.startY,\n x = _ref.x,\n y = _ref.y,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n scl = _ref.canvasScale,\n width = _ref.width,\n height = _ref.height,\n angle = _ref.angle,\n scaleLimit = _ref.scaleLimit,\n _ref$scaleFromCenter = _ref.scaleFromCenter,\n scaleFromCenter =\n _ref$scaleFromCenter === undefined ? false : _ref$scaleFromCenter,\n _ref$enableScaleFromC = _ref.enableScaleFromCenter,\n enableScaleFromCenter =\n _ref$enableScaleFromC === undefined ? true : _ref$enableScaleFromC,\n _ref$aspectRatio = _ref.aspectRatio,\n aspectRatio = _ref$aspectRatio === undefined ? false : _ref$aspectRatio,\n _ref$enableAspectRati = _ref.enableAspectRatio,\n enableAspectRatio =\n _ref$enableAspectRati === undefined ? true : _ref$enableAspectRati,\n lockAspect = _ref.lockAspect;\n\n var ratio = (width * scaleX) / (height * scaleY);\n\n var point = (0, _pointFinder.getPoint)(scaleType, {\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n angle: angle,\n scaleFromCenter: scaleFromCenter,\n });\n\n var oppositePoint = (0, _pointFinder.getOppositePoint)(scaleType, {\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n angle: angle,\n });\n\n var currentProps = {\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n };\n\n return function (event) {\n if (\n enableScaleFromCenter &&\n ((event.altKey && !scaleFromCenter) || (!event.altKey && scaleFromCenter))\n ) {\n startX = event.pageX;\n startY = event.pageY;\n\n scaleFromCenter = event.altKey && !scaleFromCenter;\n\n point = (0, _pointFinder.getPoint)(\n scaleType,\n _extends({}, currentProps, {\n width: width,\n height: height,\n angle: angle,\n scaleFromCenter: scaleFromCenter,\n })\n );\n\n oppositePoint = (0, _pointFinder.getOppositePoint)(\n scaleType,\n _extends({}, currentProps, {\n width: width,\n height: height,\n angle: angle,\n })\n );\n }\n\n if (!event.shiftKey && aspectRatio) {\n aspectRatio = false;\n } else if (event.shiftKey && !aspectRatio) {\n aspectRatio = true;\n }\n\n if (!enableAspectRatio) {\n aspectRatio = false;\n }\n // new\n if (lockAspect) {\n aspectRatio = true;\n }\n // newer and somewhat hacky (shouldn't be here...)\n // Hmm....seems to fail if we have resized the wg......\n // yeah wwhy not just override lockAspect property?\n // const isCornerHandle = !scaleType.includes('m');\n // if (widgetType === 'Text' && isCornerHandle) {\n // aspectRatio = true;\n // }\n\n var moveDiff = {\n x: (event.pageX - startX) / scl,\n y: (event.pageY - startY) / scl,\n };\n\n var movePoint = (0, _pointFinder.getMovePoint)(\n scaleType,\n oppositePoint,\n point,\n moveDiff\n );\n\n // console.log(\"move\", movePoint);\n\n if (enableScaleFromCenter && scaleFromCenter) {\n movePoint.x *= 2;\n movePoint.y *= 2;\n }\n\n var _getSineCosine = (0, _pointFinder.getSineCosine)(scaleType, angle),\n sin = _getSineCosine.sin,\n cos = _getSineCosine.cos;\n\n var rotationPoint = {\n x: movePoint.x * cos + movePoint.y * sin,\n y: movePoint.y * cos - movePoint.x * sin,\n };\n\n currentProps.scaleX =\n rotationPoint.x / width > scaleLimit\n ? rotationPoint.x / width\n : scaleLimit;\n currentProps.scaleY =\n rotationPoint.y / height > scaleLimit\n ? rotationPoint.y / height\n : scaleLimit;\n\n switch (scaleType) {\n case \"ml\":\n case \"mr\":\n currentProps.scaleY = scaleY;\n if (aspectRatio) {\n currentProps.scaleY =\n (width * currentProps.scaleX * (1 / ratio)) / height;\n }\n break;\n case \"tm\":\n case \"bm\":\n currentProps.scaleX = scaleX;\n if (aspectRatio) {\n currentProps.scaleX = (height * currentProps.scaleY * ratio) / width;\n }\n break;\n default:\n if (aspectRatio) {\n currentProps.scaleY =\n (width * currentProps.scaleX * (1 / ratio)) / height;\n }\n }\n\n if (enableScaleFromCenter && scaleFromCenter) {\n var center = (0, _pointFinder.getCenter)({\n x: x,\n y: y,\n width: width,\n height: height,\n scaleX: currentProps.scaleX,\n scaleY: currentProps.scaleY,\n });\n currentProps.x = x + (point.x - center.x);\n currentProps.y = y + (point.y - center.y);\n } else {\n var freshOppositePoint = (0, _pointFinder.getOppositePoint)(scaleType, {\n width: width,\n height: height,\n angle: angle,\n x: x,\n y: y,\n scaleX: currentProps.scaleX,\n scaleY: currentProps.scaleY,\n });\n\n currentProps.x = x + (oppositePoint.x - freshOppositePoint.x);\n currentProps.y = y + (oppositePoint.y - freshOppositePoint.y);\n }\n\n // Ok....top corners do NOT change y, and left corners do NOT change x.\n // bottom and right ones DO. and scales always change.\n\n onUpdate(currentProps);\n };\n}\n","\"use strict\";\n\nimport * as _pointFinder from \"./point-finder\";\nimport { subtractVectors } from \"@/utils\";\nimport { screenCoordsToCanvas } from \"@/utils\";\n\n// Takes a position in canvas coordinates and finds its coordinates relative to a target point (e.g. a widget's center):\nconst getPositionRelativeTo = (pt, tgt, artboardPosition) => {\n const ptRelArtboard = subtractVectors(pt, artboardPosition);\n const ptRelTgt = subtractVectors(ptRelArtboard, tgt);\n return ptRelTgt;\n};\n\nexport default function (_ref, onUpdate) {\n const x = _ref.x,\n y = _ref.y,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n scl = _ref.canvasScale,\n width = _ref.width,\n height = _ref.height,\n angle = _ref.angle,\n startX = _ref.startX,\n startY = _ref.startY,\n canvasX = _ref.canvasX,\n canvasY = _ref.canvasY,\n canvasW = _ref.canvasW,\n canvasH = _ref.canvasH,\n artboardX = _ref.artboardX,\n artboardY = _ref.artboardY;\n\n const wgCenter = (0, _pointFinder.getCenter)({\n x: x,\n y: y,\n scaleX: scaleX,\n scaleY: scaleY,\n width: width,\n height: height,\n });\n\n const cBox = {\n x: canvasX,\n y: canvasY,\n w: canvasW,\n h: canvasH,\n };\n\n const ptRelCanvas = screenCoordsToCanvas({ x: startX, y: startY }, cBox, scl);\n const ptRelWgCtr = getPositionRelativeTo(ptRelCanvas, wgCenter, {\n x: artboardX,\n y: artboardY,\n });\n const pressAngle = (Math.atan2(ptRelWgCtr.y, ptRelWgCtr.x) * 180) / Math.PI;\n // console.log(\"pressAng\", pressAngle);s\n\n return function (event) {\n const ptRelCanvas = screenCoordsToCanvas(\n { x: event.pageX, y: event.pageY },\n cBox,\n scl\n );\n const ptRelWgCtr = getPositionRelativeTo(ptRelCanvas, wgCenter, {\n x: artboardX,\n y: artboardY,\n });\n const degree = (Math.atan2(ptRelWgCtr.y, ptRelWgCtr.x) * 180) / Math.PI;\n let ang = angle + degree - pressAngle;\n\n if (event.shiftKey) {\n ang = ((ang / 15) >> 0) * 15;\n }\n\n // console.log(\"rot....\");\n\n onUpdate({\n angle: ang,\n });\n };\n}\n","\"use strict\";\n\nconst SNAPPING_RANGE = 20;\n\n// - [ ] NOTE: Do we just need to do \"touches\" stuff in here to fix Stylus bugs?\n// Or are they Windows related somehow\n// Handle snapping in here?\n// With the \"real\" dims idea, to capture where it is even if appears \"glued\" in place?\n// and the we use THOSE dims to look for new snap pairs, rather than actual x,y\n\n// - [x] don't forget to test with diff canvas scales.\n\nconst isSameAxis = (first, second) => {\n const isX = first.name.includes(\"x\");\n const tgtX = second.name.includes(\"x\");\n const isY = first.name.includes(\"y\");\n const tgtY = second.name.includes(\"y\");\n return (isX && tgtX) || (isY && tgtY);\n};\n\n// Oh jeez, we're going to have to map to getApparentDims before passing to this, eh?\nconst getAxes = (dims) => {\n const { x, y, w, h } = dims;\n return {\n x: x,\n cx: x + w / 2,\n rx: x + w,\n y: y,\n cy: y + h / 2,\n by: y + h,\n };\n};\n\n// NOTE that \"artboard\" can now also refer to repeater editing cell\n// Filter wgs by wid before calling this:\nconst getSnapTargets = (widgets, artboard) => {\n const axes = widgets\n .map((wg) => {\n const apparent = getApparentDims(wg);\n const axes = getAxes(apparent);\n\n return Object.keys(axes).map((key) => {\n return {\n name: key,\n value: axes[key],\n // wid: wg.wid\n };\n });\n })\n .flat();\n\n const artboardAxes = [\n { name: \"x\", value: 0 },\n { name: \"cx\", value: artboard.w / 2 },\n { name: \"rx\", value: artboard.w },\n { name: \"y\", value: 0 },\n { name: \"cy\", value: artboard.h / 2 },\n { name: \"by\", value: artboard.h },\n ].map((axis) => {\n return { ...axis, type: \"artboard\" };\n });\n\n return [...axes, ...artboardAxes];\n};\n\n// Note: Keep track of whether snapped to artboard or other wg (hint lines could change)\nconst getSnappedAxes = (dims, snapTargets) => {\n // Get element's own axes:\n const myAxes = getAxes(dims);\n\n // const RANGE = 8; // unsure whether we should divide by this.scale\n\n // Keep track of minimum-distance x- and y-axis snapping pairs:\n let srcX = {\n name: \"cx\",\n value: -1,\n dist: Infinity,\n };\n let srcY = {\n name: \"cy\",\n value: -1,\n dist: Infinity,\n };\n let tgtX = {\n name: \"\",\n value: -1,\n };\n let tgtY = {\n name: \"\",\n value: -1,\n };\n\n // Find nearest targets:\n snapTargets.forEach((possibleTgt) => {\n Object.keys(myAxes).forEach((key) => {\n const axisVal = myAxes[key];\n const myAxis = {\n name: key,\n value: axisVal,\n };\n const dist = Math.abs(possibleTgt.value - axisVal);\n\n if (\n dist < srcX.dist &&\n isSameAxis(myAxis, possibleTgt) &&\n isSameAxis(myAxis, srcX)\n ) {\n srcX.name = key;\n srcX.value = axisVal;\n srcX.dist = dist;\n tgtX = possibleTgt;\n }\n if (\n dist < srcY.dist &&\n isSameAxis(myAxis, possibleTgt) &&\n isSameAxis(myAxis, srcY)\n ) {\n srcY.name = key;\n srcY.value = axisVal;\n srcY.dist = dist;\n tgtY = possibleTgt;\n }\n });\n });\n\n const allTgts = [\n { src: srcX, tgt: tgtX },\n { src: srcY, tgt: tgtY },\n ].filter((pair) => {\n return pair.src.dist < SNAPPING_RANGE;\n });\n\n return allTgts;\n};\n\nconst getApparentDims = (wg) => {\n if (!wg) return { x: 0, y: 0, w: 0, h: 0 };\n const right = wg.x + wg.w;\n const x = right - wg.scaleX * wg.w;\n const btm = wg.y + wg.h;\n const y = btm - wg.scaleY * wg.h;\n const w = wg.w * wg.scaleX;\n const h = wg.h * wg.scaleY;\n return { ...wg, x, y, w, h };\n};\n\nconst getApparentDimsInvert = (data) => {\n const w = data.w / data.scaleX;\n const h = data.h / data.scaleY;\n const x = data.x - (1 - data.scaleX) * w;\n const y = data.y - (1 - data.scaleY) * h;\n return { x, y, w, h };\n};\n\nconst applySnapping = (snapPairs, dims) => {\n const { x, y, w, h } = dims;\n\n var newX = x;\n var newY = y;\n\n snapPairs.forEach((pair) => {\n const snapAxis = pair.src;\n const tgt = pair.tgt;\n let xOff = 0;\n let yOff = 0;\n\n // snap or \"glue\" the element into place:\n if (tgt.name.includes(\"x\")) {\n newX = tgt.value;\n\n if (snapAxis.name === \"cx\") {\n xOff = w / 2;\n }\n if (snapAxis.name === \"rx\") {\n xOff = w;\n }\n newX -= xOff;\n } else {\n newY = tgt.value;\n\n if (snapAxis.name === \"cy\") {\n yOff = h / 2;\n }\n if (snapAxis.name === \"by\") {\n yOff = h;\n }\n newY -= yOff;\n }\n });\n\n return { x: newX, y: newY };\n};\n\nexport default function (_ref, onUpdate) {\n var x = _ref.x,\n y = _ref.y,\n startX = _ref.startX,\n startY = _ref.startY,\n scl = _ref.canvasScale,\n w = _ref.w,\n h = _ref.h,\n angle = _ref.angle,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n widgets = _ref.widgets,\n wid = _ref.wid,\n artboard = _ref.artboard,\n isBaseEditingContext = _ref.isBaseEditingContext,\n editingContext = _ref.editingContext;\n\n // return function (dragEvent) {\n // x += (dragEvent.pageX - startX) / scl;\n // y += (dragEvent.pageY - startY) / scl;\n // onUpdate({ x: x, y: y });\n // startX = dragEvent.pageX;\n // startY = dragEvent.pageY;\n // };\n\n // Do not overwrite initial startX,startY,x,y values\n // but instead compute based on difference from those initial values every time\n return function (dragEvent) {\n // Testing....\n const pageX = dragEvent.touches\n ? dragEvent.touches[0].pageX\n : dragEvent.pageX;\n const pageY = dragEvent.touches\n ? dragEvent.touches[0].pageY\n : dragEvent.pageY;\n\n let newx = x + (pageX - startX) / scl;\n let newy = y + (pageY - startY) / scl;\n\n const apparent = getApparentDims({\n x: newx,\n y: newy,\n w,\n h,\n scaleX,\n scaleY,\n });\n\n let possibleSnapPairs = [];\n\n let widgetSnapTargets = widgets.filter((wg) => wg.wid !== wid);\n let secondarySnapTarget = artboard;\n\n /*\n * If a repeater is being edited, update secondarySnapTarget to cell's dimensions\n */\n\n let ignoreSnapping = angle !== 0;\n\n if (!ignoreSnapping && !isBaseEditingContext) {\n const { offsetX, offsetY, parentId, height, width } = editingContext;\n const parent = widgets.find((w) => w.wid === parentId);\n const { x, y } = getApparentDims(parent);\n const cellDims = {\n x: x + offsetX,\n y: y + offsetY,\n w: width,\n h: height,\n };\n secondarySnapTarget = cellDims;\n\n // Ignore snapping in a rotated Repeater, or a Repeater within a Group (whose parent cannot be found here):\n if (parent?.angle !== 0 || !parent) {\n ignoreSnapping = true;\n }\n\n // Not working... But maybe it's too busy anyway. Just snap to cell borders.\n // widgetSnapTargets = widgetSnapTargets.filter(\n // wg => wg.parentId === parentId\n // );\n widgetSnapTargets = [];\n }\n\n if (!ignoreSnapping) {\n const snapTargets = getSnapTargets(\n widgetSnapTargets,\n secondarySnapTarget\n );\n possibleSnapPairs = getSnappedAxes(apparent, snapTargets);\n }\n\n // -[ ] DO NOT try to snap if rotated\n\n // Have to do the apparent => merge => invert rigamarole:\n if (possibleSnapPairs.length > 0) {\n const newDims = applySnapping(possibleSnapPairs, apparent);\n // console.log(\"snappairs\", possibleSnapPairs);\n const merged = { ...apparent, ...newDims };\n const dims = getApparentDimsInvert(merged);\n newx = dims.x;\n newy = dims.y;\n }\n\n // THIS IS IT!\n // if (Math.abs(newx - 0) < 100) {\n // newx = 0;\n // }\n // And this works!\n // if (Math.abs(apparent.x - 0) < 100) {\n // const merged = { ...apparent, ...{ x: 0 } };\n // const dims = getApparentDimsInvert(merged);\n // newx = dims.x;\n // }\n\n onUpdate({ x: newx, y: newy, snapPairs: possibleSnapPairs });\n };\n}\n","export function isUndefined(val) {\n return typeof val === 'undefined';\n}","/**\n * Calculate a translate matrix\n * @param tx Translation on axis x\n * @param [ty = 0] Translation on axis y\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix\n */\nexport function translate(tx) {\n var ty = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 0;\n\n return {\n a: 1,\n c: 0,\n e: tx,\n b: 0,\n d: 1,\n f: ty\n };\n}","function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }\n\nfunction _toArray(arr) { return Array.isArray(arr) ? arr : Array.from(arr); }\n\n/**\n * Merge multiple matrices into one\n * @param matrices {...object} list of matrices\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix\n */\nexport function transform() {\n for (var _len = arguments.length, matrices = Array(_len), _key = 0; _key < _len; _key++) {\n matrices[_key] = arguments[_key];\n }\n\n matrices = Array.isArray(matrices[0]) ? matrices[0] : matrices;\n\n var multiply = function multiply(m1, m2) {\n return {\n a: m1.a * m2.a + m1.c * m2.b,\n c: m1.a * m2.c + m1.c * m2.d,\n e: m1.a * m2.e + m1.c * m2.f + m1.e,\n b: m1.b * m2.a + m1.d * m2.b,\n d: m1.b * m2.c + m1.d * m2.d,\n f: m1.b * m2.e + m1.d * m2.f + m1.f\n };\n };\n\n switch (matrices.length) {\n case 0:\n throw new Error('no matrices provided');\n\n case 1:\n return matrices[0];\n\n case 2:\n return multiply(matrices[0], matrices[1]);\n\n default:\n var _matrices = matrices,\n _matrices2 = _toArray(_matrices),\n m1 = _matrices2[0],\n m2 = _matrices2[1],\n rest = _matrices2.slice(2);\n\n var m = multiply(m1, m2);\n return transform.apply(undefined, [m].concat(_toConsumableArray(rest)));\n }\n}\n\n/**\n * Merge multiple matrices into one (alias of `transform`)\n * @param matrices {...object} list of matrices\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix\n */\nexport function compose() {\n return transform.apply(undefined, arguments);\n}","import { isUndefined } from './utils';\nimport { translate } from './translate';\nimport { transform } from './transform';\n\nvar cos = Math.cos,\n sin = Math.sin,\n PI = Math.PI;\n/**\n * Calculate a rotation matrix\n * @param angle Angle in radians\n * @param [cx] If (cx,cy) are supplied the rotate is about this point\n * @param [cy] If (cx,cy) are supplied the rotate is about this point\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix *\n */\n\nexport function rotate(angle, cx, cy) {\n var cosAngle = cos(angle);\n var sinAngle = sin(angle);\n var rotationMatrix = {\n a: cosAngle,\n c: -sinAngle,\n e: 0,\n b: sinAngle,\n d: cosAngle,\n f: 0\n };\n if (isUndefined(cx) || isUndefined(cy)) {\n return rotationMatrix;\n }\n\n return transform([translate(cx, cy), rotationMatrix, translate(-cx, -cy)]);\n}\n\n/**\n * Calculate a rotation matrix with a DEG angle\n * @param angle Angle in degree\n * @param [cx] If (cx,cy) are supplied the rotate is about this point\n * @param [cy] If (cx,cy) are supplied the rotate is about this point\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix\n */\nexport function rotateDEG(angle) {\n var cx = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : undefined;\n var cy = arguments.length > 2 && arguments[2] !== undefined ? arguments[2] : undefined;\n\n return rotate(angle * PI / 180, cx, cy);\n}","import { isUndefined } from './utils';\n\n/**\n * Calculate a scaling matrix\n * @param sx Scaling on axis x\n * @param [sy = sx] Scaling on axis y (default sx)\n * @returns {{a: number, b: number, c: number, e: number, d: number, f: number}} Affine matrix\n */\nexport function scale(sx) {\n var sy = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : undefined;\n\n if (isUndefined(sy)) sy = sx;\n return {\n a: sx,\n c: 0,\n e: 0,\n b: 0,\n d: sy,\n f: 0\n };\n}","/**\n * Serialize the matrix to a string that can be used with CSS or SVG\n * @param matrix Affine matrix\n * @returns {string} String that contains a matrix formatted as matrix(a,b,c,d,e,f)\n */\nexport function toCSS(matrix) {\n return toString(matrix);\n}\n\n/**\n * Serialize the matrix to a string that can be used with CSS or SVG\n * @param matrix Affine matrix\n * @returns {string} String that contains a matrix formatted as matrix(a,b,c,d,e,f)\n */\nexport function toSVG(matrix) {\n return toString(matrix);\n}\n\n/**\n * Serialize the matrix to a string that can be used with CSS or SVG\n * @param matrix Affine matrix\n * @returns {string} String that contains a matrix formatted as matrix(a,b,c,d,e,f)\n */\nexport function toString(matrix) {\n return \"matrix(\" + matrix.a + \",\" + matrix.b + \",\" + matrix.c + \",\" + matrix.d + \",\" + matrix.e + \",\" + matrix.f + \")\";\n}","\"use strict\";\n\nimport * as _transformationMatrix from \"transformation-matrix\";\n\n//https://stackoverflow.com/questions/15762768/javascript-math-round-to-two-decimal-places\nvar roundTo = function roundTo(n) {\n var digits =\n arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 2;\n\n var multiplicator = Math.pow(10, digits);\n n = parseFloat((n * multiplicator).toFixed(11));\n var test = Math.round(n) / multiplicator;\n return +test.toFixed(2);\n};\n\nexport default function (_ref) {\n var x = _ref.x,\n y = _ref.y,\n angle = _ref.angle,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n width = _ref.width,\n height = _ref.height,\n _ref$disableScale = _ref.disableScale,\n disableScale = _ref$disableScale === undefined ? false : _ref$disableScale;\n\n var changedWidth = width * (1 - scaleX);\n var newWidth = width - changedWidth;\n var changedHeight = height * (1 - scaleY);\n var newHeight = height - changedHeight;\n\n var transformMatrix = void 0;\n\n if (disableScale === false) {\n transformMatrix = (0, _transformationMatrix.transform)(\n (0, _transformationMatrix.translate)(\n roundTo(x + changedWidth / 2),\n roundTo(y + changedHeight / 2)\n ),\n (0, _transformationMatrix.rotate)(angle * (Math.PI / 180)),\n (0, _transformationMatrix.scale)(scaleX, scaleY)\n );\n } else {\n transformMatrix = (0, _transformationMatrix.transform)(\n (0, _transformationMatrix.translate)(\n roundTo(x + changedWidth),\n roundTo(y + changedHeight)\n ),\n (0, _transformationMatrix.rotate)(angle * (Math.PI / 180))\n );\n width = newWidth;\n height = newHeight;\n }\n\n return {\n element: {\n width: width,\n height: height,\n transform: (0, _transformationMatrix.toCSS)(transformMatrix),\n position: \"absolute\",\n },\n controls: {\n width: newWidth,\n height: newHeight,\n transform: (0, _transformationMatrix.toCSS)(\n (0, _transformationMatrix.transform)(\n (0, _transformationMatrix.translate)(\n roundTo(x + changedWidth),\n roundTo(y + changedHeight)\n ),\n (0, _transformationMatrix.rotate)(angle * (Math.PI / 180))\n )\n ),\n position: \"absolute\",\n },\n };\n}\n","// Modules\nimport Vue from \"vue\";\nimport { AppState, makeInitialAppState } from \"@/stores/AppState\";\nimport { ConditionsData, RendererData } from \"@/types/rendererData\";\nimport { DataBinding, NodeData, NodeSetData } from \"@/types/data\";\nimport { Size, ResolvedWidgetData } from \"@/types\";\nimport { getScaleFactor } from \"@/utils\";\nimport { Widget } from \"@/components/widgets/Widget\";\nimport { AppRendererOptions } from \"./rendererOptions\";\n\nexport interface RendererState extends AppState {\n renderId: string | null;\n\n renderScale: number;\n renderWidth: number;\n renderHeight: number;\n\n /**\n * Whether the app has a valid license associated with it.\n */\n isLicensed: boolean;\n\n /**\n * Whether the app is published.\n */\n isPublished: boolean;\n\n /**\n * Whether the app is a bundle\n */\n isBundled: boolean;\n\n /**\n * Whether the app is being snapshotted.\n */\n isSnapshot: boolean;\n\n bundledAssetUrls?: Record;\n conditions?: ConditionsData;\n\n dataBindings: DataBinding[];\n widgetData: Record>;\n dataUrl: string;\n\n showIntroImage: boolean;\n showFallbackImage: boolean;\n}\n\nexport const clearState = (state: RendererState) => {\n state.renderId = null;\n state.isLicensed = false;\n state.isPublished = false;\n state.isBundled = false;\n state.isSnapshot = false;\n\n state.dataBindings = [];\n state.widgetData = {};\n state.dataUrl = \"\";\n state.conditions = undefined;\n\n state.showIntroImage = false;\n state.showFallbackImage = false;\n};\n\nexport const makeInitialState = () => {\n const state = makeInitialAppState() as RendererState;\n clearState(state);\n return Vue.observable(state);\n};\n\nconst state = makeInitialState();\n\nexport { state as renderState };\n\nexport const adjustWidgetOffset = (\n widget: Widget,\n scaleFactor: number,\n appSize: Size,\n renderSize: Size\n) => {\n const renderAspect = renderSize.w / renderSize.h;\n const appAspect = appSize.w / appSize.h;\n const xOffset =\n renderAspect > appAspect ? (renderSize.w - appSize.w * scaleFactor) / 2 : 0;\n const yOffset =\n renderAspect > appAspect ? 0 : (renderSize.h - appSize.h * scaleFactor) / 2;\n\n return Object.assign(widget, {\n x: widget.x + xOffset,\n y: widget.y + yOffset,\n });\n};\n\n/**\n * When an app is bundled, the bundle will include all the images\n * for the app in the bundle. The bundler will also include an additional\n * property into the app data model called `bundled`.\n * {\n * bundled: {\n * urls: {\n * // The 'key' is the original asset url\n * \"http://kitchen-qa.screenfeed.com/app/asset/*******.jpg\":\n * // The value is the file path to the bundled url\n * \"bundled_files/*******.jpg\"\n * }\n * }\n * }\n * This provides an easy-to-use lookup table we can use at runtime\n * to replace remote urls with local, bundled urls.\n *\n * This function simply scans through the `WidgetData` structure\n * and attempts to replace any remote url with a local bundled url\n * if it exists in the lookup.\n */\nexport const replaceBundledUrls = (\n widgetData: ResolvedWidgetData,\n urls: Record | undefined\n) => {\n if (typeof urls !== \"undefined\" && Object.keys(urls).length >= 0) {\n for (const widgetId in widgetData) {\n widgetData[widgetId].forEach((record) => {\n for (const prop in record) {\n const currentUrl = record[prop];\n if (prop === \"url\" && urls[currentUrl]) {\n record[prop] = urls[currentUrl];\n }\n }\n });\n }\n }\n return widgetData;\n};\n\nexport const replaceState = (payload: {\n rendererData: RendererData;\n renderId: string;\n rendererOptions: AppRendererOptions;\n}) => {\n const { rendererData, renderId, rendererOptions } = payload;\n\n if (!rendererData) return;\n\n state.renderId = renderId;\n state.assets = rendererData.assets;\n state.height = rendererData.height;\n state.width = rendererData.width;\n state.name = rendererData.name;\n state.uuid = rendererData.uuid;\n state.widgets = rendererData.model.widgets;\n state.custom = rendererData.model.custom;\n state.parents = rendererData.model.parents;\n state.appVersionUuid = rendererData.currentVersion;\n state.ianaTimeZone = rendererData.ianaTimeZone;\n\n state.fallbackImageTimeoutSec = rendererData.fallbackImageTimeoutSec;\n state.introImageTimeoutSec = rendererData.introImageTimeoutSec;\n\n state.isPublished = rendererData.isPublished;\n state.isLicensed = rendererData.isLicensed;\n state.dataBindings = rendererData.dataBindings;\n state.widgetData = rendererData.widgetData;\n state.isBundled = typeof rendererData.bundled === \"object\";\n state.bundledAssetUrls = rendererData.bundled?.urls;\n state.conditions = rendererData.conditions;\n\n state.isSnapshot = rendererOptions.snapshot;\n};\n\nexport const updateRendererConditionsData = (payload: RendererData) => {\n const { conditions, dataBindings, widgetData, assets } = payload;\n Vue.set(state, \"conditions\", conditions);\n Vue.set(state, \"dataBindings\", dataBindings);\n Vue.set(state, \"widgetData\", widgetData);\n Vue.set(state, \"assets\", assets);\n};\n\nexport const updateRendererData = (payload: RendererData) => {\n const { widgetData } = payload;\n\n for (const widgetId in widgetData) {\n Vue.set(state.widgetData, widgetId, widgetData[widgetId]);\n }\n};\n\nexport const updateRenderSize = (payload: Size) => {\n state.renderHeight = payload.h;\n state.renderWidth = payload.w;\n\n const appSize = { w: state.width, h: state.height };\n const renderSize = payload;\n state.renderScale = getScaleFactor(renderSize, appSize);\n};\n\nexport const updateLicenseStatus = (payload: { isLicensed: boolean }) => {\n state.isLicensed = payload.isLicensed;\n};\n","import i18n from \"@/i18n\";\nimport { gsap } from \"gsap\";\nimport {\n AnimationOptions,\n AnimationPresetList,\n AnimationTween,\n Animations,\n} from \"@/types/animation\";\nimport { getApparentDims } from \"@/utils\";\nimport { TransformOptions } from \"./TransformOptions\";\nimport { styler } from \"@/lib/free-transform/lib\";\nimport { renderState } from \"@/rendererState\";\n\nexport const Eases = [\n { label: \"None\", value: \"none\" },\n {\n label: `${i18n.tc(\"animation.easeNames.slow\")}`,\n value: \"power1.inOut\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.slowAccelerate\")}`,\n value: \"power1.in\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.slowDecelerate\")}`,\n value: \"power1.out\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.natural\")}`,\n value: \"power3.inOut\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.accelerate\")}`,\n value: \"power3.in\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.decelerate\")}`,\n value: \"power3.out\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.quick\")}`,\n value: \"expo.inOut\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.quickAccelerate\")}`,\n value: \"expo.in\",\n },\n {\n label: `${i18n.tc(\"animation.easeNames.quickDecelerate\")}`,\n value: \"expo.out\",\n },\n];\n\nexport const AnimatableWidgetTypes = [\n \"Image\",\n \"Text\",\n \"Group\",\n \"Rectangle\",\n \"Ellipse\",\n \"Svg\",\n \"Triangle\",\n \"Line\",\n];\n\nexport const AppTimeline = gsap.timeline();\n\nexport function getAnimationId(widgetId: string, cellIndex?: number) {\n const suffix = typeof cellIndex === \"undefined\" ? \"\" : `-${cellIndex}`;\n return `data-anim-${widgetId}${suffix}`;\n}\n\nexport function resetAnimation(timeline: gsap.core.Timeline) {\n timeline.revert();\n timeline.clear();\n timeline.pause();\n}\n\nfunction convertTweenValue(\n x: number,\n y: number,\n attribute: string,\n value: number\n) {\n if (attribute === \"opacity\" || attribute === \"scale\") {\n return value / 100;\n } else if (attribute === \"y\") {\n return y + value;\n } else if (attribute === \"x\") {\n return x + value;\n } else {\n return value;\n }\n}\n\nfunction pauseTimeline(timeline: gsap.core.Timeline) {\n console.log(\"pausing timeline\");\n timeline.pause();\n}\n\nfunction outroEnd(\n end: number | undefined,\n duration: number,\n cycleDuration: number | undefined\n) {\n if (cycleDuration != null) {\n return cycleDuration - duration;\n } else {\n return end ? end - duration : duration;\n }\n}\n\nexport type AnimationInstructions = {\n selector: string;\n animations: Animations;\n cycleDuration: number | undefined;\n x: number;\n y: number;\n opacity: number;\n transform: string;\n};\n\nfunction gatherAnimationInstructions(vnode: Vue) {\n const result: AnimationInstructions[] = [];\n\n function traverse(vnode: Vue) {\n if (vnode.$props && vnode.$props[\"animations\"]) {\n const animations = vnode.$props[\"animations\"] as Animations;\n const hasAnimations =\n animations.loop.presetId !== \"none\" ||\n animations.transition.presetId !== \"none\";\n\n if (hasAnimations) {\n const apparentDims = getApparentDims(vnode.$props as TransformOptions);\n const { element } = styler({\n x: vnode.$props.x,\n y: vnode.$props.y,\n scaleX: vnode.$props.scaleX,\n scaleY: vnode.$props.scaleY,\n width: vnode.$props.w || 0,\n height: vnode.$props.h || 0,\n angle: vnode.$props.angle,\n disableScale: true,\n });\n\n result.push({\n selector: getAnimationId(\n vnode.$props[\"wid\"],\n vnode.$props[\"cellIndex\"]\n ),\n animations: vnode.$props[\"animations\"],\n cycleDuration: vnode.$props[\"cycleDuration\"],\n x: apparentDims.x,\n y: apparentDims.y,\n opacity: vnode.$props[\"opacity\"] / 100,\n transform: element.transform,\n });\n }\n }\n const children = Array.from(vnode.$children);\n for (const child of children) {\n traverse(child);\n }\n }\n\n traverse(vnode);\n return result;\n}\n\nexport function animate(root: Vue) {\n // Do not animate during screenshot service's process:\n if (renderState.isSnapshot === true) return;\n gsap.config({\n nullTargetWarn: false,\n });\n const instructions = gatherAnimationInstructions(root);\n instructions.forEach((i) => {\n const widgetTimeline = gsap.timeline();\n const loopTimeline = gsap.timeline();\n const inOutTimeline = gsap.timeline();\n const element = document.getElementById(i.selector);\n if (element == null) {\n console.warn(`Could not find element with selector ${element}`);\n return;\n }\n const animations = i.animations;\n const cycleDuration = i.cycleDuration;\n\n gsap.set(element, {\n x: i.x,\n y: i.y,\n opacity: i.opacity,\n transform: i.transform,\n });\n\n for (const key in animations) {\n const animation = animations[key as keyof typeof animations];\n if (animation.tweens && animation.tweens.length > 0) {\n animation.tweens?.forEach((tween: AnimationTween) => {\n if (element && animation.presetId != \"none\") {\n element.style.willChange = \"transform, scale, opacity\";\n if (key === \"transition\") {\n inOutTimeline.from(\n element,\n {\n [`${tween.attribute}`]: convertTweenValue(\n i.x,\n i.y,\n tween.attribute,\n tween.value\n ),\n force3D: false,\n duration: animation.duration,\n ease: animation.ease,\n },\n animation.delay\n );\n if (animation.outro) {\n inOutTimeline.to(\n element,\n {\n [`${tween.attribute}`]: convertTweenValue(\n i.x,\n i.y,\n tween.attribute,\n animation.reverse ? -tween.value : tween.value\n ),\n duration: animation.duration,\n ease: animation.ease,\n force3D: false,\n onStart: pauseTimeline,\n onStartParams: [loopTimeline],\n },\n outroEnd(animation.end, animation.duration, cycleDuration)\n );\n }\n } else if (key === \"loop\") {\n loopTimeline.to(\n element,\n {\n [`${tween.attribute}`]: convertTweenValue(\n i.x,\n i.y,\n tween.attribute,\n tween.value\n ),\n duration: animation.duration,\n ease: animation.ease,\n force3D: false,\n repeat: tween.repeat,\n repeatDelay: tween.repeatDelay,\n yoyo: true,\n },\n animation.delay +\n animations.transition.delay +\n animations.transition.duration\n );\n }\n }\n });\n }\n }\n widgetTimeline.add(inOutTimeline, 0);\n widgetTimeline.add(loopTimeline, 0);\n AppTimeline.add(widgetTimeline, 0);\n });\n\n AppTimeline.eventCallback(\"onComplete\", () => {\n instructions.forEach((i) => {\n const element = document.getElementById(i.selector) as HTMLElement;\n if (!element) return;\n element.style.willChange = \"auto\";\n });\n });\n AppTimeline.play(0);\n}\n\nexport function getDefaultAnimationOptions(): AnimationOptions {\n return {\n animations: {\n transition: {\n presetId: \"none\",\n label: i18n.tc(\"animation.presetNames.none\"),\n duration: 0,\n delay: 0,\n ease: \"none\",\n tweens: [\n {\n attribute: \"opacity\",\n value: 100,\n },\n ],\n },\n loop: {\n presetId: \"none\",\n label: i18n.tc(\"animation.presetNames.none\"),\n duration: 0,\n delay: 0,\n ease: \"none\",\n tweens: [\n {\n attribute: \"opacity\",\n value: 100,\n },\n ],\n },\n },\n };\n}\n\nexport const AnimationPresets: AnimationPresetList = {\n transition: [\n {\n presetId: \"none\",\n label: i18n.tc(\"animation.presetNames.none\"),\n duration: 0,\n delay: 0,\n ease: \"none\",\n tweens: [\n {\n attribute: \"opacity\",\n value: 100,\n },\n ],\n },\n {\n presetId: \"fade\",\n label: i18n.tc(\"animation.presetNames.fade\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n tweens: [\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"grow\",\n label: i18n.tc(\"animation.presetNames.grow\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n tweens: [\n {\n attribute: \"scale\",\n value: 0,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"shrink\",\n label: i18n.tc(\"animation.presetNames.shrink\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n tweens: [\n {\n attribute: \"scale\",\n value: 200,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"slideUp\",\n label: i18n.tc(\"animation.presetNames.slideUp\"),\n duration: 2.0,\n delay: 0,\n outro: false,\n end: 10,\n canReverse: true,\n reverse: false,\n ease: \"power3.out\",\n tweens: [\n {\n attribute: \"y\",\n value: 500,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"slideDown\",\n label: i18n.tc(\"animation.presetNames.slideDown\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n canReverse: true,\n reverse: false,\n tweens: [\n {\n attribute: \"y\",\n value: -500,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"slideLeft\",\n label: i18n.tc(\"animation.presetNames.slideLeft\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n canReverse: true,\n reverse: false,\n tweens: [\n {\n attribute: \"x\",\n value: 500,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n {\n presetId: \"slideRight\",\n label: i18n.tc(\"animation.presetNames.slideRight\"),\n duration: 2.0,\n delay: 0,\n ease: \"power3.out\",\n outro: false,\n end: 10,\n canReverse: true,\n reverse: false,\n tweens: [\n {\n attribute: \"x\",\n value: -500,\n },\n {\n attribute: \"opacity\",\n value: 0,\n },\n ],\n },\n ],\n loop: [\n {\n presetId: \"none\",\n label: i18n.tc(\"animation.presetNames.none\"),\n duration: 0,\n delay: 0,\n ease: \"none\",\n tweens: [\n {\n attribute: \"opacity\",\n value: 100,\n },\n ],\n },\n {\n presetId: \"blink\",\n label: i18n.tc(\"animation.presetNames.blink\"),\n duration: 1.0,\n delay: 0,\n ease: \"power3.inOut\",\n tweens: [\n {\n attribute: \"opacity\",\n repeat: -1,\n repeatDelay: 1,\n stagger: 0,\n value: 0,\n yoyo: true,\n },\n ],\n },\n {\n presetId: \"pulse\",\n label: i18n.tc(\"animation.presetNames.pulse\"),\n duration: 2.0,\n delay: 0,\n ease: \"expo.inOut\",\n tweens: [\n {\n attribute: \"scale\",\n repeat: -1,\n repeatDelay: 0,\n stagger: 0,\n value: 120,\n yoyo: true,\n },\n ],\n },\n ],\n};\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface GroupOptions\n extends ComponentOptions,\n TransformOptions,\n AnimationOptions {\n verticalDynamism: boolean;\n verticalMargin: number;\n orderedChildIds: string[];\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...getDefaultAnimationOptions(),\n verticalDynamism: false,\n verticalMargin: 20,\n orderedChildIds: [],\n};\n\nexport const CreateGroup = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Group\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Group,\n });\n};\n","export const isObjectLiteral = (obj: any): boolean => {\n let o = obj;\n return typeof obj !== \"object\" || obj === null\n ? false\n : (function () {\n while (Object.getPrototypeOf((o = Object.getPrototypeOf(o))) !== null);\n return Object.getPrototypeOf(obj) === o;\n })();\n};\n","import { isObjectLiteral } from \"./isObjectLiteral\";\n\n/**\n *\n * @param value Any value you want to check\n */\nexport const isEmpty = (value: any): boolean => {\n if (typeof value === \"undefined\") {\n return true;\n }\n\n if (value === null) {\n return true;\n }\n\n if (\n typeof value === \"string\" &&\n (value.length === 0 || (value.length > 0 && value.trim() === \"\"))\n ) {\n return true;\n }\n\n if (isObjectLiteral(value)) {\n return Object.keys(value).length === 0;\n }\n\n if (Array.isArray(value)) {\n return value.every(isEmpty);\n }\n\n return false;\n};\n","import { isEmpty } from \"@core/utils/isEmpty\";\nimport { BASE_PARENT_ID } from \"./constants\";\nimport { AppModel } from \"./types/data\";\n// import { TextOptions } from \"./components/widgets/Text/TextOptions\";\n// import { makeTextContent } from \"./text\";\n\nconst cleanParents = (app: AppModel) => {\n /**\n * A bug in our code used to allow duplicate widget ids to\n * be added to the state.parents Record.\n *\n * Also, it seems widgets that have been deleted still remain\n * in the state.parents Record as well.\n *\n * This bug has been fixed, but some apps were persisted with\n * this corrupt data and so now we always need to clean it.\n *\n * We don't want to do it in a migration, because migrations\n * are about changing the app model.\n */\n if (typeof app.parents !== \"object\") {\n app.parents = {};\n }\n\n for (const parentId in app.parents) {\n const childIds = app.parents[parentId] ?? [];\n app.parents[parentId] = [...new Set(childIds)].filter(\n (wid) => typeof app.widgets[wid] !== \"undefined\"\n );\n }\n};\n\nconst cleanWidgets = (app: AppModel) => {\n for (const wid in app.widgets) {\n const widget = app.widgets[wid];\n\n // If a widget doesn't have a parentId, set it to default\n if (isEmpty(widget.parentId)) {\n widget.parentId = BASE_PARENT_ID;\n }\n\n // There was a bug with graphs, where every time the widget was selected,\n // the customBarColors property had its contents multiplied, resulting in\n // an enormous array and a performance hit. The bug has been fixed, but\n // this fix is needed to address any existing graphs that are affected\n if (widget.type === \"BarGraph\" || widget.type === \"ColumnGraph\") {\n Object.values(widget.conditionalVersions).forEach((widgetVersion) => {\n //Arbitrary value of 50 chosen. It is higher than the number of items anyone\n //would ever display in a graph, and low enough to not impact performance.\n //Arbitrary value used so this file does not need to be rewritten to use AppPayload.\n //The value doesn't really matter, it will be reset by the widget on load, this is to\n //ensure that it does load and does not time out.\n if (widgetVersion.customBarColors.length > 50) {\n widgetVersion.customBarColors = widgetVersion.customBarColors.slice(\n 0,\n 50\n );\n }\n });\n }\n\n if (widget.type === \"PieGraph\" || widget.type === \"StackedGraph\") {\n Object.values(widget.conditionalVersions).forEach((widgetVersion) => {\n if (widgetVersion.colors.length > 50) {\n widgetVersion.colors = widgetVersion.colors.slice(0, 50);\n }\n });\n }\n\n if (widget.type === \"CalendarRoomSchedule\") {\n Object.values(widget.conditionalVersions).forEach((widgetVersion) => {\n if (\n !(\n \"clockTime_fontSize\" in widgetVersion &&\n \"clockTime_fontFamily\" in widgetVersion\n )\n ) {\n widgetVersion.clockTime_fontSize = 100;\n widgetVersion.clockTime_fontWeight = 400;\n widgetVersion.clockTime_textColor = \"#000000\";\n widgetVersion.clockTime_fontFamily = \"Montserrat\";\n widgetVersion.clockTime_textTransform = \"none\";\n widgetVersion.clockTime_textDecoration = \"none\";\n widgetVersion.clockTime_fontStyle = \"normal\";\n widgetVersion.clockTime_lineHeight = 1.2;\n widgetVersion.clockTime_textAlign = \"left\";\n }\n });\n }\n\n // Don't think this is needed any longer\n // if (widget.type === \"Text\") {\n // const textWidget = widget as TextOptions;\n // if (typeof textWidget.content === \"string\") {\n // // Old text content could contain HTML, let's gather textContent\n // const oldHtml = `
${w.content}
`;\n // const doc = new DOMParser().parseFromString(oldHtml, \"text/html\");\n // const textContent = doc.body.textContent || \"\";\n // textWidget.content = makeTextContent(textContent);\n // }\n // }\n }\n};\n\n/**\n * Replaces known system fonts with similar Google Fonts.\n *\n * We used to allow System fonts, but no longer because...\n * - Sometimes system fonts would render differently on different machines\n * - Google Font loader would fail to load system fonts.\n */\nconst cleanFonts = (app: AppModel) => {\n const widgets = app.widgets;\n Object.entries(widgets).forEach(([, widget]) => {\n const keysWithInvalidBoldFont: string[] = [];\n Object.entries(widget).forEach(([key, value]) => {\n if (key.endsWith(\"fontFamily\")) {\n switch ((value as string).toLowerCase()) {\n case \"monospace\":\n (widget as any)[key] = \"Courier Prime\";\n break;\n case \"serif\":\n (widget as any)[key] = \"Crimson Text\";\n break;\n case \"sans serif\":\n case \"sans-serif\":\n (widget as any)[key] = \"Inter\";\n break;\n case \"abril fatface\":\n case \"lobster\":\n keysWithInvalidBoldFont.push(key);\n break;\n }\n }\n });\n\n // We allowed users to choose non-existent font weights for\n // \"Abril Fatface\" and \"Lobster\" which could effect snapshot timing.\n // This will reset to weight 400.\n keysWithInvalidBoldFont.forEach((key) => {\n const fontWeightKey = key.replace(\"fontFamily\", \"fontWeight\");\n const fontWeightValue = (widget as any)[fontWeightKey];\n if (![\"regular\" || \"400\" || 400].includes(fontWeightValue)) {\n (widget as any)[fontWeightKey] = 400;\n }\n });\n });\n};\n\n/**\n * Cleans the app model of any corrupted data.\n */\nexport const cleanAppModel = (app: AppModel) => {\n const methods = [cleanParents, cleanWidgets, cleanFonts];\n\n methods.forEach((method) => method(app));\n};\n","import { AppPayload } from \"./types/data\";\nimport { RepeaterOptions } from \"./components/widgets/Repeater/RepeaterOptions\";\nimport { RendererData } from \"./types/rendererData\";\nimport { AppMode } from \"@/types\";\nimport {\n getDefaultAnimationOptions,\n AnimatableWidgetTypes,\n} from \"./components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { cleanAppModel } from \"./cleanModel\";\nimport { Widget } from \"./components/widgets/Widget\";\nimport {\n addConditionalVersions,\n isConditionalBindingType,\n} from \"./util/conditionsUtils\";\nimport { DEFAULT_CONDITION_ID } from \"./constants\";\nimport { logger } from \"@core/logger\";\nimport { PieGraphOptions } from \"./components/widgets/PieGraph/PieGraphOptions\";\nimport { StackedGraphOptions } from \"./components/widgets/StackedGraph/StackedGraphOptions\";\n\ntype AppMigration = {\n version: number;\n migrate: (app: AppPayload | RendererData, mode: AppMode) => void;\n};\n\nexport const migrations: AppMigration[] = [\n {\n version: 7,\n migrate: function (app: AppPayload | RendererData) {\n for (const widgetId in app.model.widgets) {\n const widget = app.model.widgets[widgetId];\n\n for (const cid in widget.conditionalVersions) {\n const props = widget.conditionalVersions[cid] as\n | StackedGraphOptions\n | PieGraphOptions;\n if (\n (widget.type === \"PieGraph\" || widget.type === \"StackedGraph\") &&\n typeof props.colors === \"string\"\n ) {\n props.colors = (props.colors as string).split(\"|\");\n }\n }\n }\n },\n },\n {\n version: 6,\n migrate: function (app: AppPayload | RendererData) {\n for (const wid in app.model.widgets) {\n /**\n * Convert each `Widget` into a `WidgetWithCondtions`\n */\n addConditionalVersions(app.model.widgets[wid] as unknown as Widget);\n\n /**\n * Set each `Asset` and `Scalar` and `DataNodeSet` binding to use the default condition id.\n */\n app.dataBindings.forEach((db) => {\n db.conditionUuid =\n db.widgetId === wid && isConditionalBindingType(db.bindingType)\n ? DEFAULT_CONDITION_ID\n : undefined;\n });\n }\n },\n },\n {\n version: 5,\n migrate: function (app: AppPayload | RendererData) {\n const widgets = Object.values(app.model.widgets).filter((widget) =>\n AnimatableWidgetTypes.includes(widget.type)\n );\n Object.values(widgets).forEach((widget) => {\n if (!(\"animations\" in widget)) {\n (widget as unknown as AnimationOptions).animations =\n getDefaultAnimationOptions().animations;\n return;\n }\n });\n },\n },\n {\n version: 4,\n migrate: function () {\n /**\n * This used to be a migration to cleanup font problems\n * and is now covered by cleanAppModel.cleanFonts()\n *\n * Migrations should only exist to apply schema changes\n * to the app model. They should not be used to fix data.\n */\n return;\n },\n },\n\n {\n version: 3,\n migrate: function (app: AppPayload | RendererData) {\n const widgets = app.model.widgets;\n Object.entries(widgets).forEach((widget) => {\n if (widget[1].type === \"Slide\") {\n widget[1].type = \"Repeater\";\n (widget[1] as unknown as RepeaterOptions).columns = 1;\n (widget[1] as unknown as RepeaterOptions).rows = 1;\n (widget[1] as unknown as RepeaterOptions).rowGap = 20;\n (widget[1] as unknown as RepeaterOptions).columnGap = 20;\n (widget[1] as unknown as RepeaterOptions).flow = \"row\";\n }\n if (\n widget[1].type === \"Repeater\" &&\n (widget[1] as unknown as RepeaterOptions).flow === undefined\n ) {\n (widget[1] as unknown as RepeaterOptions).flow = \"row\";\n }\n });\n },\n },\n {\n version: 2,\n migrate: function (app: AppPayload | RendererData) {\n if (app.dataBindings) {\n app.dataBindings.forEach((e) => {\n if (e.bindingType === \"DataSetParent\") {\n if (!e.parentWidgetId) {\n e.parentWidgetId = app.model.widgets[e.widgetId].parentId;\n }\n }\n if (\n e.bindingType === \"DataSet\" &&\n e.property === \"data\" &&\n app.model.widgets[e.widgetId]?.type === \"Repeater\"\n ) {\n e.bindingType = \"DataSetParent\";\n }\n });\n\n /**\n * This kind of data binding gets created accidentally somehow, and should be removed.\n * They give rise to \"Scalar bindings must have a query defined\" internal server error.\n */\n app.dataBindings = app.dataBindings.filter(\n (e) =>\n !(\n e.bindingType === \"Scalar\" &&\n e.query === null &&\n e.dataName === \"NA\"\n )\n );\n }\n },\n },\n {\n version: 1,\n migrate: function () {\n //Base version\n },\n },\n];\n\nexport const appSchemaVersion = Math.max(...migrations.map((m) => m.version));\n\nexport function applyModelMigrations(\n app: AppPayload | RendererData,\n mode: AppMode\n) {\n const currentVersion = parseInt(\n app.model.appSchemaVersion?.toString() ?? \"1\",\n 10\n );\n\n if (currentVersion > appSchemaVersion) {\n logger.track(\n `App schema version ${currentVersion} is greater than the maximum supported version ${appSchemaVersion}.`\n );\n }\n\n if (currentVersion < appSchemaVersion) {\n console.info(\n `Schema version is ${currentVersion}. But it should be ${appSchemaVersion}`\n );\n }\n\n if (currentVersion < appSchemaVersion) {\n for (let i = currentVersion + 1; i <= appSchemaVersion; i++) {\n migrations.find(({ version }) => version === i)?.migrate(app, mode);\n }\n app.model.appSchemaVersion = appSchemaVersion;\n }\n\n cleanAppModel(app.model);\n}\n","import Vue from \"vue\";\nimport { defineStore } from \"pinia\";\nimport {\n NodeData,\n NodeSetData,\n DataSortDirection,\n DataBinding,\n} from \"@/types/data\";\n\nimport { api } from \"@/api/backend\";\nimport { removeReactivity } from \"@/utils\";\nimport { useAppEditorStore } from \"@/stores/appEditor\";\nimport { EventBus } from \"@/eventbus\";\n\n/**\n * This is a dictionary of all datasets that have been loaded into the app.\n * The outer key is the widgetId.\n */\nexport type WidgetDataCache = Record<\n string,\n Record\n>;\n\nexport interface RefreshWidgetDataOptions {\n widgetId?: string;\n appUuid?: string;\n sortDirection?: DataSortDirection;\n orderByDataUuid?: string;\n conditionUuid?: string;\n}\n\ninterface AppWidgetData {\n widgetData: WidgetDataCache;\n}\n\ninterface AppDataState {\n // See encodeDataCacheKey for info about these keys\n dataCache: Record;\n}\n\nconst CACHE_KEY_SEPARATOR = \",\";\nconst NO_VALUE_INDICATOR = \"\";\n\nconst orderedBindingProperties: string[] = [\n \"widgetId\",\n \"property\",\n \"query\",\n \"dataConnectionUuid\",\n \"orderByDataUuid\",\n \"shouldRandomize\",\n \"sortDirection\",\n \"filterUuid\",\n \"conditionUuid\",\n];\n\nexport const encodeDataCacheKey = (db: DataBinding) => {\n const props = orderedBindingProperties.map(\n (prop) => db[prop as keyof DataBinding]\n );\n\n const cacheKey = props.reduce((acc: string, val) => {\n return acc + (val ?? NO_VALUE_INDICATOR) + CACHE_KEY_SEPARATOR;\n }, \"\");\n\n return cacheKey;\n};\n\nconst decodeDataCacheKey = (key: string) => {\n return Object.fromEntries(\n key\n .split(CACHE_KEY_SEPARATOR)\n .map((value, idx) => [orderedBindingProperties[idx], value])\n );\n};\n\nexport const useAppDataStore = defineStore(\"appData\", {\n state: (): AppDataState => {\n return {\n dataCache: {},\n };\n },\n\n getters: {\n /**\n * Looks at data bindings and the cache of dynamic data to compute the dynamic data for each widget,\n * in the shape that the widgetData getter is expecting.\n */\n data() {\n const result: Record> = {};\n const { dataBindings } = useAppEditorStore();\n Object.entries(this.dataCache).forEach(([cacheKey, dataset]) => {\n const keyParts = decodeDataCacheKey(cacheKey);\n const { widgetId, property } = keyParts;\n\n if (!(widgetId in result)) {\n result[widgetId] = {};\n }\n\n const db = dataBindings.find((db) =>\n Object.entries(keyParts).every(\n ([key, value]) =>\n value === NO_VALUE_INDICATOR ||\n db[key as keyof DataBinding]?.toString() === value\n )\n );\n\n if (db) {\n result[widgetId][property] = dataset;\n }\n });\n\n return result;\n },\n },\n\n actions: {\n async initializeWidgetData(appUuid: string) {\n return api.get(`apps/${appUuid}/data`).then((res) => {\n // console.log(\"Widget data:\", JSON.stringify(res.widgetData));\n Object.keys(res.widgetData).forEach((widgetId) => {\n const record = res.widgetData[widgetId];\n this.updateDataCache(widgetId, record);\n });\n });\n },\n\n async refreshWidgetData(widgetId: string) {\n const appUuid = useAppEditorStore().uuid;\n return api\n .get(`apps/${appUuid}/data?widgetId=${widgetId}`)\n .then((res) => {\n if (widgetId in res.widgetData) {\n this.updateDataCache(widgetId, res.widgetData[widgetId]);\n }\n });\n },\n\n // Children of repeaters have their dynamic data passed through the parent, not via refreshing widget data directly\n isCacheableBinding(binding: DataBinding) {\n return ![\"DataSetNode\", \"Asset\"].includes(binding?.bindingType);\n },\n\n /**\n * Cache the dynamic data corresponding to each property in the dataset.\n */\n updateDataCache(\n widgetId: string,\n dataset: Record\n ) {\n const { dataBindings } = useAppEditorStore();\n\n Object.entries(dataset).forEach(([prop, value]) => {\n const binding = dataBindings\n .filter(this.isCacheableBinding)\n .find((db) => db.widgetId === widgetId && db.property === prop);\n\n if (!binding) return;\n\n // console.log(\"update cache for\", binding, value);\n\n const cacheKey = encodeDataCacheKey(binding);\n\n Vue.set(this.dataCache, cacheKey, removeReactivity(value));\n });\n },\n\n syncDataForBindings() {\n const { dataBindings } = useAppEditorStore();\n const widgetIdsNeedingRefresh: string[] = [];\n dataBindings.forEach((db) => {\n if (\n this.isCacheableBinding(db) &&\n !(encodeDataCacheKey(db) in this.dataCache)\n ) {\n widgetIdsNeedingRefresh.push(db.widgetId);\n }\n });\n\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n return Promise.all(\n widgetIdsNeedingRefresh.map((widgetId) => {\n return this.refreshWidgetData(widgetId);\n })\n ).finally(() => {\n EventBus.emit(\"AWAITING_SERVER\", false);\n });\n },\n\n /**\n * This ensures that we re-fetch data for all widgets in the app using this connection.\n */\n invalidateCacheForConnection(connectionUuid: string) {\n Object.keys(this.dataCache)\n .filter((key) => key.includes(connectionUuid))\n .forEach((key) => {\n Vue.delete(this.dataCache, key);\n });\n\n this.syncDataForBindings();\n },\n },\n});\n","import { defineStore } from \"pinia\";\n\nimport { DataConnection, SchemaType } from \"@/types/data\";\nimport { api } from \"@/api/backend\";\nimport uniqWith from \"lodash.uniqwith\";\nimport { useAppEditorStore } from \"@/stores/appEditor\";\nimport Vue from \"vue\";\n\nconst PAGE_SIZE = 20;\n\nexport interface ConnectionsState {\n connections: DataConnection[];\n isFetchingPage: boolean;\n numberSuccessfulPageFetchs: number;\n}\n\nexport const useConnectionsStore = defineStore(\"connections\", {\n state: (): ConnectionsState => {\n return {\n connections: [],\n isFetchingPage: false,\n numberSuccessfulPageFetchs: 0,\n };\n },\n getters: {\n appConnections(): DataConnection[] {\n const appEditor = useAppEditorStore();\n const dataBindings = appEditor.dataBindings;\n const dataBoundConnectionUuids = new Set(\n dataBindings.map((db) => db.dataConnectionUuid)\n );\n return this.connections.filter((c) =>\n dataBoundConnectionUuids.has(c.uuid)\n );\n },\n\n nonAppConnections(): DataConnection[] {\n const appEditor = useAppEditorStore();\n const dataBindings = appEditor.dataBindings;\n const dataBoundConnectionUuids = new Set(\n dataBindings.map((db) => db.dataConnectionUuid)\n );\n return this.connections.filter(\n (c) => !dataBoundConnectionUuids.has(c.uuid)\n );\n },\n },\n\n actions: {\n fetchPagedConnections() {\n if (this.isFetchingPage) return;\n this.isFetchingPage = true;\n const pageNumber = this.numberSuccessfulPageFetchs + 1;\n\n return api\n .get(\n `dataconnection?expandNodes=true&pageSize=${PAGE_SIZE}&page=${pageNumber}&sortBy=name`\n )\n .then(async (connections) => {\n this.addConnections(connections);\n this.numberSuccessfulPageFetchs++;\n })\n .finally(() => {\n /**\n * Ensure a bit of time for connections to render, stretching out scroll space,\n * so that we don't accidentally trigger new fetch immediately\n */\n setTimeout(() => {\n this.isFetchingPage = false;\n }, 100);\n });\n },\n\n async getConnections(payload: {\n appUuid?: string;\n onlyAppConnections?: boolean;\n onlyCollections?: boolean;\n onlyNonCollections?: boolean;\n includeSchemaTypes?: SchemaType[];\n excludeSchemaTypes?: SchemaType[];\n }) {\n const queryParams = new URLSearchParams();\n if (payload?.appUuid) queryParams.append(\"appUuid\", payload.appUuid);\n\n if (payload?.onlyAppConnections)\n queryParams.append(\n \"onlyAppConnections\",\n payload.onlyAppConnections === true ? \"true\" : \"false\"\n );\n\n if (payload?.onlyCollections)\n queryParams.append(\n \"onlyCollections\",\n payload.onlyCollections === true ? \"true\" : \"false\"\n );\n\n if (payload?.onlyNonCollections)\n queryParams.append(\n \"onlyNonCollections\",\n payload.onlyNonCollections === true ? \"true\" : \"false\"\n );\n\n if (\n payload?.includeSchemaTypes &&\n payload.includeSchemaTypes.length !== 0\n )\n queryParams.append(\n \"includeSchemaTypes\",\n payload.includeSchemaTypes.join(\",\")\n );\n\n if (\n payload?.excludeSchemaTypes &&\n payload.excludeSchemaTypes.length !== 0\n )\n queryParams.append(\n \"excludeSchemaTypes\",\n payload.excludeSchemaTypes.join(\",\")\n );\n\n return api\n .get(`dataconnection?${queryParams.toString()}`)\n .then((res) => {\n this.connections = res;\n return res;\n });\n },\n\n async initializeConnections(appUuid: string) {\n return api\n .get(\n `dataconnection?onlyAppConnections=true&appUuid=${appUuid}&expandNodes=true`\n )\n .then(async (connections) => {\n this.addConnections(connections);\n });\n },\n\n addConnections(connections: DataConnection[]) {\n this.connections = uniqWith(\n this.connections.concat(connections),\n (a, b) => a.uuid === b.uuid\n );\n },\n\n addConnection(connection: DataConnection) {\n const connIndex = this.connections.findIndex(\n (c) => c.uuid === connection.uuid\n );\n connIndex !== -1\n ? Vue.set(this.connections, connIndex, connection)\n : this.connections.push(connection);\n },\n\n removeConnection(connectionUuid: string) {\n const connIndex = this.connections.findIndex(\n (c) => c.uuid === connectionUuid\n );\n if (connIndex !== -1) {\n Vue.delete(this.connections, connIndex);\n }\n },\n },\n});\n","import { DataType } from \"@/types/data\";\n\nexport const getComponentName = (dataType: DataType) => {\n switch (dataType) {\n case \"Time\":\n case \"Date\":\n case \"DateTime\":\n return `DataNodeDateTime`;\n case \"String\":\n case \"Bool\":\n case \"Number\":\n case \"Url\":\n return `DataNodeValue`;\n case \"Color\":\n return `DataNodeColor`;\n case \"ImageUrl\":\n case \"ImageUpload\":\n // case \"Url\":\n return `DataNodeImageUrl`;\n case \"ObjectArray\":\n return `DataNodeEmpty`;\n default:\n return `DataNode${dataType}`;\n }\n};\n","/**\n * Returns a reasonably-random uuid\n * https://stackoverflow.com/a/2117523/5651\n * https://creativecommons.org/licenses/by-sa/4.0/\n */\nexport const uuidv4 = () => {\n return \"10000000-1000-4000-8000-100000000000\".replace(/[018]/g, (c: any) =>\n (\n c ^\n (crypto.getRandomValues(new Uint8Array(1))[0] & (15 >> (c / 4)))\n ).toString(16)\n );\n};\n","import { ManualDataRow } from \"@/types/data\";\nimport {\n NodeData,\n NodeSetData,\n NodeKind,\n ModerationDataNode,\n} from \"@/types/data\";\nimport { uuidv4 } from \"@core/utils/uuid\";\n\nexport function getChildDataNodes(\n children: (NodeData | NodeSetData)[][]\n): NodeData[][] {\n return (\n (children?.filter((ch) =>\n ch.some((c) => c.kind === NodeKind.node)\n ) as NodeData[][]) ?? []\n );\n}\n\nexport function getChildDataNodeSets(\n children: (NodeData | NodeSetData)[][]\n): NodeSetData[][] {\n return (\n (children?.filter((ch) =>\n ch.some((c) => c.kind === NodeKind.nodeSet)\n ) as NodeSetData[][]) ?? []\n );\n}\n\n/**\n * These fns convert placeholderData (for graphs, calendars(?)) to manualData format (schema + rows)\n * Can also be used to convert widgetData to manualData format\n */\n\nexport const parseDataSchema = (data: NodeData[][]): any[] => {\n // console.log(\"parse schema\", data);\n return (\n (data?.[0] ?? [])\n .filter((n) => n.kind === NodeKind.node)\n .map((node) => {\n const { uuid, displayName, kind, dataType } = node;\n\n return {\n uuid,\n kind,\n query: uuid,\n dataType,\n name: displayName,\n columnId: uuid,\n columnUuid: uuid,\n altTypes: [],\n sampleValues: [],\n isSelected: true,\n isArtificial: node.isArtificial ?? false,\n isRequired: true, // not sure about this\n isPrimaryKey: node.isPrimaryKey,\n };\n })\n // Ignore _sf_index:\n .filter((n) => !n.isArtificial)\n );\n};\n\n// const parseDataNode = (node: NodeData): ManualDataNode => {\n// const { dataType, uuid, displayName, value, formattedValue, query } = node;\n// // console.log(\"parse node\", value, formattedValue);\n// return {\n// dataType: dataType,\n// columnId: uuid,\n// columnUuid: uuid,\n// name: displayName,\n// isArtificial: node.isArtificial ?? false,\n// value,\n// formattedValue: formattedValue ?? value,\n// query,\n// };\n// };\n\nexport const parseModeratedRows = (\n data: ModerationDataNode[]\n): ManualDataRow[] => {\n return data.map((row) => {\n return {\n rowUuid: row.refUuid,\n isSelected: row.isSelected,\n columns: row.data\n // .map(parseDataNode)\n // Ignore _sf_index:\n .filter((n) => !n.isArtificial),\n };\n });\n};\n\nexport const parseDataRows = (data: NodeData[][]): ManualDataRow[] => {\n if (!data) return [];\n return data.map((row) => {\n return {\n // The groupUuid i.e. rowUuid is buried in the nodes, so just grab from the first one\n rowUuid: row[0]?.groupUuid || uuidv4(),\n // isSelected: row.isSelected,\n columns: row\n .filter((n) => n.kind === NodeKind.node)\n // .map(parseDataNode)\n // Ignore _sf_index:\n .filter((n) => !n.isArtificial),\n };\n });\n};\n","/**\n * It's possible this should be named \"collectionDataStore\" instead.\n * It deals peripherally with non-editable collection data, via the moderation filter.\n * And the fact that it powers EditDataTable (which is also mis-named, and used for non-editable collections.)\n */\n\nimport Vue from \"vue\";\nimport { defineStore } from \"pinia\";\nimport {\n DataBinding,\n DataConnection,\n DataType,\n ManualDataRow,\n NodeData,\n NodeKind,\n SchemaNode,\n} from \"@/types/data\";\nimport { isCurrency } from \"@/utils\";\nimport { Point } from \"@/types\";\nimport { uuidv4 } from \"@core/utils/uuid\";\nimport { EventBus } from \"@/eventbus\";\nimport { api } from \"@/api/backend\";\nimport { useConnectionDataStore } from \"./connectionData\";\nimport { Widget } from \"@/components/widgets/Widget\";\nimport { useAppDataStore } from \"./appData\";\nimport { useAppEditorStore } from \"./appEditor\";\nimport { useConnectionEditorStore } from \"./connectionEditor\";\n/**\n *\n * @param value Any value you want to check (except object, array)\n */\nexport const isEmpty = (value: any): boolean => {\n if (typeof value === \"undefined\") {\n return true;\n }\n\n if (value === null) {\n return true;\n }\n\n if (\n typeof value === \"string\" &&\n (value.length === 0 || (value.length > 0 && value.trim() === \"\"))\n ) {\n return true;\n }\n\n return false;\n};\n\nexport interface DataTypeInterface {\n name: DataType;\n validate: (val: any) => boolean;\n errorMessage: string;\n getComponent: (val: any, showArea?: boolean) => string;\n}\n\n// TODO: Date and Time types... different inputs?\n// Is that working now?\n\nexport const NodeDataTypes: DataTypeInterface[] = [\n {\n name: \"String\",\n validate: (val: any) => {\n return true; // Accept anything\n },\n errorMessage: \"Text is required.\",\n getComponent: (val: string, showArea = false) => {\n if (val?.length > 50 && showArea) {\n return \"TextAreaInput\";\n }\n return \"TextInput\";\n },\n },\n {\n name: \"Number\",\n /**\n * Not sure it makes sense to check type here\n * because we use a number input, and then have to parseInt the value\n * Just to check that it is indeed a number here\n * So just return true\n * (Assumes that all browsers disallow invalid input for number inputs)\n */\n validate: (val: any) => {\n // return typeof val === \"number\";\n return true;\n },\n errorMessage: \"A number is required.\",\n getComponent: (val: number) => {\n return isCurrency(val?.toString()) ? \"CurrencyInput\" : \"NumberInput\";\n },\n },\n {\n name: \"DateTime\",\n validate: (val: any) => {\n return true;\n },\n errorMessage: \"A datetime is required.\",\n\n getComponent: () => {\n return \"DatetimeInput\";\n },\n },\n {\n name: \"Bool\",\n validate: (val: any) => {\n return true;\n },\n errorMessage: \"A true/false value is required.\",\n getComponent: () => {\n return \"BooleanInput\";\n },\n },\n {\n name: \"ImageUrl\",\n validate: (val: any) => {\n return true;\n },\n errorMessage: \"An image URL is required.\",\n getComponent: () => {\n return \"TextInput\";\n },\n },\n {\n name: \"ImageUpload\",\n validate: (val: any) => {\n return true;\n },\n errorMessage: \"An valid image file is required.\",\n getComponent: () => {\n return \"ImageUploadInput\";\n },\n },\n {\n name: \"Color\",\n validate: (val: any) => {\n return val?.match(/^#[0-9A-F]{6,8}$/i);\n },\n errorMessage: \"A color is required.\",\n getComponent: () => {\n return \"ColorInput\";\n },\n },\n];\n\nexport const validateCell = (schemaNode: SchemaNode, value: any) => {\n const { dataType, isRequired } = schemaNode;\n\n if (isRequired && isEmpty(value)) {\n return false;\n }\n\n // NOTE: Using includes rather than === to handle Date/Time\n return NodeDataTypes.find((dt) =>\n dt.name.includes(dataType as string)\n )?.validate(value);\n};\n\nexport const encodeLookupKey = (cell: EditableCellInfo) => {\n const { rowUuid, columnUuid } = cell;\n return `${rowUuid},${columnUuid}`;\n};\n\nexport const decodeLookupKey = (key: string) => {\n const [rowUuid, columnUuid] = key.split(\",\");\n return { rowUuid, columnUuid };\n};\n\nexport const EMPTY_ROW_UUID = \"empty\";\n\nexport interface EditableCellInfo {\n rowUuid: string;\n columnUuid: string;\n}\n\ninterface SchemaPostResult extends DataConnection {\n appDataBindings: DataBinding[] | null;\n}\n\nexport type ModerationFilter = \"All\" | \"Selected\" | \"Deselected\";\n\n// export enum ModerationFilter {\n// All = 1,\n// Selected = 2,\n// Deselected = 3,\n// }\n\nconst createRow = (\n schema: SchemaNode[],\n insertValue?: Partial\n): ManualDataRow => {\n return {\n rowUuid: uuidv4(), // Generate a valid uuid\n // isNew: true, // Use this for painting cells green. But also can be used when posting data to backend\n columns: schema.map((node) => {\n const { uuid, name } = node;\n const value = insertValue?.uuid === uuid ? insertValue?.value : \"\";\n return {\n ...node,\n displayName: name,\n value: value as any,\n formattedValue: value as string,\n uuid: node.uuid || null,\n };\n }),\n };\n};\n\ninterface EditableDataState {\n editingRowUuid: string;\n editingCell: EditableCellInfo;\n modificationsLookup: Record;\n deletedRowsLookup: Record;\n addedRowsLookup: Record;\n validationErrorsLookup: Record;\n rows: ManualDataRow[];\n rowsInitialState: ManualDataRow[];\n searchText: string;\n applyModificationsFilter: boolean;\n applyValidationErrorsFilter: boolean;\n emptyRowValue: any;\n\n errorMessage: string;\n showFixValidationErrorsMessage: boolean;\n dropdownPosition: Point;\n showConfirmCloseModal: boolean;\n showMoveRowModal: boolean;\n moveRowUuid: string;\n showModal: boolean;\n dropdownOpenRowUuid: string;\n\n dataManagerTabIndex: number;\n settingsTabIndex: number;\n showDisconnectSuccessMessage: boolean;\n moderationFilter: ModerationFilter;\n}\n\nconst makeInitialState = () => {\n return {\n showModal: false,\n dropdownOpenRowUuid: \"\",\n editingRowUuid: \"\",\n editingCell: {\n rowUuid: \"\",\n columnUuid: \"\",\n },\n modificationsLookup: {},\n deletedRowsLookup: {},\n addedRowsLookup: {},\n validationErrorsLookup: {},\n rows: [],\n searchText: \"\",\n rowsInitialState: [],\n applyModificationsFilter: false,\n applyValidationErrorsFilter: false,\n dropdownPosition: { x: 0, y: 0 },\n showFixValidationErrorsMessage: false,\n emptyRowValue: null,\n errorMessage: \"\",\n showConfirmCloseModal: false,\n showMoveRowModal: false,\n moveRowUuid: \"\",\n dataManagerTabIndex: 0,\n settingsTabIndex: 0,\n showDisconnectSuccessMessage: false,\n moderationFilter: \"All\" as ModerationFilter,\n };\n};\n\nexport const useEditableDataStore = defineStore(\"editableData\", {\n state: (): EditableDataState => {\n return makeInitialState();\n },\n\n getters: {\n /**\n *\n * Scope text searches to current moderation filter/bucket\n * If errors filter, show all errors\n *\n */\n filteredRows(): ManualDataRow[] {\n if (this.applyValidationErrorsFilter) {\n return this.validationErrorRows;\n }\n\n let rows = this.rows;\n\n if (this.moderationFilter === \"Selected\") {\n rows = this.moderationSelectedRows;\n }\n if (this.moderationFilter === \"Deselected\") {\n rows = this.moderationDeselectedRows;\n }\n\n if (this.searchText === \"\") {\n return rows || [];\n }\n\n // Apply search text filter:\n return rows.slice(0).filter((row) => {\n return row.columns.some((node) =>\n node.value\n ?.toString()\n .toLowerCase()\n .includes(this.searchText.toLowerCase())\n );\n });\n },\n\n editingRow(): ManualDataRow | null {\n if (!this.editingRowUuid) return null;\n return this.rows.find(\n (row) => row.rowUuid === this.editingRowUuid\n ) as ManualDataRow;\n },\n\n rowIndex(): number {\n return this.filteredRows.findIndex(\n (r: ManualDataRow) => r.rowUuid === this.editingRowUuid\n );\n },\n\n /**\n * Remove deleted rows from addedRows, modifiedRows, for upsertRequest\n */\n addedRows(): ManualDataRow[] {\n const deletedRowUuids: string[] = Object.keys(this.deletedRowsLookup);\n const addedRowUuids: string[] = Object.keys(this.addedRowsLookup);\n\n return this.rows.filter(\n (r) =>\n addedRowUuids.includes(r.rowUuid) &&\n !deletedRowUuids.includes(r.rowUuid)\n );\n },\n\n modifiedRows(): ManualDataRow[] {\n const deletedRowUuids: string[] = Object.keys(this.deletedRowsLookup);\n const modifiedCellKeys: string[] = Object.keys(this.modificationsLookup);\n const modifiedRowUuids = Array.from(\n new Set(modifiedCellKeys.map((key) => decodeLookupKey(key).rowUuid))\n );\n\n return this.rows.filter(\n (r) =>\n modifiedRowUuids.includes(r.rowUuid) &&\n !deletedRowUuids.includes(r.rowUuid)\n );\n },\n\n deletedRows(): ManualDataRow[] {\n const deletedRowUuids: string[] = Object.keys(this.deletedRowsLookup);\n return this.rows.filter((r) => deletedRowUuids.includes(r.rowUuid));\n },\n\n validationErrorRows(): ManualDataRow[] {\n const errorCellKeys: string[] = Object.keys(this.validationErrorsLookup);\n const deletedRowUuids: string[] = Object.keys(this.deletedRowsLookup);\n\n const errorRowUuids = Array.from(\n new Set(errorCellKeys.map((key) => decodeLookupKey(key).rowUuid))\n );\n\n return (this.rows || []).filter(\n (r) =>\n !deletedRowUuids.includes(r.rowUuid) &&\n errorRowUuids.includes(r.rowUuid)\n );\n },\n\n // Do not include deleted rows here, because count is filter button. So is weird to exclude from count but show in filter.\n validationErrorsCount(): number {\n const errorCellKeys: string[] = Object.keys(this.validationErrorsLookup);\n const deletedRowUuids: string[] = Object.keys(this.deletedRowsLookup);\n return errorCellKeys.filter((key) => {\n const { rowUuid } = decodeLookupKey(key);\n return !deletedRowUuids.includes(rowUuid);\n }).length;\n },\n\n hasUnsavedChanges(): boolean {\n return (\n Object.keys(this.deletedRowsLookup).length > 0 ||\n Object.keys(this.addedRowsLookup).length > 0 ||\n Object.keys(this.modificationsLookup).length > 0\n );\n },\n\n moderationSelectedRows(): ManualDataRow[] {\n return this.rows.filter(\n (r) => this.rows.find((x) => x.rowUuid === r.rowUuid)?.isSelected\n );\n },\n\n moderationDeselectedRows(): ManualDataRow[] {\n return this.rows.filter(\n (r) => !this.rows.find((x) => x.rowUuid === r.rowUuid)?.isSelected\n );\n },\n },\n\n actions: {\n clearFilters() {\n this.searchText = \"\";\n this.applyModificationsFilter = false;\n },\n\n setRowsInitialState(rows: ManualDataRow[]) {\n if (!rows) return;\n // Not sure about this..\n if (rows.length === 0) {\n this.rowsInitialState = [];\n }\n this.rowsInitialState = JSON.parse(JSON.stringify(rows));\n this.rows = rows;\n },\n\n clearState() {\n Object.assign(this, makeInitialState());\n },\n\n clearLookups() {\n Vue.set(this, \"modificationsLookup\", {});\n Vue.set(this, \"validationErrorsLookup\", {});\n Vue.set(this, \"addedRowsLookup\", {});\n Vue.set(this, \"deletedRowsLookup\", {});\n },\n\n setModerationDataRowSelected(payload: { refUuid: string; value: boolean }) {\n const { refUuid, value } = payload;\n const row = this.rows.find((n) => n.rowUuid === refUuid) as ManualDataRow;\n Vue.set(row, \"isSelected\", value);\n // We want to set this node as modified, so that hasUnsavedChanges becomes true\n const key = encodeLookupKey({ rowUuid: refUuid, columnUuid: \"none\" });\n Vue.set(this.modificationsLookup, key, true);\n },\n\n addRowAtIndex(payload: { row: ManualDataRow; index: number }) {\n const { row, index } = payload;\n this.rows.splice(index, 0, row);\n },\n\n addToAddedRows(uuid: string) {\n Vue.set(this.addedRowsLookup, uuid, true);\n },\n\n removeRow(uuid: string) {\n Vue.set(this.deletedRowsLookup, uuid, true);\n },\n\n undoRemoveRow(uuid: string) {\n Vue.delete(this.deletedRowsLookup, uuid);\n },\n\n clearModifications() {\n Vue.set(this, \"modificationsLookup\", {});\n },\n\n addValidationError(key: string) {\n Vue.set(this.validationErrorsLookup, key, true);\n },\n\n clearValidationErrors() {\n Vue.set(this, \"validationErrorsLookup\", {});\n },\n\n setNodeValue(payload: {\n rowUuid: string;\n columnUuid: string;\n value: any;\n formattedValue?: any;\n assetUuid?: any;\n }) {\n const { rowUuid, columnUuid, value } = payload;\n const row = this.rows.find((r) => r.rowUuid === rowUuid);\n const node = row?.columns.find((c) => c.uuid === columnUuid) as NodeData;\n\n Vue.set(node, \"value\", value); // Data grid inputs rely on this\n Vue.set(node, \"formattedValue\", payload.formattedValue ?? value); // Used for currency\n if (payload.assetUuid) {\n Vue.set(node, \"assetUuid\", payload.assetUuid); //Used for ImageUpload\n }\n },\n\n clearEditingCell(payload: {\n rowUuid: string;\n columnUuid: string;\n value?: any;\n schema: SchemaNode[];\n }) {\n const { rowUuid, columnUuid, schema } = payload;\n\n // Check needed because this is called on every click...Yikes should fix that\n if (!columnUuid) {\n return;\n }\n\n if ((rowUuid === EMPTY_ROW_UUID || !rowUuid) && !payload.value) {\n this.editingCell.columnUuid = \"\";\n this.editingCell.rowUuid = \"\";\n return;\n }\n\n // TODO: should be in fn\n const value =\n payload.value && rowUuid == EMPTY_ROW_UUID\n ? payload.value\n : this.rows\n .find((r) => r.rowUuid === rowUuid)\n ?.columns.find((node) => node.uuid === columnUuid)?.value;\n const initialValue = this.rowsInitialState\n .find((r) => r.rowUuid === rowUuid)\n ?.columns.find((node) => node.uuid === columnUuid)?.value;\n\n const cellKey = encodeLookupKey({ rowUuid, columnUuid });\n\n // VALIDATE DATA\n const schemaNode = schema.find(\n (n) => n.uuid === columnUuid\n ) as SchemaNode;\n const isValid = validateCell(schemaNode, value);\n\n if (!isValid) {\n Vue.set(this.validationErrorsLookup, cellKey, true);\n } else {\n Vue.delete(this.validationErrorsLookup, cellKey);\n }\n\n // We check against snapshot of initial state to calculate number of modifications\n if (value === initialValue) {\n // Must remove from modifications if there\n if (cellKey in this.modificationsLookup) {\n Vue.delete(this.modificationsLookup, cellKey);\n }\n } else {\n // Undefined check has result that new additions (anything not present in original dataset) are ignored\n if (initialValue !== undefined) {\n Vue.set(this.modificationsLookup, cellKey, true);\n }\n }\n\n this.editingCell.columnUuid = \"\";\n this.editingCell.rowUuid = \"\";\n },\n\n moveRowToPosition(payload: { rowUuid: string; position: number }) {\n const { rowUuid, position } = payload;\n const startIdx = this.rows.findIndex((r) => r.rowUuid === rowUuid);\n const row = this.rows.splice(startIdx, 1)[0];\n this.rows.splice(position, 0, row);\n const key = encodeLookupKey({ rowUuid, columnUuid: \"none\" });\n Vue.set(this.modificationsLookup, key, true);\n },\n\n insertRow(payload: { position: \"above\" | \"below\"; rowUuid: string }) {\n const { position, rowUuid } = payload;\n const rowIdx = this.filteredRows.findIndex(\n (r: ManualDataRow) => r.rowUuid === rowUuid\n );\n const idx = position === \"above\" ? rowIdx : rowIdx + 1;\n const connectionEditor = useConnectionEditorStore();\n if (connectionEditor.schema !== null) {\n const schema = connectionEditor.schema;\n const row = createRow(schema);\n this.addToAddedRows(row.rowUuid);\n this.addRowAtIndex({ row, index: idx });\n return row.rowUuid;\n }\n },\n\n createRowWithNode(node: Partial) {\n const connectionEditor = useConnectionEditorStore();\n if (connectionEditor.schema !== null) {\n const schema = connectionEditor.schema;\n const row = createRow(schema, node);\n const newRows = [...(this.rows || []), row];\n this.rows = newRows;\n this.addToAddedRows(row.rowUuid);\n return row.rowUuid;\n }\n },\n\n adjustEditingRowIndex(direction: \"inc\" | \"dec\") {\n const adjustValue = direction === \"inc\" ? 1 : -1;\n let currIdx = this.rowIndex;\n let newRowUuid = \"\";\n\n /**\n * Skip deleted rows\n */\n while (true) {\n const nextIdx =\n (currIdx + this.filteredRows.length + adjustValue) %\n this.filteredRows.length;\n\n const { rowUuid } = this.filteredRows.find(\n (r: ManualDataRow, idx: number) => idx === nextIdx\n ) as ManualDataRow;\n currIdx = nextIdx;\n if (!Object.keys(this.deletedRowsLookup).includes(rowUuid)) {\n newRowUuid = rowUuid;\n break;\n }\n }\n\n this.editingRowUuid = newRowUuid;\n },\n\n incrementEditingRowIndex() {\n this.adjustEditingRowIndex(\"inc\");\n },\n\n decrementEditingRowIndex() {\n this.adjustEditingRowIndex(\"dec\");\n },\n\n // ================================================================================\n\n async upsertManualTableData(args: { dcUuid: string; widgetIds: string[] }) {\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.clearErrors();\n EventBus.emit(\"AWAITING_SERVER\", true);\n connectionEditor.isLoading = true;\n\n const { dcUuid, widgetIds } = args;\n\n const addedRows = [...this.addedRows].map((row: ManualDataRow) => {\n return { ...row, action: \"insert\" };\n });\n const modifiedRows = [...this.modifiedRows].map((row: ManualDataRow) => {\n return { ...row, action: \"update\" };\n });\n const deletedRows = [...this.deletedRows].map((row: ManualDataRow) => {\n return { ...row, action: \"delete\" };\n });\n\n const alteredRows = [...addedRows, ...modifiedRows, ...deletedRows];\n\n const rows = alteredRows.map((row: any) => {\n const index = this.rows.findIndex((r) => r.rowUuid === row.rowUuid);\n return { ...row, index };\n });\n\n const payload = {\n rows,\n };\n\n // console.log(\n // \"======= UPSERTING MANUAL ROWS =======\",\n // payload,\n // JSON.stringify(payload),\n // alteredRows.length,\n // widgetIds,\n // appUuid\n // );\n\n /**\n * This endpoint throws an error if you call it without any rows in the payload.\n * This case can occur when this method is called from the SaveButton,\n * and the user has only made changes to moderation selections for rows.\n */\n if (alteredRows.length > 0) {\n await api.post(`dataconnection/${dcUuid}/manualtabledata`, payload);\n\n // Let connectionDataStore know about changes so RecordView (in left hand DataMenu) can reflect changes\n const connectionDataStore = useConnectionDataStore();\n connectionDataStore.fetchConnectionData({ connectionId: dcUuid });\n }\n\n // WidgetIds might not be available if edited from dashboard.\n if (widgetIds && widgetIds.length > 0) {\n // Ensures data fetches for the widget\n useAppDataStore().invalidateCacheForConnection(dcUuid);\n }\n },\n\n // This would get called from both of below, which would get exposed to outer world\n // and may in turn call a series of internal functions, such as convertToManualTableData\n async createEditableConnection(payload: {\n dataRows: any[];\n columnSchema: any[];\n type: \"disconnect\" | \"create\";\n appUuid: string;\n connectionName?: string;\n widget?: Widget | undefined;\n convertFromConnectionUuid?: string;\n }) {\n const {\n dataRows,\n columnSchema,\n type,\n appUuid,\n connectionName,\n widget,\n convertFromConnectionUuid,\n } = payload;\n let name = `${connectionName} - Manual`;\n if (type === \"create\" && widget) {\n name = `${widget.type} Data`;\n }\n const postPayload: any = {\n source: \"ManualSource\",\n name: name,\n nodes: columnSchema,\n rows: dataRows,\n };\n // If disconnecting from a widget, pass that info along\n if (widget && type === \"disconnect\") {\n postPayload.convertForWidgetId = widget.wid;\n postPayload.appUuid = appUuid;\n }\n if (type === \"disconnect\") {\n postPayload.convertFromConnectionUuid = convertFromConnectionUuid;\n }\n\n // post to endpoint\n const newConnection: SchemaPostResult = await api.post(\n \"dataconnection/manualtabledata/schema\",\n postPayload\n );\n // init connection\n const store = useConnectionDataStore();\n return store.initializeConnection(newConnection).then(async () => {\n // either replace or create data binding\n\n if (type === \"disconnect\") {\n this.replaceDataBinding(newConnection);\n } else if (type === \"create\") {\n this.createDataBinding(newConnection, widget);\n }\n\n await store.refreshConnectionData({\n connectionId: newConnection.uuid,\n isModerated: newConnection.moderationMode !== null,\n });\n return newConnection;\n });\n },\n\n replaceDataBinding(connection: SchemaPostResult) {\n const { nodeSets, appDataBindings } = connection;\n // NOTE: Must strip off the sf_index property here!\n const schema = (nodeSets[0].nodes || []).filter(\n (n: any) => !n.isArtificial\n );\n const appEditor = useAppEditorStore();\n appEditor.replaceDataBindings({ bindings: appDataBindings });\n\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.schema = schema;\n },\n\n createDataBinding(\n connection: SchemaPostResult,\n widget: Widget | undefined\n ) {\n const { uuid, nodeSets } = connection;\n // NOTE: Must strip off the sf_index property here!\n const schema = nodeSets[0].nodes?.filter((n) => !n.isArtificial);\n // console.log(\"schema\", schema);\n if (widget) {\n const bindingType =\n widget.type === \"Repeater\" ? \"DataSetParent\" : \"DataSet\";\n const binding: DataBinding = {\n widgetId: widget.wid,\n property: \"data\",\n dataUuid: nodeSets[0].uuid as string,\n bindingType,\n dataConnectionUuid: uuid,\n };\n\n const appEditor = useAppEditorStore();\n appEditor.addDataBinding(binding);\n\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.setConnection(connection);\n connectionEditor.schema = schema;\n }\n },\n\n async disconnectAndCreateEditableConnection(args: {\n appUuid: string;\n convertFromConnectionUuid: string;\n name: string;\n widget: Widget | undefined;\n }) {\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.clearErrors();\n connectionEditor.isLoading = true;\n\n const { appUuid, convertFromConnectionUuid, name, widget } = args;\n\n // Convert the data to \"manaul table data\" form:\n const data: { rows: ManualDataRow[]; schema: SchemaNode[] } =\n await api.get(\n `dataconnection/${convertFromConnectionUuid}/convertto/manualtabledata`\n );\n\n const { rows, schema } = data;\n\n const dataRows = rows.map((r, i) => {\n return {\n index: i,\n isSelected: !!r.isSelected,\n columns: r.columns,\n };\n });\n\n // Need to NOT take the first one, which is NodeSet -- and also want to avoid any PrimitiveArrays or ObjectArrays\n // And we also want to strip off \"Name\", \"Email\" and \"ResponseStatus\" nodes from disconnected Calendar schemas\n // We use slice(1) because the FIRST item in the schema is the entire nodeset, and all first-level nodes have that as parentUuid\n const columnSchema = schema\n .map((n) => {\n return { ...n, columnId: n.uuid }; // Avoids the \"invalid columnId error\"\n })\n .filter(\n (n) =>\n !schema\n .slice(1)\n .map((x) => x.uuid)\n .includes(n.parentUuid)\n )\n .filter((n) => n.kind === NodeKind.node);\n\n return this.createEditableConnection({\n dataRows,\n columnSchema,\n type: \"disconnect\",\n appUuid,\n widget,\n connectionName: name,\n convertFromConnectionUuid,\n });\n },\n\n createEditableConnectionFromPlaceholderData(args: {\n appUuid: string;\n widget?: Widget | undefined;\n }) {\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.clearErrors();\n connectionEditor.isLoading = true;\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n const { appUuid, widget } = args;\n const { rows, deletedRowsLookup } = this;\n const schema: SchemaNode[] = connectionEditor.schema;\n\n const dataRows = rows\n ?.filter((r) => !Object.keys(deletedRowsLookup).includes(r.rowUuid))\n .map((r, i) => {\n // NOTE: strip off rowUuid here\n return {\n index: i,\n isSelected: !!r.isSelected, // Pass moderation selection info\n columns: r.columns.map((n) => {\n const { value, uuid } = n;\n // console.log(\"create new node\", uuid, n);\n return {\n formattedValue: value, // TODO: IS this wrong?\n query: uuid, // NOTE: This must be query, not columnId\n };\n }),\n };\n });\n\n const columnSchema = schema.map((n) => {\n const { name, dataType, uuid, isRequired } = n;\n return {\n name,\n dataType,\n query: uuid,\n isSelected: true, // This is needed\n isRequired: isRequired || true,\n };\n });\n\n return this.createEditableConnection({\n dataRows,\n columnSchema,\n type: \"create\",\n appUuid,\n widget,\n });\n },\n },\n});\n","import { defineStore } from \"pinia\";\n\nimport {\n DataConnection,\n NodeSetData,\n NodeData,\n ConnectionDataResponse,\n ModerationDataNode,\n ModerationDataQueryResult,\n ManualDataRow,\n SchemaNode,\n SchemaType,\n} from \"@/types/data\";\nimport { api, BackendError } from \"@/api/backend\";\nimport { useConnectionEditorStore } from \"@/stores/connectionEditor\";\nimport { getComponentName } from \"@/components/data/dataNodeComponents\";\nimport { useConnectionsStore } from \"@/stores/connections\";\nimport uniqWith from \"lodash.uniqwith\";\nimport {\n parseDataRows,\n parseDataSchema,\n parseModeratedRows,\n} from \"@/util/dataUtils\";\nimport { useEditableDataStore } from \"@/stores/editableData\";\nimport Vue from \"vue\";\n\nexport interface ConnectionDataState {\n connectionData: NodeSetData | NodeData | null;\n isLoadingData: boolean;\n errorLoadingData: string | null;\n recordIndex: number;\n moderationDataRows: ModerationDataNode[];\n isModerationKeyError: boolean;\n\n /**\n * Used in ReadonlyData.vue We need to refactor to a Pinia store\n */\n readOnlyTableData: NodeData[][] | null;\n /**\n * Used in ReadonlyData.vue We need to refactor to a Pinia store\n */\n readOnlyTreeData: NodeSetData | null;\n\n /**\n * Used in ReadonlyData.vue We need to refactor to a Pinia store\n */\n readOnlySchemaType: SchemaType | null;\n}\n\nexport const useConnectionDataStore = defineStore(\"connectionData\", {\n state: (): ConnectionDataState => {\n return {\n connectionData: null,\n isLoadingData: false,\n isModerationKeyError: false,\n errorLoadingData: null,\n recordIndex: 0,\n moderationDataRows: [],\n readOnlySchemaType: null,\n readOnlyTableData: null,\n readOnlyTreeData: null,\n };\n },\n getters: {\n connectionDataSize(): number {\n return this.connectionData === null\n ? 0\n : (this.connectionData as NodeSetData).children?.length || 0;\n },\n\n connectionIsTree(): boolean {\n if (!this.connectionData) return false;\n\n return (\n this.connectionData?.kind === \"NodeObject\" ||\n (this.connectionData as any).children?.[0]?.kind === \"NodeObject\"\n );\n },\n\n isEmpty(): boolean {\n if (!this.connectionData) return true;\n if (!(\"children\" in this.connectionData)) {\n return true;\n }\n if (this.connectionData.children === null) return true;\n return this.connectionData.children?.length === 0;\n },\n\n dataRow(): any[] {\n let columns: NodeData[] = [];\n\n if (!this.connectionData) return [];\n if (!(\"children\" in this.connectionData)) return [];\n if (this.connectionData?.children?.length > 0) {\n columns = (this.connectionData as NodeSetData)?.children?.[\n this.recordIndex\n ] as NodeData[];\n }\n\n // Possible to get here... even though should detect as tree....\n if (!Array.isArray(columns)) return [];\n\n const selectedConnection = useConnectionEditorStore().connection;\n\n // NOTE: copying from DataSetExplorer this pattern of passing nodeSetUuid, nodeUuid and connectionUuid here.\n return (columns || []).map((node) => {\n return {\n ...node,\n connectionUuid: selectedConnection?.uuid,\n nodeSetUuid: selectedConnection?.nodeSets?.[0]?.uuid,\n nodeUuid: node.uuid,\n component: getComponentName(node.dataType),\n };\n });\n },\n\n moderationRow(): ModerationDataNode {\n // Hopefully we can rely on index..\n return this.moderationDataRows[this.recordIndex];\n },\n },\n\n actions: {\n setReadOnlyData(payload: { schemaType: SchemaType; data: unknown }) {\n this.readOnlySchemaType = payload.schemaType;\n\n if (payload.schemaType === \"Tabular\") {\n this.readOnlyTableData = payload.data as NodeData[][];\n }\n\n if (payload.schemaType === \"Tree\") {\n this.readOnlyTreeData = payload.data as NodeSetData;\n }\n },\n\n /**\n * Only used by initializeConnection\n * Need to wrap data up like {data: result.data} because widget data is organized by widgetId, and then property\n * And we can assume the property name is \"data\"\n *\n * NOTE that this data is filtered by moderation status, if moderation is active.\n * To get unfiltered moderation data for moderated connections, must use getModerationData.\n */\n async loadData(payload: { connectionId: string }) {\n const { connectionId } = payload;\n\n return api\n .get(`dataconnection/${connectionId}/data`)\n .then((result) => {\n const connectionStore = useConnectionEditorStore();\n connectionStore.setLastSourceUpdate(result.lastUpdated);\n\n return (result.data as NodeSetData).children as NodeData[][];\n });\n },\n\n /**\n * Loads data for read-only view in data manager.\n *\n * All the other methods in this jungle are loading data for collections.\n * We'll use this until we convert to the one true faith, Piniaism\n */\n async loadReadOnlyData(args: { uuid: string }) {\n // console.log(\"loadReadOnlyData\", args.uuid);\n return api\n .get(`dataconnection/${args.uuid}/data`)\n .then((result) => {\n const dataResponse = result.data;\n let schemaType: SchemaType | null = null;\n\n if (dataResponse.kind === \"NodeObject\") {\n schemaType = \"Tree\";\n } else if (dataResponse.kind === \"NodeSet\") {\n const isTable = (dataResponse as NodeSetData).children.some((c) =>\n Array.isArray(c)\n );\n\n schemaType = isTable ? \"Tabular\" : \"Tree\";\n }\n\n if (schemaType === \"Tabular\") {\n this.setReadOnlyData({\n schemaType,\n data: (dataResponse as NodeSetData).children as NodeData[][],\n });\n }\n\n if (schemaType === \"Tree\") {\n this.setReadOnlyData({\n schemaType,\n data: dataResponse as NodeSetData,\n });\n }\n });\n },\n\n async initializeConnection(connection: DataConnection) {\n const connectionsStore = useConnectionsStore();\n connectionsStore.addConnection(connection);\n\n this.loadData({\n connectionId: connection.uuid,\n });\n },\n\n /**\n * Fetches connection data for collection connections.\n * Will grab moderation data instead if the connection is moderated.\n * (This ensures that *all* connection data is returned, even for moderated connections that have some rows removed.)\n *\n */\n async refreshConnectionData(args: {\n connectionId: string;\n isModerated: boolean;\n updateSchema?: boolean;\n }) {\n const { connectionId, isModerated, updateSchema } = args;\n let rows: ManualDataRow[] = [];\n let schema: SchemaNode[] = [];\n\n if (isModerated) {\n const dataSet = await this.getModerationData({\n dcUuid: connectionId,\n });\n rows = parseModeratedRows(dataSet);\n } else {\n const fetchPayload: any = {\n connectionId,\n };\n\n const dataSet = await this.loadData(fetchPayload);\n schema = parseDataSchema(dataSet);\n\n if (!dataSet) {\n return { rows: null, schema };\n }\n rows = parseDataRows(dataSet);\n }\n\n const editableDataStore = useEditableDataStore();\n editableDataStore.setRowsInitialState(rows);\n\n if (updateSchema && schema.length > 0) {\n const connectionStore = useConnectionEditorStore();\n connectionStore.schema = schema;\n }\n return { rows };\n },\n\n openNewConnection(connection: DataConnection) {\n const connectionEditorStore = useConnectionEditorStore();\n const connectionsStore = useConnectionsStore();\n connectionEditorStore.connection = connection;\n connectionsStore.connections = uniqWith(\n [connection].concat(connectionsStore.connections),\n (a, b) => a.uuid === b.uuid\n );\n return this.fetchConnectionData({\n connectionId: connection.uuid,\n });\n },\n\n async fetchConnectionData(payload: { connectionId: string }) {\n const { connectionId } = payload;\n const { moderationMode, connection } = useConnectionEditorStore();\n this.isLoadingData = true;\n\n if (moderationMode !== null) {\n await this.getModerationData({\n dcUuid: connection?.uuid || \"\",\n })\n .then((rows) => {\n const children = rows.map((row: ModerationDataNode) => row.data);\n if (!this.connectionData) this.connectionData = {} as NodeData;\n Vue.set(this.connectionData, \"children\", children);\n })\n .finally(() => {\n this.isLoadingData = false;\n });\n }\n return api\n .get(`dataconnection/${connectionId}/data`)\n .then((result) => {\n if (moderationMode === null) {\n this.connectionData = result.data;\n } else {\n // We need to set uuid here in order for createDynamicWidget to be able to pick it up\n Vue.set(this.connectionData as NodeData, \"uuid\", result.data.uuid);\n }\n })\n .finally(() => {\n this.isLoadingData = false;\n });\n },\n\n async getModerationData(args: { dcUuid: string }) {\n this.errorLoadingData = null;\n\n return api\n .get(\n `dataconnection/${args.dcUuid}/moderation`\n )\n .then((res) => {\n const nodes = res?.data?.data?.moderationDataNodes;\n this.moderationDataRows = nodes;\n return nodes;\n })\n .catch((err) => {\n // TODO: translate\n this.errorLoadingData = \"dataModeration.getModDataError\";\n return Promise.reject(err);\n });\n },\n\n async updateModerationData(args: {\n dcUuid: string;\n data: ModerationDataNode[];\n }) {\n this.errorLoadingData = null;\n\n const payload = {\n moderatedData: {\n data: {\n moderationDataNodes: args.data,\n },\n },\n };\n\n return api\n .put(`dataconnection/${args.dcUuid}/moderation`, payload)\n .then(() => {\n this.moderationDataRows = args.data;\n })\n .catch((err: BackendError[]) => {\n // TODO: translate\n this.errorLoadingData = \"dataModeration.updateModDataError\";\n\n if (err[0].code === \"nonUniquePrimaryKeys\") {\n err[0].code = `dataModeration.${err[0].code}`;\n throw err[0];\n }\n\n throw err;\n });\n },\n },\n});\n","import { defineStore } from \"pinia\";\nimport Vue from \"vue\";\nimport {\n DataConnection,\n NodeSetData,\n DataBindingUsageInfo,\n SchemaNode,\n ModerationMode,\n GetConnectionBindingsResult,\n EditSchemaResult,\n} from \"@/types/data\";\nimport { api, BackendError } from \"@/api/backend\";\nimport { logger } from \"@core/logger\";\nimport { EventBus } from \"@/eventbus\";\nimport { useAppDataStore } from \"@/stores/appData\";\nimport { useConnectionsStore } from \"@/stores/connections\";\nimport { useAppEditorStore } from \"@/stores/appEditor\";\nimport { useConnectionDataStore } from \"./connectionData\";\n\nexport interface ConnectionReqInfo {\n dataConnectionUuid: string;\n appUuid?: string;\n skipSetConnection?: boolean;\n}\n\nexport interface RemapCollectionLogicRefsPayload {\n targetConnectionUuid: string;\n appUuid: string;\n logicUuids: string[];\n sourceNodeToTargetNode?: Record;\n}\n\nexport interface RemapCollectionLogicRefsInfo\n extends RemapCollectionLogicRefsPayload {\n sourceConnectionUuid: string;\n}\nexport interface ConnectionEditorState {\n connection: DataConnection | null;\n schema: any | null;\n\n /**\n * Bindings that are currently in use in Published Apps only.\n */\n dataBindingUsageInfo: DataBindingUsageInfo[] | null;\n\n /**\n * Binding mappings for Published Apps that no longer exist in the data source.\n * Determined server side\n */\n brokenBindings: DataBindingUsageInfo[] | null;\n\n errorCode: string | null;\n errorMessage: string | null;\n dataUuid: string | null;\n isConnectionInitialized: boolean;\n isConnectionSuccess: boolean;\n isLoading: boolean;\n isEditMode: boolean;\n selectedModerationPrimaryKeyNodeQuery: string | null;\n moderationPrimaryKeyNodeCandidates: Partial[];\n missingSchemaNodes: SchemaNode[] | null;\n schemaInfoUuid: string | null;\n lastDataSourceUpdate: Date | null;\n nodesRequiringRemap: SchemaNode[];\n logicRefsRequiringRemap: string[];\n previouslySelectedConnection: DataConnection | null;\n}\n\nexport const useConnectionEditorStore = defineStore(\"connectionEditor\", {\n state: (): ConnectionEditorState => {\n return {\n connection: null,\n schema: null,\n dataUuid: null,\n dataBindingUsageInfo: null,\n brokenBindings: null,\n errorCode: null,\n errorMessage: null,\n isConnectionInitialized: false,\n isConnectionSuccess: false,\n isLoading: false,\n isEditMode: false,\n selectedModerationPrimaryKeyNodeQuery: null,\n moderationPrimaryKeyNodeCandidates: [],\n missingSchemaNodes: null,\n schemaInfoUuid: null,\n lastDataSourceUpdate: null,\n nodesRequiringRemap: [],\n logicRefsRequiringRemap: [],\n previouslySelectedConnection: null,\n };\n },\n getters: {\n moderationMode(): ModerationMode {\n return this.connection?.moderationMode || null;\n },\n\n isEditable(): boolean {\n return this.connection?.canEditData || false;\n },\n\n canModerate(): boolean {\n return this.connection?.canModerate || false;\n },\n },\n\n actions: {\n // MUTATIONS ============================================\n\n setConnection(connection: DataConnection | null) {\n this.connection = connection;\n this.errorCode = connection?.faultErrorCode ?? null;\n this.errorMessage = connection?.faultErrorMessage ?? null;\n },\n\n updateConnection(connection: DataConnection) {\n this.connection = { ...this.connection, ...connection };\n },\n\n setLastSourceUpdate(date?: string | Date) {\n if (typeof date === \"string\") {\n try {\n this.lastDataSourceUpdate = new Date(date);\n } catch (err) {\n logger.track(`setLastSourceUpdate: error parsing date ${date}`);\n }\n }\n if (date instanceof Date) {\n this.lastDataSourceUpdate = date;\n }\n },\n\n setRefreshRate(refreshRate: number) {\n if (this.connection !== null) {\n Vue.set(this.connection, \"refreshRateSec\", refreshRate);\n }\n },\n\n setModerationMode(moderationMode: ModerationMode) {\n if (this.connection) {\n if (this.connection.moderationMode && moderationMode === null)\n this.connection.shouldRemoveModeration = true;\n\n this.connection.moderationMode = moderationMode;\n }\n },\n\n // ACTIONS ============================================\n\n /**\n * synchronizeData\n * updateDataConnection\n */\n\n async updateRemapLogicRefs(payload: RemapCollectionLogicRefsInfo) {\n const {\n sourceConnectionUuid,\n appUuid,\n logicUuids,\n sourceNodeToTargetNode,\n targetConnectionUuid,\n } = payload;\n\n return api.post(`dataconnection/${sourceConnectionUuid}/logic/map`, {\n appUuid,\n logicUuids,\n sourceNodeToTargetNode,\n targetConnectionUuid,\n });\n },\n\n async updateSheetSelection(args: { connectionUuid: string; nodes: any }) {\n const { connectionUuid, nodes } = args;\n const connection = await this.getConnection({\n dataConnectionUuid: connectionUuid,\n });\n // console.log(\"conn in update sheet\", connection);\n // Needs name -- could pass in default name or do what we're doing now, fetch connection info\n return api.put(`dataconnection/${connectionUuid}`, {\n ...connection,\n nodes,\n });\n },\n\n async getConnection(payload: ConnectionReqInfo) {\n const { dataConnectionUuid, appUuid, skipSetConnection } = payload;\n let url = `dataconnection/${dataConnectionUuid}`;\n if (appUuid) {\n url += `?appUuid=${appUuid}`;\n }\n return api.get(url).then((res) => {\n if (!skipSetConnection) this.setConnection(res);\n return res || null;\n });\n },\n\n async getConnectionBindings(payload: { dcUuid: string }) {\n this.isLoading = true;\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n return api\n .get(\n `dataconnection/${payload.dcUuid}/databindings`\n )\n .then((res) => {\n /**\n * Do NOT setConnection here, because the dataConnection comes back without nodeSets!\n */\n\n this.dataBindingUsageInfo = res.appBindingInfo;\n })\n .finally(() => {\n EventBus.emit(\"AWAITING_SERVER\", false);\n this.isLoading = false;\n });\n },\n\n clearErrors() {\n this.errorCode = null;\n this.errorMessage = null;\n },\n\n async deleteConnection(connection: DataConnection) {\n this.clearErrors();\n this.isLoading = true;\n\n if (!connection) return;\n\n EventBus.emit(\"AWAITING_SERVER\", true);\n return api\n .delete(`dataconnection/${connection.uuid}`)\n .catch((err) => {\n this.errorCode = \"deleteError\";\n // TODO\n // this.setErrorMessage(TranslationKeys.deleteError);\n throw err;\n })\n .finally(() => {\n EventBus.emit(\"AWAITING_SERVER\", false);\n this.isLoading = false;\n });\n },\n\n removeDeletedConnection(connection: DataConnection) {\n const connectionsStore = useConnectionsStore();\n connectionsStore.removeConnection(connection.uuid);\n },\n\n async cloneConnection(args: { dcUuid: string; name: string }) {\n this.clearErrors();\n this.isLoading = true;\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n return api\n .post(`dataconnection/${args.dcUuid}/clone`, {\n name: args.name,\n })\n .catch((err) => {\n this.errorCode = \"cloneError\";\n // TODO\n // this.setErrorMessage(TranslationKeys.cloneError);\n throw err;\n })\n .finally(() => {\n EventBus.emit(\"AWAITING_SERVER\", false);\n this.isLoading = false;\n });\n },\n\n async synchronizeData() {\n if (this.connection === null) {\n return Promise.reject(\n \"No Connection Loaded in this. Can't synchronize\"\n );\n }\n\n const { uuid, moderationMode } = this.connection;\n\n this.clearErrors();\n this.isLoading = true;\n\n return api\n .post(`dataconnection/${uuid}/schema`, {})\n .then((res) => {\n this.setConnection(res.dataConnection);\n this.dataBindingUsageInfo = res.appBindings;\n this.brokenBindings = res.brokenBindings;\n this.missingSchemaNodes = res.missingNodes;\n this.schemaInfoUuid = res.schemaInfoUuid;\n this.setLastSourceUpdate(res.lastDataSourceUpdate as Date);\n })\n .then(async (): Promise => {\n const store = useConnectionDataStore();\n const connection = this.connection as DataConnection;\n\n /**\n * If a user presses \"Sync Now\" on a non-collection connection, we need to load the data\n */\n if (\n connection.isCollection === false &&\n connection.schemaType !== \"Calendar\" &&\n typeof connection.moderationMode !== \"string\"\n ) {\n await store.loadReadOnlyData({ uuid: connection.uuid });\n return connection.uuid;\n }\n\n /**\n * Afterwards, we need to update our copy of the data on the front end (rows + schema).\n * And if we are in app editor, must refresh widget's data -- accomplished by invalidating the data cache for this connection.\n * If dataset is moderated, must use moderated data endpoint instead.\n */\n\n useAppDataStore().invalidateCacheForConnection(connection.uuid);\n\n await store.refreshConnectionData({\n connectionId: uuid,\n isModerated: moderationMode !== null,\n updateSchema: true,\n });\n\n return connection.uuid;\n })\n .then((connectionId) => {\n EventBus.emit(\"DATA_CONNECTION_SYNCHRONIZED\", { uuid: connectionId });\n })\n .catch((err: BackendError[] | string | null) => {\n let code = \"unknown\";\n // TODO\n // let message = TranslationKeys.unknownError;\n let message = \"\";\n\n if (Array.isArray(err) && err.length > 0) {\n code = err[0].code;\n message = err[0].message;\n } else if (typeof err === \"string\") {\n message = err;\n }\n\n this.errorCode = code;\n this.errorMessage = message;\n\n return Promise.reject(err);\n })\n .finally(() => {\n this.isLoading = false;\n });\n },\n\n /**\n * TODO: We should always send in entire Connection...or pull it in here somehow... because in usage, we send partial info into this.\n * That's why we need the hard coded refreshRate.\n *\n * But..is this where the \"interpret 'Never' to mean 'convert to manual'\" is supposed to happen?\n */\n async updateDataConnection(connection: Partial) {\n this.clearErrors();\n\n this.isLoading = true;\n\n if (this.connection === null) {\n throw new Error(\"No connection loaded in this. Can't update\");\n }\n\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n const defaultPayload = this.connection;\n const payload = { ...defaultPayload, ...connection };\n if (!payload.nodes)\n payload.nodes = (payload as any as NodeSetData).children as any;\n\n /**\n * Fix issue with Json/Xml data sources, which require the parent nodeSet to be included as a node\n */\n if (payload.schemaType == \"Tree\" && payload.nodes !== undefined)\n payload.nodes.push(this.connection?.nodeSets[0] as any);\n\n if (\n this.connection?.moderationMode !== null &&\n this.selectedModerationPrimaryKeyNodeQuery !== null &&\n payload.nodes !== undefined\n ) {\n const primaryKeyNodeIndex = payload.nodes.findIndex(\n (n) => n.query === this.selectedModerationPrimaryKeyNodeQuery\n );\n if (primaryKeyNodeIndex !== -1)\n payload.nodes[primaryKeyNodeIndex].isPrimaryKey = true;\n }\n\n // console.log(\"edit mode\",this.isEditMode);\n // console.log(\"Put request updateDC\", payload);\n\n const req =\n !this.isEditMode && this.connection.canEditData\n ? api.post(\n `dataconnection/manualtabledata/schema`,\n payload\n )\n : api.put(\n `dataconnection/${this.connection?.uuid}`,\n payload\n );\n\n return req\n .then(async (result) => {\n this.isConnectionSuccess = true;\n const payload = { uuid: result.uuid };\n if (this.isEditMode) {\n EventBus.emit(\"DATA_CONNECTION_UPDATED\", payload);\n } else {\n EventBus.emit(\"DATA_CONNECTION_CREATED\", payload);\n }\n\n const appEditor = useAppEditorStore();\n const connectionData = useConnectionDataStore();\n const widgetId = appEditor.selectedWidget?.wid;\n\n if (widgetId !== undefined) {\n await connectionData.initializeConnection(result);\n }\n\n // Let connectionDataStore know about changes so RecordView (in left hand DataMenu) can reflect changes\n this.connection = result;\n\n return result;\n })\n .catch((err: BackendError[]) => {\n // this.setErrorCode(err[0]?.code ? err[0].code : \"unknown\");\n this.errorCode = err[0]?.code ? err[0].code : \"unknown\";\n\n // TODO:\n // this.setErrorMessage(\n // err[0]?.code\n // ? `errors.${err[0].code}`\n // : TranslationKeys.unknownError\n // );\n console.log(\"err with dc update\", err);\n throw err;\n })\n .finally(() => {\n this.isLoading = false;\n EventBus.emit(\"AWAITING_SERVER\", false);\n });\n },\n },\n});\n","export interface BorderOptions {\n borderWidth: number;\n borderColor: string;\n borderDashSize: number;\n borderGapSize: number;\n}\n\nexport const DefaultBorderOptions = {\n borderWidth: 0,\n borderColor: \"rgba(80, 80, 80, 1)\",\n borderIsDashed: false,\n borderDashSize: 0,\n borderGapSize: 15,\n};\n","// export interface BackgroundImageOptions {\n// url: string;\n// position: Point;\n// scale: number;\n// h: number;\n// w: number;\n// [property: string]: any;\n// }\n\nexport type BackgroundSizeOptions = \"repeat\" | \"no-repeat\";\n\nexport interface BackgroundOptions {\n backgroundColor?: string;\n // backgroundImage?: string | null;\n backgroundRepeat: BackgroundSizeOptions;\n backgroundSize: \"cover\" | \"contain\";\n // backgroundPosition?: Point;\n // backgroundScale?: number;\n backgroundImageUrl: string | null;\n backgroundImageW: number | null;\n backgroundImageH: number | null;\n backgroundRepeatSize: number;\n}\n\nexport const DefaultBackgroundOptions = {\n backgroundColor: \"rgba(127, 127, 127, 1)\",\n backgroundImageUrl: null,\n backgroundSize: \"cover\",\n backgroundRepeat: \"no-repeat\",\n backgroundImageW: null,\n backgroundImageH: null,\n backgroundRepeatSize: 100,\n // backgroundPosition: { x: 0, y: 0 },\n // backgroundScale: 1\n};\n\n// export const DefaultBackgroundOptions = {\n// backgroundColor: \"rgba(127, 127, 127, 1)\",\n// backgroundImage: {\n// url: \"\",\n// h: 0,\n// w: 0,\n// scale: 1,\n// position: { x: 0, y: 0 }\n// }\n// };\n","export interface CyclingOptions {\n /**\n * Whether to rotate content (when repeaters are bound to a dataset).\n * If false, only the first \"page\" of data will be displayed.\n * If true, all \"pages\" of data will be rotated.\n */\n cycleContent: boolean;\n\n /**\n * How many seconds to display each \"page\" before rotating\n */\n cycleDuration: number;\n\n /**\n * What style of animated transition to use when paging\n */\n cycleAnimationStyle: string;\n\n /**\n * Duration for the paging animation\n */\n cycleAnimationDuration: number;\n}\n\nexport const DefaultCyclingOptions = {\n cycleContent: false,\n cycleDuration: 10,\n cycleAnimationStyle: \"fade\",\n cycleAnimationDuration: 0.5\n};\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport {\n CyclingOptions,\n DefaultCyclingOptions,\n} from \"@/components/widgets/CyclingOptions\";\nimport { Rectangle } from \"@/types/shapes\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport type RepeaterFlowValue =\n | \"row-rtl\"\n | \"row-ltr\"\n | \"column-rtl\"\n | \"column-ltr\"\n | \"row\"\n | \"column\";\n\nexport type RepeaterSortDirectionValue = \"asc\" | \"desc\";\n\nexport const DATA_ROW_INDEX = \"DATA_ROW_INDEX\";\n\nexport const createRepeaterDimensions = (\n artboard: Rectangle\n): Partial => {\n const artboardSizeConstraint = 3 * Math.min(artboard.w, artboard.h);\n const widgetMargin = Math.min(artboard.w * 0.025, artboardSizeConstraint);\n\n const repeaterOptions: Partial = {\n x: widgetMargin,\n y: widgetMargin,\n w: artboard.w - widgetMargin * 2,\n h: artboard.h - widgetMargin * 2,\n };\n\n return repeaterOptions;\n};\n\nexport interface RepeaterChildData {\n [key: string]: {\n [key: string]: any;\n };\n}\n\nexport interface RepeaterOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions,\n CyclingOptions {\n type: \"Repeater\";\n\n /**\n * The number of rows in the repeat grid\n */\n rows: number;\n\n /**\n * The space between rows in pixels.\n */\n rowGap: number;\n\n /**\n * The number of columns in the repeat grid\n */\n columns: number;\n\n /**\n * The space between columns in pixels.\n */\n columnGap: number;\n\n /**\n * Determines the direction of the flow of data.\n * See https://developer.mozilla.org/en-US/docs/Web/CSS/grid-auto-flow\n */\n flow: RepeaterFlowValue;\n\n /**\n * If true, any cells that don't have data available will not be rendered.\n */\n hideEmptyCells: boolean;\n\n /**\n * The cell background color\n */\n cellBackgroundColor: string;\n\n /**\n * Whether to cycle content\n */\n cycleContent: boolean;\n\n /**\n * Whether to show the sorting options\n */\n sortData: boolean;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...DefaultCyclingOptions,\n flow: \"row\",\n rows: 2,\n rowGap: 20,\n columnGap: 20,\n columns: 3,\n hideEmptyCells: false,\n backgroundColor: \"rgba(0,0,0,0)\",\n cellBackgroundColor: \"rgba(0,0,0,0)\",\n borderWidth: 1,\n sortData: false,\n cycleContent: true,\n};\n\nconst CreateRepeater = (options: Partial): RepeaterOptions => {\n return Object.assign(\n {},\n defaultOptions,\n {\n wid: makeId(),\n type: \"Repeater\",\n name: DEFAULT_LAYER_NAMES.Repeater,\n },\n options\n );\n};\n\nexport default CreateRepeater;\n","import { removeReactivity } from \"@/utils\";\n\nexport const DATA_TOKEN_TYPE = \"datatoken\";\n\n/**\n * This is a copy of the definition in @tiptap/core\n * It is copied here to avoid importing @tiptap into\n * the renderer bundle\n */\nexport type TextContent = {\n type: string;\n attrs?: Record;\n content?: TextContent[];\n marks?: {\n type: string;\n attrs?: Record;\n [key: string]: any;\n }[];\n text?: string;\n [key: string]: any;\n};\n\nconst generateHtml = (doc: TextContent) => {\n if (doc.content) {\n return doc.content\n .flatMap((p) => {\n const para =\n p.content &&\n p.content\n .map((n) => {\n return n.type === \"datatoken\" ? n.attrs?.text : n.text;\n })\n .join(\"\");\n return `

${para ?? \" \"}

`;\n })\n .join(\"\");\n }\n return \"\";\n};\n\nexport const contentToHtml = (content: TextContent) => {\n if (typeof content === \"undefined\") {\n return \"\";\n }\n\n if (typeof content === \"string\") {\n return content;\n }\n\n const plainContent = removeReactivity(content);\n\n const html = generateHtml(plainContent);\n\n // Add empty space inside empty p tags to ensure the browser renders them with height:\n const res = html\n .replace(/

<\\/p>/g, \"

 

\")\n .replace(/

<\\/p>/g, \"

 

\")\n .replace(/

<\\/span><\\/p>/g, \"

 

\");\n return res;\n};\n\nexport const makeTextContent = (text: string) => {\n return {\n type: \"doc\",\n content: [\n {\n type: \"paragraph\",\n content: [\n {\n type: \"text\",\n text: text,\n },\n ],\n },\n ],\n };\n};\n\nexport const makeDynamicTextContent = (\n dataUuid: string,\n content: string | number\n) => {\n return {\n type: \"doc\",\n content: [\n {\n type: \"paragraph\",\n content: [\n {\n type: \"datatoken\",\n attrs: {\n uuid: dataUuid,\n // Ensure that when user drags in an empty data token, the widget doesn't get 0 height:\n text: content === \"\" ? \" \" : content,\n },\n },\n {\n type: \"text\",\n text: \" \",\n },\n ],\n },\n ],\n };\n};\n","export type TextTransformValue =\n | \"none\"\n | \"capitalize\"\n | \"uppercase\"\n | \"lowercase\";\n\nexport type HorizontalAlignValue = \"left\" | \"center\" | \"right\";\nexport type VerticalAlignValue = \"top\" | \"middle\" | \"bottom\";\n\nexport interface TextStyleOptions {\n alignHorizontal?: HorizontalAlignValue;\n alignVertical?: VerticalAlignValue;\n fontFamily?: string;\n fontWeight?: number;\n fontStyle?: string;\n fontSize?: number;\n textColor?: string;\n textTransform?: TextTransformValue;\n textStrike?: boolean;\n textUnderline?: boolean;\n letterSpacing?: number;\n lineHeight?: number;\n paragraphSpacing?: number;\n}\n\nexport const DefaultTextStyleOptions = {\n alignHorizontal: \"Left\",\n alignVertical: \"top\",\n fontFamily: \"Inter\",\n fontWeight: 400,\n fontStyle: \"normal\",\n fontSize: 96,\n textColor: \"#000\",\n textTransform: \"none\",\n textStrike: false,\n textUnderline: false,\n letterSpacing: 0,\n lineHeight: 1.2,\n paragraphSpacing: 1,\n};\n","export interface ShadowOptions {\n shadowX: number;\n shadowY: number;\n shadowBlur: number;\n shadowColor: string;\n shadowDisplay: boolean;\n}\n\nexport const DefaultShadowOptions = {\n shadowX: 0,\n shadowY: 0,\n shadowBlur: 0,\n shadowColor: \"black\",\n shadowDisplay: false,\n};\n\nexport const getShadowCss = (options: ShadowOptions) => {\n if (!options.shadowDisplay) return {};\n return {\n filter: `drop-shadow(${options.shadowX}px ${options.shadowY}px ${options.shadowBlur}px ${options.shadowColor})`,\n };\n};\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n TextStyleOptions,\n DefaultTextStyleOptions,\n} from \"@/components/widgets/TextStyleOptions\";\nimport {\n ShadowOptions,\n DefaultShadowOptions,\n} from \"@/components/widgets/ShadowOptions\";\nimport { makeTextContent, TextContent } from \"@/text\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface TextOptions\n extends ComponentOptions,\n TransformOptions,\n ShadowOptions,\n TextStyleOptions,\n AnimationOptions {\n content: TextContent;\n fitText: boolean;\n // Used when TextComp is child of TextGroup\n name?: string;\n spaceBefore?: number;\n}\n\nconst defaultContent = makeTextContent(\"Text\");\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultTextStyleOptions,\n ...DefaultShadowOptions,\n ...getDefaultAnimationOptions(),\n content: defaultContent,\n fitText: false,\n};\n\nconst CreateText = (options: Partial) => {\n const wid = makeId();\n // Must start with empty object\n return Object.assign({ wid }, defaultOptions, options, {\n type: \"Text\",\n name: DEFAULT_LAYER_NAMES.Text,\n });\n};\n\nexport default CreateText;\n","import { StyleValue } from \"vue/types/jsx\";\nimport { DefaultShadowOptions } from \"./components/widgets/ShadowOptions\";\nimport { TextOptions } from \"./components/widgets/Text/TextOptions\";\nimport { VerticalAlignValue } from \"./components/widgets/TextStyleOptions\";\nimport { contentToHtml } from \"./text\";\nimport { Size } from \"./types\";\n\nconst toInt = (value: string | null): number => {\n return value === null ? 0 : parseInt(value, 10);\n};\n\nconst measureContentSize = (el: HTMLElement | null): Size => {\n if (el) {\n const s = getComputedStyle(el);\n if (s) {\n const padX = toInt(s.paddingLeft) + toInt(s.paddingRight);\n const padY = toInt(s.paddingTop) + toInt(s.paddingBottom);\n return {\n w: el.clientWidth - padX,\n h: el.clientHeight - padY,\n };\n }\n }\n return { w: 0, h: 0 };\n};\n\nconst isTooBig = (el: HTMLElement) => {\n const parent = el.parentElement;\n const width = el.scrollWidth > el.offsetWidth;\n const height = el.scrollHeight > el.offsetHeight;\n const p = parent && el.clientHeight > parent.clientHeight + 1;\n return width || height || p;\n};\n\nexport const textFit = (el: HTMLElement, minFontSize = 10) => {\n const style = window.getComputedStyle(el);\n const display = style.display;\n\n // fill available space in the parent element\n el.style.display = \"block\";\n el.style.width = \"100%\";\n el.style.height = \"100%\";\n\n let fontSize = parseInt(style.fontSize || \"\", 10);\n const startSize = fontSize;\n\n while (isTooBig(el) && fontSize >= minFontSize) {\n fontSize--;\n el.style.fontSize = `${fontSize}px`;\n }\n\n // Reset styles\n el.style.width = \"auto\";\n el.style.height = \"auto\";\n el.style.display = display;\n\n return { startSize, endSize: fontSize };\n};\n\nexport const fitTextLine = (el: HTMLElement, area?: Size) => {\n if (!area) {\n area = measureContentSize(el.parentElement);\n }\n const es = getComputedStyle(el);\n const startSize = parseInt(es.fontSize || \"\", 10);\n const scale = Math.max(el.clientWidth / area.w);\n const newSize = startSize * (1 / scale);\n el.style.fontSize = `${newSize.toFixed(2)}px`;\n return { startSize, endSize: newSize, scale };\n};\n\nexport const fitTextLines = (els: HTMLElement[]) => {\n if (els.length > 0) {\n let parent = els[0].parentElement;\n const area = measureContentSize(parent);\n let h = 0;\n let sizes: number[] = [];\n els.forEach((el) => {\n sizes.push(fitTextLine(el, area).endSize);\n h += el.clientHeight;\n });\n\n if (h > area.h) {\n const scale = Math.max(area.h / h);\n els.forEach((el, index) => {\n let newSize = sizes[index] * scale;\n el.style.fontSize = `${newSize.toFixed(2)}px`;\n });\n }\n }\n};\n\nconst alignContent = (value: VerticalAlignValue | undefined) => {\n switch (value) {\n case \"top\":\n return \"flex-start\";\n case \"middle\":\n return \"center\";\n case \"bottom\":\n return \"flex-end\";\n default:\n return \"flex-start\";\n }\n};\n\nexport const getTextStyleCss = (settings: TextOptions) => {\n const decorations = [];\n if (settings.textUnderline) {\n decorations.push(\"underline\");\n }\n if (settings.textStrike) {\n decorations.push(\"line-through\");\n }\n\n const style: StyleValue = {\n display: \"flex\",\n flexDirection: \"column\",\n textAlign: settings.alignHorizontal,\n justifyContent: alignContent(settings.alignVertical),\n fontFamily: settings.fontFamily,\n fontWeight: settings.fontWeight,\n fontStyle: settings.fontStyle,\n fontSize: `${settings.fontSize}px`,\n textDecoration: decorations.join(\" \"),\n textTransform: settings.textTransform,\n letterSpacing: `${settings.letterSpacing}pt`,\n lineHeight: settings.lineHeight,\n color: settings.textColor,\n opacity: settings.opacity,\n };\n\n if (settings.shadowDisplay) {\n const { shadowX, shadowY, shadowBlur, shadowColor } = DefaultShadowOptions;\n const x = isNaN(settings.shadowX) ? shadowX : settings.shadowX;\n const y = isNaN(settings.shadowY) ? shadowY : settings.shadowY;\n const blur = isNaN(settings.shadowBlur) ? shadowBlur : settings.shadowBlur;\n const color = settings.shadowColor ?? shadowColor;\n style.filter = `drop-shadow(${x}px ${y}px ${blur}px ${color})`;\n }\n\n return style;\n};\n\nexport const measureText = (\n settings: TextOptions,\n compiledHtml?: string,\n draggedNode?: boolean\n): Size => {\n const htmlContent =\n typeof compiledHtml === \"string\"\n ? compiledHtml\n : contentToHtml(settings.content);\n\n const div = document.createElement(\"div\");\n div.innerHTML = htmlContent;\n const styles = getTextStyleCss(settings);\n Object.keys(styles).forEach((key: string) => {\n (div.style as any)[key] = (styles as any)[key];\n });\n\n div.style.position = \"fixed\";\n div.style.top = \"0\";\n div.style.left = \"0\";\n\n // Trying to make consistent with tip-tap....\n // But thinking may be better to just NOT break words in editor.\n // div.style.wordWrap = \"break-word\";\n\n // if (constrainAxis === \"x\" && constrainValue > 0) {\n // div.style.width = `${constrainValue}px`;\n // }\n\n // Must set width to get it to work:\n if (!draggedNode) {\n div.style.width = `${settings.w * settings.scaleX}px`;\n }\n\n // if (constrainAxis === \"y\" && constrainValue > 0) {\n // div.style.height = `${constrainValue}px`;\n // }\n document.body.appendChild(div);\n\n const bounds = div.getBoundingClientRect();\n document.body.removeChild(div);\n return {\n w: bounds.width,\n h: bounds.height,\n };\n};\n","import { TextOptions } from \"@/components/widgets/Text/TextOptions\";\nimport { DataBinding, NodeData } from \"@/types/data\";\nimport { TextContent, DATA_TOKEN_TYPE } from \"@/text\";\nimport { removeReactivity } from \"@/utils\";\n\nconst traverse = (\n node: TextContent,\n data: Record,\n func: Function\n) => {\n for (let i in node) {\n func.call(null, node[i], data);\n if (node[i] !== null && typeof node[i] == \"object\") {\n traverse(node[i], data, func);\n }\n }\n};\n\nexport const bindDynamicTextContent = (\n textWidget: TextOptions,\n dataBindings: DataBinding[],\n record: NodeData[]\n): TextContent => {\n if (!textWidget) {\n return { type: \"doc\", content: [] };\n }\n\n const process = (node: TextContent, data: Record) => {\n if (node && node.type === DATA_TOKEN_TYPE && node.attrs) {\n node.attrs[\"text\"] = data[node.attrs.uuid];\n }\n };\n\n /**\n * Create a simple key/value Record with token binding data\n */\n const data: Record = dataBindings.reduce(\n (result: any, b: DataBinding) => {\n if (b.widgetId === textWidget.wid && b.property.startsWith(\"content\")) {\n const dataNode = record?.find((n) => n.uuid === b.dataUuid);\n result[b.dataUuid] = dataNode?.formattedValue.toString();\n }\n return result;\n },\n {}\n );\n\n // Make a copy of the content\n const content = removeReactivity(textWidget.content);\n\n // If we have token data, scan through the content tree\n // and replace datatoken nodes with value;\n if (Object.keys(data).length > 0) {\n traverse(content, data, process);\n }\n\n // console.log(\"Bound content\", content);\n\n return content;\n};\n\ninterface DataTokenValue {\n uuid: string;\n value: any;\n}\n\nexport const populateDataToken = (\n content: TextContent,\n dataBinding: DataBinding,\n record: NodeData[] | NodeData\n): TextContent => {\n // If binding is scalar, we just pass in the node directly, rather than an array of nodes\n const dataNode = Array.isArray(record)\n ? record.find((n) => n.uuid === dataBinding.dataUuid)\n : record;\n\n const dataValue = dataNode?.formattedValue?.toString();\n\n const data: DataTokenValue = { uuid: dataBinding.dataUuid, value: dataValue };\n\n // Make a copy of the content\n const contentCopy = removeReactivity(content);\n\n const process = (node: TextContent, data: DataTokenValue) => {\n if (\n node &&\n node.type === DATA_TOKEN_TYPE &&\n node.attrs &&\n node.attrs.uuid === data.uuid\n ) {\n node.attrs[\"text\"] = data.value;\n }\n };\n\n // If we have token data, scan through the content tree\n // and replace datatoken nodes with value;\n if (dataNode) {\n traverse(contentCopy, data, process);\n }\n\n return contentCopy;\n};\n\n/**\n * This replaces all occurences of a dataUuid with a new dataUuid.\n * It is called from the remapCollection action in Vuex.\n * @param content Content of a Text widget\n * @param oldUuid Old dataUuid of a binding (to be replaced)\n * @param newUuid dataUuid of the new, replacement binding\n * @returns TextContent\n */\nexport const updateTokenBindings = (\n document: TextContent,\n oldUuid: string,\n newUuid: string | undefined\n): TextContent => {\n document.content?.forEach((paragraph) => {\n paragraph.content = paragraph.content?.reduce((result, text) => {\n let shouldInclude = true;\n if (text.type === \"datatoken\" && text.attrs?.uuid === oldUuid) {\n if (typeof newUuid !== \"undefined\") {\n text.attrs.uuid = newUuid;\n } else {\n shouldInclude = false;\n }\n }\n if (shouldInclude) {\n result.push(text);\n }\n return result;\n }, [] as TextContent[]);\n });\n return document;\n};\n\n/**\n * Determines whether TextContent has any content or any data bindings.\n * If neither are present, it is considered \"empty\" and returns true.\n */\nexport const isTextContentEmpty = (content: TextContent): boolean => {\n let isEmpty = true;\n\n content.content?.forEach((paragraph) => {\n paragraph.content?.forEach((text) => {\n if (text.type === \"datatoken\" && typeof text.attrs?.uuid !== undefined) {\n isEmpty = false;\n } else if (\n text.type === \"text\" &&\n typeof text.text === \"string\" &&\n text.text.trim().length > 0\n ) {\n isEmpty = false;\n }\n });\n });\n\n return isEmpty;\n};\n","export interface BlurOptions {\n blurValue: number;\n blurDisplay: boolean;\n}\n\nexport const DefaultBlurOptions = {\n blurValue: 5,\n blurDisplay: false,\n};\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BlurOptions,\n DefaultBlurOptions,\n} from \"@/components/widgets/BlurOptions\";\nimport {\n ShadowOptions,\n DefaultShadowOptions,\n} from \"@/components/widgets/ShadowOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\nexport interface EllipseOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n ShadowOptions,\n BlurOptions,\n AnimationOptions,\n BackgroundOptions {}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultShadowOptions,\n ...DefaultBlurOptions,\n ...DefaultBackgroundOptions,\n ...getDefaultAnimationOptions(),\n};\n\nconst CreateEllipse = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Ellipse\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Ellipse,\n });\n};\n\nexport default CreateEllipse;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport { BackgroundOptions } from \"../BackgroundOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\n\nexport interface ImageOptions\n extends ComponentOptions,\n TransformOptions,\n BackgroundOptions,\n AnimationOptions,\n BorderOptions {\n datasets: any[] | null;\n url: string;\n name: string;\n uuid: string;\n // backgroundSize: string;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...getDefaultAnimationOptions(),\n datasets: [],\n backgroundSize: \"contain\",\n};\n\nconst CreateImage = (options: Partial) => {\n const wid = makeId();\n return Object.assign({ wid }, defaultOptions, options, { type: \"Image\" });\n};\n\nexport default CreateImage;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\n\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\ninterface RectangleOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n AnimationOptions,\n BackgroundOptions {\n borderRadius?: number;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...getDefaultAnimationOptions(),\n borderRadius: 0,\n};\n\nconst CreateRectangle = (options: Partial) => {\n return Object.assign(\n {\n type: \"Rectangle\",\n },\n defaultOptions,\n options,\n {\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Rectangle,\n }\n );\n};\n\nexport default CreateRectangle;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\ninterface MultiframeOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n backgroundImageUrl2: string;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n};\n\nconst MultiframeOptions = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Multiframe\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Multiframe,\n });\n};\n\nexport default MultiframeOptions;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n ShadowOptions,\n DefaultShadowOptions,\n} from \"@/components/widgets/ShadowOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface TriangleOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n ShadowOptions,\n AnimationOptions,\n BackgroundOptions {\n strokeLinejoin: \"miter\" | \"round\" | \"bevel\" | \"inherit\";\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...DefaultShadowOptions,\n ...getDefaultAnimationOptions(),\n strokeLinejoin: \"round\",\n};\n\nconst CreateTriangle = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Triangle\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Triangle,\n });\n};\n\nexport default CreateTriangle;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n ShadowOptions,\n DefaultShadowOptions,\n} from \"@/components/widgets/ShadowOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"../BackgroundOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface LineOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n ShadowOptions,\n AnimationOptions,\n BackgroundOptions {\n linecap: string;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultShadowOptions,\n ...DefaultBackgroundOptions,\n ...getDefaultAnimationOptions(),\n linecap: \"square\",\n};\n\nconst CreateLine = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Line\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.Line,\n });\n};\n\nexport default CreateLine;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface AnalogClockOptions extends ComponentOptions, TransformOptions {\n timezone: string | null;\n usePlayerTimezone: boolean;\n clock_textColor: string;\n clock_fontFamily: string;\n clock_fontSize: number;\n clock_fontWeight: number;\n clock_fontStyle: string;\n secondsColor: string;\n faceColor: string;\n strokeColor: string;\n showSeconds: boolean;\n numberInset: number;\n}\n\nexport const defaultTextOptions = {\n clock_textColor: \"#1d283a\",\n clock_fontFamily: \"Montserrat\",\n clock_fontSize: 48,\n clock_fontWeight: 700,\n clock_fontStyle: \"normal\",\n};\n\nconst defaultOptions: Partial = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...defaultTextOptions,\n timezone: null,\n usePlayerTimezone: true,\n secondsColor: \"#CC0000\",\n faceColor: \"#FFFFFF\",\n strokeColor: \"#000000\",\n showSeconds: true,\n numberInset: 50,\n};\n\nconst CreateAnalogClock = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"AnalogClock\",\n wid: makeId(6),\n lockAspect: true,\n name: DEFAULT_LAYER_NAMES.AnalogClock,\n });\n};\n\nexport default CreateAnalogClock;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface PieGraphOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n h: number;\n w: number;\n value_fontFamily: string;\n value_fontSize: number;\n value_fontWeight: number;\n value_fontStyle: string;\n value_textColor: string;\n legend_fontFamily: string;\n legend_fontSize: number;\n legend_fontWeight: number;\n legend_fontStyle: string;\n legend_textColor: string;\n legend_textTransform: string;\n lineColor: string;\n lineThickness: number;\n\n showLegend: boolean;\n legendMargin: number;\n legendPosition: string;\n showValues: boolean;\n valuesInside: boolean;\n valuesAsPercent: boolean;\n xAxisColumnId: string;\n yAxisColumnId: string;\n lockAspect: boolean;\n\n colors: string[];\n}\n\nexport const defaultPieGraphOptions = {\n height: 400,\n w: 400,\n value_fontFamily: \"Montserrat\",\n value_fontSize: 24,\n value_fontWeight: 600,\n value_fontStyle: \"normal\",\n value_textColor: \"#000000\",\n legend_fontFamily: \"Montserrat\",\n legend_fontSize: 24,\n legend_fontWeight: 300,\n legend_fontStyle: \"normal\",\n legend_textColor: \"#000000\",\n legend_textTransform: \"capitalize\",\n lineColor: \"#ffffff\",\n lineThickness: 5,\n showLegend: true,\n legendMargin: 30,\n legendPosition: \"Right\",\n showValues: false,\n valuesInside: true,\n valuesAsPercent: false,\n valueColumnId: \"\",\n labelColumnId: \"\",\n lockAspect: true,\n colors: [\n \"#134E4A\",\n \"#15756D\",\n \"#25B8A5\",\n \"#5DEAD5\",\n \"#CBFBF0\",\n \"#115F5A\",\n \"#1D968B\",\n \"#2CD4BD\",\n \"#98F6E3\",\n \"#E0FFF8\",\n ],\n\n /**\n * Stores the UUID of the data column representing the x axis\n */\n xAxisColumnId: \"b\",\n /**\n * Stores the UUID of the data column representing the y axis\n */\n yAxisColumnId: \"a\",\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultPieGraphOptions,\n};\n\nconst CreatePieGraph = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"PieGraph\",\n wid: makeId(),\n lockAspect: true,\n name: DEFAULT_LAYER_NAMES.PieGraph,\n });\n};\n\nexport default CreatePieGraph;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface LineGraphOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n h: number;\n w: number;\n minValue: number;\n maxValue: number;\n xTitle: string;\n yTitle: string;\n\n title_fontFamily: string;\n title_fontSize: number;\n title_fontWeight: number;\n title_textColor: string;\n title_textTransform: string;\n label_fontFamily: string;\n label_fontSize: number;\n label_fontWeight: number;\n label_textColor: string;\n label_textTransform: string;\n\n labelTicks: number;\n leftTitlePadding: number;\n bottomTitlePadding: number;\n bgColor: string;\n lineColor: string;\n lineThickness: number;\n plotlineColor: string;\n plotlineThickness: number;\n xLabelTilt: boolean;\n showBg: boolean;\n showGrid: boolean;\n showValues: boolean;\n xAxisColumnId: string;\n yAxisColumnId: string;\n colorColumn: string;\n xTitleManual: string | null;\n yTitleManual: string | null;\n minManual: number | null;\n maxManual: number | null;\n}\n\nexport const defaultLineGraphOptions = {\n h: 400,\n w: 500,\n minValue: 336,\n maxValue: 875,\n xTitle: \"Name\",\n yTitle: \"Height (m)\",\n barPadding: 0.4,\n title_fontFamily: \"Montserrat\",\n title_fontSize: 36,\n title_fontWeight: 300,\n title_fontStyle: \"normal\",\n title_textColor: \"#000000\",\n title_textTransform: \"none\",\n label_fontFamily: \"Montserrat\",\n label_fontSize: 20,\n label_fontWeight: 700,\n label_fontStyle: \"normal\",\n label_textColor: \"#000000\",\n label_textTransform: \"none\",\n labelTicks: 6,\n leftTitlePadding: 0,\n bottomTitlePadding: 0,\n bgColor: \"#f2f2f2\",\n lineColor: \"#000000\",\n lineThickness: 1,\n plotlineColor: \"#e21d48\",\n plotlineThickness: 7,\n xLabelTilt: false,\n showBg: false,\n showGrid: false,\n showValues: false,\n xAxisColumnId: \"a\",\n yAxisColumnId: \"b\",\n xTitleManual: null,\n yTitleManual: null,\n minManual: null,\n maxManual: null,\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultLineGraphOptions,\n};\n\nconst CreateLineGraph = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"LineGraph\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.LineGraph,\n });\n};\n\nexport default CreateLineGraph;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface BarGraphOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n h: number;\n w: number;\n topMargin: number;\n rightMargin: number;\n bottomMargin: number;\n leftMargin: number;\n minValue: number;\n maxValue: number;\n barPadding: number;\n xTitle: string;\n yTitle: string;\n\n title_fontFamily: string;\n title_fontSize: number;\n title_fontWeight: number;\n title_textColor: string;\n title_textTransform: string;\n label_fontFamily: string;\n label_fontSize: number;\n label_fontWeight: number;\n label_textColor: string;\n label_textTransform: string;\n\n lineColor: string;\n lineThickness: number;\n initialSet: boolean;\n labelTicks: number;\n leftTitlePadding: number;\n bottomTitlePadding: number;\n bgColor: string;\n barColor: string;\n customBarColors: string[];\n useCustomBarColors: boolean;\n showBg: boolean;\n showGrid: boolean;\n showValues: boolean;\n xAxisColumnId: string;\n yAxisColumnId: string;\n xTitleManual: string | null;\n yTitleManual: string | null;\n minManual: number | null;\n maxManual: number | null;\n}\n\nexport const defaultBarGraphOptions = {\n h: 400,\n w: 500,\n minValue: 0,\n maxValue: 4130,\n barPadding: 0.55,\n xTitle: \"Name\",\n yTitle: \"Length (mi.)\",\n title_fontFamily: \"Montserrat\",\n title_fontSize: 36,\n title_fontWeight: 300,\n title_fontStyle: \"normal\",\n title_textColor: \"#000000\",\n title_textTransform: \"none\",\n label_fontFamily: \"Montserrat\",\n label_fontSize: 20,\n label_fontWeight: 700,\n label_fontStyle: \"normal\",\n label_textColor: \"#000000\",\n label_textTransform: \"capitalize\",\n lineColor: \"#000000\",\n lineThickness: 0,\n labelTicks: 6,\n leftTitlePadding: 15,\n bottomTitlePadding: 15,\n bgColor: \"#f2f2f2\",\n barColor: \"#113054\",\n customBarColors: [\"#0C3656\", \"#185783\", \"#2984C5\", \"#3ABFF8\", \"#BAE5FD\"],\n useCustomBarColors: true,\n xLabelTilt: false,\n showBg: false,\n showGrid: false,\n showValues: false,\n xTitleManual: null,\n yTitleManual: null,\n minManual: null,\n maxManual: null,\n /**\n * Stores the UUID of the data column representing the x axis\n */\n xAxisColumnId: \"b\",\n /**\n * Stores the UUID of the data column representing the y axis\n */\n yAxisColumnId: \"a\",\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultBarGraphOptions,\n};\n\nconst CreateBarGraph = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"BarGraph\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.BarGraph,\n });\n};\n\nexport default CreateBarGraph;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface ColumnGraphOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n h: number;\n w: number;\n minValue: number;\n maxValue: number;\n barPadding: number;\n xTitle: string;\n yTitle: string;\n\n title_fontFamily: string;\n title_fontSize: number;\n title_fontWeight: number;\n title_textColor: string;\n title_textTransform: string;\n label_fontFamily: string;\n label_fontSize: number;\n label_fontWeight: number;\n label_textColor: string;\n label_textTransform: string;\n\n lineColor: string;\n lineThickness: number;\n labelTicks: number;\n leftTitlePadding: number;\n bottomTitlePadding: number;\n bgColor: string;\n barColor: string;\n customBarColors: string[];\n useCustomBarColors: boolean;\n xLabelTilt: boolean;\n showBg: boolean;\n showGrid: boolean;\n showValues: boolean;\n initialSet: boolean;\n xAxisColumnId: string;\n yAxisColumnId: string;\n xTitleManual: string | null;\n yTitleManual: string | null;\n minManual: number | null;\n maxManual: number | null;\n}\n\nexport const defaultColumnGraphOptions = {\n h: 768,\n w: 1355,\n minValue: 0,\n maxValue: 830,\n xTitle: \"Name\",\n yTitle: \"Height (m)\",\n barPadding: 0.45,\n title_fontFamily: \"Montserrat\",\n title_fontSize: 36,\n title_fontWeight: 300,\n title_fontStyle: \"normal\",\n title_textColor: \"#000000\",\n title_textTransform: \"none\",\n label_fontFamily: \"Montserrat\",\n label_fontSize: 20,\n label_fontWeight: 700,\n label_fontStyle: \"normal\",\n label_textColor: \"#000000\",\n label_textTransform: \"none\",\n lineColor: \"#000000\",\n lineThickness: 0,\n labelTicks: 8,\n leftTitlePadding: 0,\n bottomTitlePadding: 0,\n bgColor: \"#f2f2f2\",\n barColor: \"#113054\",\n customBarColors: [\"#FDCDD3\", \"#FB6F84\", \"#E21D48\", \"#A1123A\", \"#790A25\"],\n useCustomBarColors: true,\n xLabelTilt: false,\n showBg: false,\n showGrid: false,\n showValues: false,\n xTitleManual: null,\n yTitleManual: null,\n minManual: null,\n maxManual: null,\n /**\n * Stores the UUID of the data column representing the x axis\n */\n xAxisColumnId: \"a\",\n /**\n * Stores the UUID of the data column representing the y axis\n */\n yAxisColumnId: \"b\",\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultColumnGraphOptions,\n};\n\nconst CreateColumnGraph = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"ColumnGraph\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.ColumnGraph,\n });\n};\n\nexport default CreateColumnGraph;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface ProgressDonutOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n w: number;\n h: number;\n bgFill: string;\n fgFill: string;\n bgInnerRadius: number;\n bgOuterRadius: number;\n fgInnerRadius: number;\n fgOuterRadius: number;\n cornerRadius: number;\n minValue: number;\n maxValue: number;\n currentValue: number;\n lockAspect: boolean;\n}\n\nexport const defaultProgressDonutOptions = {\n w: 100,\n h: 400,\n minValue: 0,\n maxValue: 500,\n currentValue: 330,\n bgFill: \"rgba(254, 205, 211, 100)\",\n fgFill: \"rgba(244, 62, 92, 100)\",\n bgInnerRadius: 55,\n bgOuterRadius: 100,\n fgInnerRadius: 55,\n fgOuterRadius: 100,\n cornerRadius: 0,\n lockAspect: true,\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultProgressDonutOptions,\n};\n\nconst CreateProgressDonut = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"ProgressDonut\",\n wid: makeId(6),\n lockAspect: true,\n name: DEFAULT_LAYER_NAMES.ProgressDonut,\n });\n};\n\nexport default CreateProgressDonut;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport { TransformOptions } from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface ProgressBarOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n w: number;\n h: number;\n bgFill: string;\n fgFill: string;\n stroke: string;\n strokeWidth: number;\n fgPadding: number;\n borderRadius: number;\n showGradient: boolean;\n barGradientStart: string;\n barGradientEnd: string;\n showDivider: boolean;\n dividerFill: string;\n dividerStroke: string;\n dividerStrokeWidth: number;\n dividerHeight: number;\n dividerWidth: number;\n bgInnerRadius: number;\n bgOuterRadius: number;\n fgInnerRadius: number;\n fgOuterRadius: number;\n\n minValue: number;\n maxValue: number;\n currentValue: number;\n}\n\nexport const defaultProgressBarOptions = {\n w: 100,\n h: 400,\n minValue: 0,\n maxValue: 100,\n currentValue: 65,\n bgFill: \"rgba(187, 247, 208, 100)\",\n fgFill: \"rgba(33, 196, 93, 100)\",\n stroke: \"rgba(0, 0, 0, 100)\",\n strokeWidth: 0,\n fgPadding: 0,\n borderRadius: 0,\n showGradient: false,\n barGradientStart: \"rgba(0, 0, 0, 0)\",\n barGradientEnd: \"rgba(17, 48, 84, 100)\",\n showDivider: true,\n dividerFill: \"rgba(255, 255, 255, 100)\",\n dividerStroke: \"rgba(0, 0, 0, 100)\",\n dividerStrokeWidth: 0,\n dividerHeight: 1,\n dividerWidth: 100,\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultProgressBarOptions,\n w: 540,\n h: 360,\n x: 0,\n y: 0,\n z: 0,\n angle: 90,\n scaleX: 1,\n scaleY: 1,\n canScaleY: true,\n canScaleX: true,\n lockAspect: false,\n};\n\nconst CreateProgressBar = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"ProgressBar\",\n wid: makeId(6),\n name: DEFAULT_LAYER_NAMES.ProgressBar,\n });\n};\n\nexport default CreateProgressBar;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CalendarRoomScheduleOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n use24Hour: boolean;\n locationTitle: string | null;\n data?: any[];\n previewLiveData: boolean;\n\n progressBarBackgroundColor: string;\n progressBarFillColor: string;\n dividerFillColor: string;\n dividerSize: number;\n syncFonts: boolean;\n syncTextColors: boolean;\n\n locationTitle_fontFamily: string | null;\n locationTitle_textColor: string | null;\n locationTitle_fontSize: number;\n locationTitle_fontWeight: number;\n locationTitle_letterSpacing: number;\n locationTitle_lineHeight: number;\n locationTitle_textAlign: string;\n locationTitle_fontStyle: string;\n locationTitle_textTransform: string;\n locationTitle_textDecoration: string;\n\n headerEventTitle_fontFamily: string | null;\n headerEventTitle_textColor: string | null;\n headerEventTitle_fontSize: number;\n headerEventTitle_fontWeight: number;\n headerEventTitle_letterSpacing: number;\n headerEventTitle_lineHeight: number;\n headerEventTitle_textAlign: string;\n headerEventTitle_fontStyle: string;\n headerEventTitle_textTransform: string;\n headerEventTitle_textDecoration: string;\n\n headerEventTime_fontFamily: string | null;\n headerEventTime_textColor: string | null;\n headerEventTime_fontSize: number;\n headerEventTime_fontWeight: number;\n headerEventTime_letterSpacing: number;\n headerEventTime_lineHeight: number;\n headerEventTime_textAlign: string;\n headerEventTime_fontStyle: string;\n headerEventTime_textTransform: string;\n headerEventTime_textDecoration: string;\n\n subheaderEventTitle_fontFamily: string | null;\n subheaderEventTitle_textColor: string | null;\n subheaderEventTitle_fontSize: number;\n subheaderEventTitle_fontWeight: number;\n subheaderEventTitle_letterSpacing: number;\n subheaderEventTitle_lineHeight: number;\n subheaderEventTitle_textAlign: string;\n subheaderEventTitle_fontStyle: string;\n subheaderEventTitle_textTransform: string;\n subheaderEventTitle_textDecoration: string;\n\n subheaderEventTime_fontFamily: string | null;\n subheaderEventTime_textColor: string | null;\n subheaderEventTime_fontSize: number;\n subheaderEventTime_fontWeight: number;\n subheaderEventTime_letterSpacing: number;\n subheaderEventTime_lineHeight: number;\n subheaderEventTime_textAlign: string;\n subheaderEventTime_fontStyle: string;\n subheaderEventTime_textTransform: string;\n subheaderEventTime_textDecoration: string;\n\n clockTime_fontFamily: string | null;\n clockTime_textColor: string | null;\n clockTime_fontSize: number;\n clockTime_fontWeight: number;\n clockTime_letterSpacing: number;\n clockTime_lineHeight: number;\n clockTime_textAlign: string;\n clockTime_fontStyle: string;\n clockTime_textTransform: string;\n clockTime_textDecoration: string;\n}\n\nconst mainColor = \"rgba(52, 66, 86, 100)\";\nconst accentColor = \"rgba(244, 62, 92, 100)\";\n\nexport const defaultTextOptions: any = {\n locationTitle_fontSize: 80,\n locationTitle_fontWeight: 700,\n locationTitle_textColor: mainColor,\n locationTitle_letterSpacing: 0,\n headerEventTitle_fontSize: 80,\n headerEventTitle_fontWeight: 700,\n headerEventTitle_textColor: mainColor,\n headerEventTitle_letterSpacing: 0,\n headerEventTime_fontSize: 55,\n headerEventTime_fontWeight: 400,\n headerEventTime_textColor: accentColor,\n headerEventTime_letterSpacing: 0,\n subheaderEventTitle_fontSize: 40,\n subheaderEventTitle_fontWeight: 700,\n subheaderEventTitle_textColor: mainColor,\n subheaderEventTitle_letterSpacing: 1,\n subheaderEventTime_fontSize: 35,\n subheaderEventTime_fontWeight: 400,\n subheaderEventTime_textColor: accentColor,\n subheaderEventTime_letterSpacing: 0,\n clockTime_fontSize: 80,\n clockTime_fontWeight: 400,\n clockTime_textColor: \"#000000\",\n};\n\n[\n \"locationTitle\",\n \"headerEventTitle\",\n \"headerEventTime\",\n \"subheaderEventTitle\",\n \"subheaderEventTime\",\n \"clockTime\",\n].forEach((prop) => {\n defaultTextOptions[`${prop}_fontFamily`] = \"Montserrat\";\n defaultTextOptions[`${prop}_textTransform`] = \"none\";\n defaultTextOptions[`${prop}_textDecoration`] = \"none\";\n defaultTextOptions[`${prop}_fontStyle`] = \"normal\";\n defaultTextOptions[`${prop}_lineHeight`] = 1.2;\n defaultTextOptions[`${prop}_textAlign`] = \"left\";\n});\n\nconst defaultOptions = {\n ...defaultTextOptions,\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n borderRadius: 0,\n use24Hour: false,\n previewLiveData: false,\n progressBarBackgroundColor: \"rgba(225, 231, 239, 100)\",\n progressBarFillColor: \"rgba(52, 66, 86, 100)\",\n dividerFillColor: \"rgba(52, 66, 86, 100)\",\n dividerSize: 1,\n syncFonts: false,\n syncTextColors: false,\n};\n\nconst CreateCalendarRoomSchedule = (\n options: Partial\n) => {\n return Object.assign({}, defaultOptions, options, {\n wid: makeId(),\n type: \"CalendarRoomSchedule\",\n name: DEFAULT_LAYER_NAMES.CalendarRoomSchedule,\n });\n};\n\nexport default CreateCalendarRoomSchedule;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CalendarAgendaOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n data?: any[];\n previewLiveData: boolean;\n\n dayOffset: number;\n dayFormat: string;\n startHour: number;\n endHour: number;\n use24Hour: boolean;\n useDefaultStartTime: boolean;\n syncFonts: boolean;\n syncTextColors: boolean;\n eventBackgroundColor: string;\n allDayEventBackgroundColor: string;\n // allDayEventTextColor: string;\n\n allDayEvent_fontFamily: string | null;\n allDayEvent_textColor: string | null;\n allDayEvent_fontSize: number;\n allDayEvent_fontWeight: number;\n allDayEvent_letterSpacing: number;\n allDayEvent_lineHeight: number;\n allDayEvent_textAlign: string;\n allDayEvent_fontStyle: string;\n allDayEvent_textTransform: string;\n allDayEvent_textDecoration: string;\n\n date_fontFamily: string | null;\n date_textColor: string | null;\n date_fontSize: number;\n date_fontWeight: number;\n date_letterSpacing: number;\n date_lineHeight: number;\n date_textAlign: string;\n date_fontStyle: string;\n date_textTransform: string;\n date_textDecoration: string;\n\n eventTitle_fontFamily: string | null;\n eventTitle_textColor: string | null;\n eventTitle_fontSize: number;\n eventTitle_fontWeight: number;\n eventTitle_letterSpacing: number;\n eventTitle_lineHeight: number;\n eventTitle_textAlign: string;\n eventTitle_fontStyle: string;\n eventTitle_textTransform: string;\n eventTitle_textDecoration: string;\n\n eventTime_fontFamily: string | null;\n eventTime_textColor: string | null;\n eventTime_fontSize: number;\n eventTime_fontWeight: number;\n eventTime_letterSpacing: number;\n eventTime_lineHeight: number;\n eventTime_textAlign: string;\n eventTime_fontStyle: string;\n eventTime_textTransform: string;\n eventTime_textDecoration: string;\n\n eventLocation_fontFamily: string | null;\n eventLocation_textColor: string | null;\n eventLocation_fontSize: number;\n eventLocation_fontWeight: number;\n eventLocation_letterSpacing: number;\n eventLocation_lineHeight: number;\n eventLocation_textAlign: string;\n eventLocation_fontStyle: string;\n eventLocation_textTransform: string;\n eventLocation_textDecoration: string;\n}\n\nconst mainColor = \"rgba(52,66,86,100)\";\nconst accentColor = \"rgba(244,62,92,100)\";\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n borderRadius: 0,\n dayOffset: 0,\n dayFormat: \"today\",\n startHour: 7,\n endHour: 18,\n previewLiveData: false,\n // title: \"Calendar Title\",\n // showTitle: true,\n use24Hour: false,\n useDefaultStartTime: true,\n eventBackgroundColor: \"rgba(0,0,0,0)\",\n eventBulletColor: mainColor,\n allDayEventBackgroundColor: mainColor,\n // allDayEventTextColor: \"white\",\n syncFonts: false,\n syncTextColors: false,\n};\n\n// Ahhh, tricky -- how to allow manipulation of multiday....\n// fake it for now with text-4xl?\n\nexport const defaultTextOptions: any = {\n date_fontSize: 96,\n date_fontWeight: 300,\n date_textColor: accentColor,\n date_lineHeight: 1.2,\n date_textTransform: \"capitalize\",\n eventTitle_fontSize: 60,\n eventTitle_fontWeight: 400,\n eventTitle_textColor: mainColor,\n eventTitle_lineHeight: 1.2,\n eventTitle_textTransform: \"none\",\n eventLocation_fontSize: 36,\n eventLocation_fontWeight: 400,\n eventLocation_textColor: accentColor,\n eventLocation_lineHeight: 1.2,\n eventLocation_textTransform: \"none\",\n eventTime_fontSize: 36,\n eventTime_fontWeight: 700,\n eventTime_textColor: mainColor,\n eventTime_lineHeight: 2.0,\n eventTime_textTransform: \"none\",\n allDayEvent_fontSize: 36,\n allDayEvent_fontWeight: 400,\n allDayEvent_textColor: \"rgba(255,255,255,100)\",\n allDayEvent_lineHeight: 1.2,\n allDayEvent_textTransform: \"none\",\n};\n\n[\"date\", \"eventTitle\", \"eventLocation\", \"eventTime\", \"allDayEvent\"].forEach(\n (prop) => {\n defaultTextOptions[`${prop}_fontFamily`] = \"Montserrat\";\n defaultTextOptions[`${prop}_textDecoration`] = \"none\";\n defaultTextOptions[`${prop}_fontStyle`] = \"normal\";\n defaultTextOptions[`${prop}_letterSpacing`] = 0;\n defaultTextOptions[`${prop}_textAlign`] = \"left\";\n }\n);\n\ndefaultTextOptions.allDayEvent_textColor = \"white\";\n\nconst CreateCalendarAgenda = (options: Partial) => {\n return Object.assign({}, defaultOptions, defaultTextOptions, options, {\n wid: makeId(),\n type: \"CalendarAgenda\",\n name: DEFAULT_LAYER_NAMES.CalendarAgenda,\n });\n};\n\nexport default CreateCalendarAgenda;\n","// these aren't really private, but nor are they really useful to document\n\n/**\n * @private\n */\nclass LuxonError extends Error {}\n\n/**\n * @private\n */\nexport class InvalidDateTimeError extends LuxonError {\n constructor(reason) {\n super(`Invalid DateTime: ${reason.toMessage()}`);\n }\n}\n\n/**\n * @private\n */\nexport class InvalidIntervalError extends LuxonError {\n constructor(reason) {\n super(`Invalid Interval: ${reason.toMessage()}`);\n }\n}\n\n/**\n * @private\n */\nexport class InvalidDurationError extends LuxonError {\n constructor(reason) {\n super(`Invalid Duration: ${reason.toMessage()}`);\n }\n}\n\n/**\n * @private\n */\nexport class ConflictingSpecificationError extends LuxonError {}\n\n/**\n * @private\n */\nexport class InvalidUnitError extends LuxonError {\n constructor(unit) {\n super(`Invalid unit ${unit}`);\n }\n}\n\n/**\n * @private\n */\nexport class InvalidArgumentError extends LuxonError {}\n\n/**\n * @private\n */\nexport class ZoneIsAbstractError extends LuxonError {\n constructor() {\n super(\"Zone is an abstract class\");\n }\n}\n","/**\n * @private\n */\n\nconst n = \"numeric\",\n s = \"short\",\n l = \"long\";\n\nexport const DATE_SHORT = {\n year: n,\n month: n,\n day: n\n};\n\nexport const DATE_MED = {\n year: n,\n month: s,\n day: n\n};\n\nexport const DATE_MED_WITH_WEEKDAY = {\n year: n,\n month: s,\n day: n,\n weekday: s\n};\n\nexport const DATE_FULL = {\n year: n,\n month: l,\n day: n\n};\n\nexport const DATE_HUGE = {\n year: n,\n month: l,\n day: n,\n weekday: l\n};\n\nexport const TIME_SIMPLE = {\n hour: n,\n minute: n\n};\n\nexport const TIME_WITH_SECONDS = {\n hour: n,\n minute: n,\n second: n\n};\n\nexport const TIME_WITH_SHORT_OFFSET = {\n hour: n,\n minute: n,\n second: n,\n timeZoneName: s\n};\n\nexport const TIME_WITH_LONG_OFFSET = {\n hour: n,\n minute: n,\n second: n,\n timeZoneName: l\n};\n\nexport const TIME_24_SIMPLE = {\n hour: n,\n minute: n,\n hour12: false\n};\n\n/**\n * {@link toLocaleString}; format like '09:30:23', always 24-hour.\n */\nexport const TIME_24_WITH_SECONDS = {\n hour: n,\n minute: n,\n second: n,\n hour12: false\n};\n\n/**\n * {@link toLocaleString}; format like '09:30:23 EDT', always 24-hour.\n */\nexport const TIME_24_WITH_SHORT_OFFSET = {\n hour: n,\n minute: n,\n second: n,\n hour12: false,\n timeZoneName: s\n};\n\n/**\n * {@link toLocaleString}; format like '09:30:23 Eastern Daylight Time', always 24-hour.\n */\nexport const TIME_24_WITH_LONG_OFFSET = {\n hour: n,\n minute: n,\n second: n,\n hour12: false,\n timeZoneName: l\n};\n\n/**\n * {@link toLocaleString}; format like '10/14/1983, 9:30 AM'. Only 12-hour if the locale is.\n */\nexport const DATETIME_SHORT = {\n year: n,\n month: n,\n day: n,\n hour: n,\n minute: n\n};\n\n/**\n * {@link toLocaleString}; format like '10/14/1983, 9:30:33 AM'. Only 12-hour if the locale is.\n */\nexport const DATETIME_SHORT_WITH_SECONDS = {\n year: n,\n month: n,\n day: n,\n hour: n,\n minute: n,\n second: n\n};\n\nexport const DATETIME_MED = {\n year: n,\n month: s,\n day: n,\n hour: n,\n minute: n\n};\n\nexport const DATETIME_MED_WITH_SECONDS = {\n year: n,\n month: s,\n day: n,\n hour: n,\n minute: n,\n second: n\n};\n\nexport const DATETIME_MED_WITH_WEEKDAY = {\n year: n,\n month: s,\n day: n,\n weekday: s,\n hour: n,\n minute: n\n};\n\nexport const DATETIME_FULL = {\n year: n,\n month: l,\n day: n,\n hour: n,\n minute: n,\n timeZoneName: s\n};\n\nexport const DATETIME_FULL_WITH_SECONDS = {\n year: n,\n month: l,\n day: n,\n hour: n,\n minute: n,\n second: n,\n timeZoneName: s\n};\n\nexport const DATETIME_HUGE = {\n year: n,\n month: l,\n day: n,\n weekday: l,\n hour: n,\n minute: n,\n timeZoneName: l\n};\n\nexport const DATETIME_HUGE_WITH_SECONDS = {\n year: n,\n month: l,\n day: n,\n weekday: l,\n hour: n,\n minute: n,\n second: n,\n timeZoneName: l\n};\n","/*\n This is just a junk drawer, containing anything used across multiple classes.\n Because Luxon is small(ish), this should stay small and we won't worry about splitting\n it up into, say, parsingUtil.js and basicUtil.js and so on. But they are divided up by feature area.\n*/\n\nimport { InvalidArgumentError } from \"../errors.js\";\n\n/**\n * @private\n */\n\n// TYPES\n\nexport function isUndefined(o) {\n return typeof o === \"undefined\";\n}\n\nexport function isNumber(o) {\n return typeof o === \"number\";\n}\n\nexport function isInteger(o) {\n return typeof o === \"number\" && o % 1 === 0;\n}\n\nexport function isString(o) {\n return typeof o === \"string\";\n}\n\nexport function isDate(o) {\n return Object.prototype.toString.call(o) === \"[object Date]\";\n}\n\n// CAPABILITIES\n\nexport function hasIntl() {\n try {\n return typeof Intl !== \"undefined\" && Intl.DateTimeFormat;\n } catch (e) {\n return false;\n }\n}\n\nexport function hasFormatToParts() {\n return !isUndefined(Intl.DateTimeFormat.prototype.formatToParts);\n}\n\nexport function hasRelative() {\n try {\n return typeof Intl !== \"undefined\" && !!Intl.RelativeTimeFormat;\n } catch (e) {\n return false;\n }\n}\n\n// OBJECTS AND ARRAYS\n\nexport function maybeArray(thing) {\n return Array.isArray(thing) ? thing : [thing];\n}\n\nexport function bestBy(arr, by, compare) {\n if (arr.length === 0) {\n return undefined;\n }\n return arr.reduce((best, next) => {\n const pair = [by(next), next];\n if (!best) {\n return pair;\n } else if (compare(best[0], pair[0]) === best[0]) {\n return best;\n } else {\n return pair;\n }\n }, null)[1];\n}\n\nexport function pick(obj, keys) {\n return keys.reduce((a, k) => {\n a[k] = obj[k];\n return a;\n }, {});\n}\n\nexport function hasOwnProperty(obj, prop) {\n return Object.prototype.hasOwnProperty.call(obj, prop);\n}\n\n// NUMBERS AND STRINGS\n\nexport function integerBetween(thing, bottom, top) {\n return isInteger(thing) && thing >= bottom && thing <= top;\n}\n\n// x % n but takes the sign of n instead of x\nexport function floorMod(x, n) {\n return x - n * Math.floor(x / n);\n}\n\nexport function padStart(input, n = 2) {\n const minus = input < 0 ? \"-\" : \"\";\n const target = minus ? input * -1 : input;\n let result;\n\n if (target.toString().length < n) {\n result = (\"0\".repeat(n) + target).slice(-n);\n } else {\n result = target.toString();\n }\n\n return `${minus}${result}`;\n}\n\nexport function parseInteger(string) {\n if (isUndefined(string) || string === null || string === \"\") {\n return undefined;\n } else {\n return parseInt(string, 10);\n }\n}\n\nexport function parseMillis(fraction) {\n // Return undefined (instead of 0) in these cases, where fraction is not set\n if (isUndefined(fraction) || fraction === null || fraction === \"\") {\n return undefined;\n } else {\n const f = parseFloat(\"0.\" + fraction) * 1000;\n return Math.floor(f);\n }\n}\n\nexport function roundTo(number, digits, towardZero = false) {\n const factor = 10 ** digits,\n rounder = towardZero ? Math.trunc : Math.round;\n return rounder(number * factor) / factor;\n}\n\n// DATE BASICS\n\nexport function isLeapYear(year) {\n return year % 4 === 0 && (year % 100 !== 0 || year % 400 === 0);\n}\n\nexport function daysInYear(year) {\n return isLeapYear(year) ? 366 : 365;\n}\n\nexport function daysInMonth(year, month) {\n const modMonth = floorMod(month - 1, 12) + 1,\n modYear = year + (month - modMonth) / 12;\n\n if (modMonth === 2) {\n return isLeapYear(modYear) ? 29 : 28;\n } else {\n return [31, null, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31][modMonth - 1];\n }\n}\n\n// covert a calendar object to a local timestamp (epoch, but with the offset baked in)\nexport function objToLocalTS(obj) {\n let d = Date.UTC(\n obj.year,\n obj.month - 1,\n obj.day,\n obj.hour,\n obj.minute,\n obj.second,\n obj.millisecond\n );\n\n // for legacy reasons, years between 0 and 99 are interpreted as 19XX; revert that\n if (obj.year < 100 && obj.year >= 0) {\n d = new Date(d);\n d.setUTCFullYear(d.getUTCFullYear() - 1900);\n }\n return +d;\n}\n\nexport function weeksInWeekYear(weekYear) {\n const p1 =\n (weekYear +\n Math.floor(weekYear / 4) -\n Math.floor(weekYear / 100) +\n Math.floor(weekYear / 400)) %\n 7,\n last = weekYear - 1,\n p2 = (last + Math.floor(last / 4) - Math.floor(last / 100) + Math.floor(last / 400)) % 7;\n return p1 === 4 || p2 === 3 ? 53 : 52;\n}\n\nexport function untruncateYear(year) {\n if (year > 99) {\n return year;\n } else return year > 60 ? 1900 + year : 2000 + year;\n}\n\n// PARSING\n\nexport function parseZoneInfo(ts, offsetFormat, locale, timeZone = null) {\n const date = new Date(ts),\n intlOpts = {\n hour12: false,\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n hour: \"2-digit\",\n minute: \"2-digit\"\n };\n\n if (timeZone) {\n intlOpts.timeZone = timeZone;\n }\n\n const modified = Object.assign({ timeZoneName: offsetFormat }, intlOpts),\n intl = hasIntl();\n\n if (intl && hasFormatToParts()) {\n const parsed = new Intl.DateTimeFormat(locale, modified)\n .formatToParts(date)\n .find(m => m.type.toLowerCase() === \"timezonename\");\n return parsed ? parsed.value : null;\n } else if (intl) {\n // this probably doesn't work for all locales\n const without = new Intl.DateTimeFormat(locale, intlOpts).format(date),\n included = new Intl.DateTimeFormat(locale, modified).format(date),\n diffed = included.substring(without.length),\n trimmed = diffed.replace(/^[, \\u200e]+/, \"\");\n return trimmed;\n } else {\n return null;\n }\n}\n\n// signedOffset('-5', '30') -> -330\nexport function signedOffset(offHourStr, offMinuteStr) {\n let offHour = parseInt(offHourStr, 10);\n\n // don't || this because we want to preserve -0\n if (Number.isNaN(offHour)) {\n offHour = 0;\n }\n\n const offMin = parseInt(offMinuteStr, 10) || 0,\n offMinSigned = offHour < 0 || Object.is(offHour, -0) ? -offMin : offMin;\n return offHour * 60 + offMinSigned;\n}\n\n// COERCION\n\nexport function asNumber(value) {\n const numericValue = Number(value);\n if (typeof value === \"boolean\" || value === \"\" || Number.isNaN(numericValue))\n throw new InvalidArgumentError(`Invalid unit value ${value}`);\n return numericValue;\n}\n\nexport function normalizeObject(obj, normalizer, nonUnitKeys) {\n const normalized = {};\n for (const u in obj) {\n if (hasOwnProperty(obj, u)) {\n if (nonUnitKeys.indexOf(u) >= 0) continue;\n const v = obj[u];\n if (v === undefined || v === null) continue;\n normalized[normalizer(u)] = asNumber(v);\n }\n }\n return normalized;\n}\n\nexport function formatOffset(offset, format) {\n const hours = Math.trunc(Math.abs(offset / 60)),\n minutes = Math.trunc(Math.abs(offset % 60)),\n sign = offset >= 0 ? \"+\" : \"-\";\n\n switch (format) {\n case \"short\":\n return `${sign}${padStart(hours, 2)}:${padStart(minutes, 2)}`;\n case \"narrow\":\n return `${sign}${hours}${minutes > 0 ? `:${minutes}` : \"\"}`;\n case \"techie\":\n return `${sign}${padStart(hours, 2)}${padStart(minutes, 2)}`;\n default:\n throw new RangeError(`Value format ${format} is out of range for property format`);\n }\n}\n\nexport function timeObject(obj) {\n return pick(obj, [\"hour\", \"minute\", \"second\", \"millisecond\"]);\n}\n\nexport const ianaRegex = /[A-Za-z_+-]{1,256}(:?\\/[A-Za-z_+-]{1,256}(\\/[A-Za-z_+-]{1,256})?)?/;\n","import * as Formats from \"./formats.js\";\nimport { pick } from \"./util.js\";\n\nfunction stringify(obj) {\n return JSON.stringify(obj, Object.keys(obj).sort());\n}\n\n/**\n * @private\n */\n\nexport const monthsLong = [\n \"January\",\n \"February\",\n \"March\",\n \"April\",\n \"May\",\n \"June\",\n \"July\",\n \"August\",\n \"September\",\n \"October\",\n \"November\",\n \"December\"\n];\n\nexport const monthsShort = [\n \"Jan\",\n \"Feb\",\n \"Mar\",\n \"Apr\",\n \"May\",\n \"Jun\",\n \"Jul\",\n \"Aug\",\n \"Sep\",\n \"Oct\",\n \"Nov\",\n \"Dec\"\n];\n\nexport const monthsNarrow = [\"J\", \"F\", \"M\", \"A\", \"M\", \"J\", \"J\", \"A\", \"S\", \"O\", \"N\", \"D\"];\n\nexport function months(length) {\n switch (length) {\n case \"narrow\":\n return [...monthsNarrow];\n case \"short\":\n return [...monthsShort];\n case \"long\":\n return [...monthsLong];\n case \"numeric\":\n return [\"1\", \"2\", \"3\", \"4\", \"5\", \"6\", \"7\", \"8\", \"9\", \"10\", \"11\", \"12\"];\n case \"2-digit\":\n return [\"01\", \"02\", \"03\", \"04\", \"05\", \"06\", \"07\", \"08\", \"09\", \"10\", \"11\", \"12\"];\n default:\n return null;\n }\n}\n\nexport const weekdaysLong = [\n \"Monday\",\n \"Tuesday\",\n \"Wednesday\",\n \"Thursday\",\n \"Friday\",\n \"Saturday\",\n \"Sunday\"\n];\n\nexport const weekdaysShort = [\"Mon\", \"Tue\", \"Wed\", \"Thu\", \"Fri\", \"Sat\", \"Sun\"];\n\nexport const weekdaysNarrow = [\"M\", \"T\", \"W\", \"T\", \"F\", \"S\", \"S\"];\n\nexport function weekdays(length) {\n switch (length) {\n case \"narrow\":\n return [...weekdaysNarrow];\n case \"short\":\n return [...weekdaysShort];\n case \"long\":\n return [...weekdaysLong];\n case \"numeric\":\n return [\"1\", \"2\", \"3\", \"4\", \"5\", \"6\", \"7\"];\n default:\n return null;\n }\n}\n\nexport const meridiems = [\"AM\", \"PM\"];\n\nexport const erasLong = [\"Before Christ\", \"Anno Domini\"];\n\nexport const erasShort = [\"BC\", \"AD\"];\n\nexport const erasNarrow = [\"B\", \"A\"];\n\nexport function eras(length) {\n switch (length) {\n case \"narrow\":\n return [...erasNarrow];\n case \"short\":\n return [...erasShort];\n case \"long\":\n return [...erasLong];\n default:\n return null;\n }\n}\n\nexport function meridiemForDateTime(dt) {\n return meridiems[dt.hour < 12 ? 0 : 1];\n}\n\nexport function weekdayForDateTime(dt, length) {\n return weekdays(length)[dt.weekday - 1];\n}\n\nexport function monthForDateTime(dt, length) {\n return months(length)[dt.month - 1];\n}\n\nexport function eraForDateTime(dt, length) {\n return eras(length)[dt.year < 0 ? 0 : 1];\n}\n\nexport function formatRelativeTime(unit, count, numeric = \"always\", narrow = false) {\n const units = {\n years: [\"year\", \"yr.\"],\n quarters: [\"quarter\", \"qtr.\"],\n months: [\"month\", \"mo.\"],\n weeks: [\"week\", \"wk.\"],\n days: [\"day\", \"day\", \"days\"],\n hours: [\"hour\", \"hr.\"],\n minutes: [\"minute\", \"min.\"],\n seconds: [\"second\", \"sec.\"]\n };\n\n const lastable = [\"hours\", \"minutes\", \"seconds\"].indexOf(unit) === -1;\n\n if (numeric === \"auto\" && lastable) {\n const isDay = unit === \"days\";\n switch (count) {\n case 1:\n return isDay ? \"tomorrow\" : `next ${units[unit][0]}`;\n case -1:\n return isDay ? \"yesterday\" : `last ${units[unit][0]}`;\n case 0:\n return isDay ? \"today\" : `this ${units[unit][0]}`;\n default: // fall through\n }\n }\n\n const isInPast = Object.is(count, -0) || count < 0,\n fmtValue = Math.abs(count),\n singular = fmtValue === 1,\n lilUnits = units[unit],\n fmtUnit = narrow\n ? singular\n ? lilUnits[1]\n : lilUnits[2] || lilUnits[1]\n : singular\n ? units[unit][0]\n : unit;\n return isInPast ? `${fmtValue} ${fmtUnit} ago` : `in ${fmtValue} ${fmtUnit}`;\n}\n\nexport function formatString(knownFormat) {\n // these all have the offsets removed because we don't have access to them\n // without all the intl stuff this is backfilling\n const filtered = pick(knownFormat, [\n \"weekday\",\n \"era\",\n \"year\",\n \"month\",\n \"day\",\n \"hour\",\n \"minute\",\n \"second\",\n \"timeZoneName\",\n \"hour12\"\n ]),\n key = stringify(filtered),\n dateTimeHuge = \"EEEE, LLLL d, yyyy, h:mm a\";\n switch (key) {\n case stringify(Formats.DATE_SHORT):\n return \"M/d/yyyy\";\n case stringify(Formats.DATE_MED):\n return \"LLL d, yyyy\";\n case stringify(Formats.DATE_MED_WITH_WEEKDAY):\n return \"EEE, LLL d, yyyy\";\n case stringify(Formats.DATE_FULL):\n return \"LLLL d, yyyy\";\n case stringify(Formats.DATE_HUGE):\n return \"EEEE, LLLL d, yyyy\";\n case stringify(Formats.TIME_SIMPLE):\n return \"h:mm a\";\n case stringify(Formats.TIME_WITH_SECONDS):\n return \"h:mm:ss a\";\n case stringify(Formats.TIME_WITH_SHORT_OFFSET):\n return \"h:mm a\";\n case stringify(Formats.TIME_WITH_LONG_OFFSET):\n return \"h:mm a\";\n case stringify(Formats.TIME_24_SIMPLE):\n return \"HH:mm\";\n case stringify(Formats.TIME_24_WITH_SECONDS):\n return \"HH:mm:ss\";\n case stringify(Formats.TIME_24_WITH_SHORT_OFFSET):\n return \"HH:mm\";\n case stringify(Formats.TIME_24_WITH_LONG_OFFSET):\n return \"HH:mm\";\n case stringify(Formats.DATETIME_SHORT):\n return \"M/d/yyyy, h:mm a\";\n case stringify(Formats.DATETIME_MED):\n return \"LLL d, yyyy, h:mm a\";\n case stringify(Formats.DATETIME_FULL):\n return \"LLLL d, yyyy, h:mm a\";\n case stringify(Formats.DATETIME_HUGE):\n return dateTimeHuge;\n case stringify(Formats.DATETIME_SHORT_WITH_SECONDS):\n return \"M/d/yyyy, h:mm:ss a\";\n case stringify(Formats.DATETIME_MED_WITH_SECONDS):\n return \"LLL d, yyyy, h:mm:ss a\";\n case stringify(Formats.DATETIME_MED_WITH_WEEKDAY):\n return \"EEE, d LLL yyyy, h:mm a\";\n case stringify(Formats.DATETIME_FULL_WITH_SECONDS):\n return \"LLLL d, yyyy, h:mm:ss a\";\n case stringify(Formats.DATETIME_HUGE_WITH_SECONDS):\n return \"EEEE, LLLL d, yyyy, h:mm:ss a\";\n default:\n return dateTimeHuge;\n }\n}\n","import * as English from \"./english.js\";\nimport * as Formats from \"./formats.js\";\nimport { hasFormatToParts, padStart } from \"./util.js\";\n\nfunction stringifyTokens(splits, tokenToString) {\n let s = \"\";\n for (const token of splits) {\n if (token.literal) {\n s += token.val;\n } else {\n s += tokenToString(token.val);\n }\n }\n return s;\n}\n\nconst macroTokenToFormatOpts = {\n D: Formats.DATE_SHORT,\n DD: Formats.DATE_MED,\n DDD: Formats.DATE_FULL,\n DDDD: Formats.DATE_HUGE,\n t: Formats.TIME_SIMPLE,\n tt: Formats.TIME_WITH_SECONDS,\n ttt: Formats.TIME_WITH_SHORT_OFFSET,\n tttt: Formats.TIME_WITH_LONG_OFFSET,\n T: Formats.TIME_24_SIMPLE,\n TT: Formats.TIME_24_WITH_SECONDS,\n TTT: Formats.TIME_24_WITH_SHORT_OFFSET,\n TTTT: Formats.TIME_24_WITH_LONG_OFFSET,\n f: Formats.DATETIME_SHORT,\n ff: Formats.DATETIME_MED,\n fff: Formats.DATETIME_FULL,\n ffff: Formats.DATETIME_HUGE,\n F: Formats.DATETIME_SHORT_WITH_SECONDS,\n FF: Formats.DATETIME_MED_WITH_SECONDS,\n FFF: Formats.DATETIME_FULL_WITH_SECONDS,\n FFFF: Formats.DATETIME_HUGE_WITH_SECONDS\n};\n\n/**\n * @private\n */\n\nexport default class Formatter {\n static create(locale, opts = {}) {\n return new Formatter(locale, opts);\n }\n\n static parseFormat(fmt) {\n let current = null,\n currentFull = \"\",\n bracketed = false;\n const splits = [];\n for (let i = 0; i < fmt.length; i++) {\n const c = fmt.charAt(i);\n if (c === \"'\") {\n if (currentFull.length > 0) {\n splits.push({ literal: bracketed, val: currentFull });\n }\n current = null;\n currentFull = \"\";\n bracketed = !bracketed;\n } else if (bracketed) {\n currentFull += c;\n } else if (c === current) {\n currentFull += c;\n } else {\n if (currentFull.length > 0) {\n splits.push({ literal: false, val: currentFull });\n }\n currentFull = c;\n current = c;\n }\n }\n\n if (currentFull.length > 0) {\n splits.push({ literal: bracketed, val: currentFull });\n }\n\n return splits;\n }\n\n static macroTokenToFormatOpts(token) {\n return macroTokenToFormatOpts[token];\n }\n\n constructor(locale, formatOpts) {\n this.opts = formatOpts;\n this.loc = locale;\n this.systemLoc = null;\n }\n\n formatWithSystemDefault(dt, opts) {\n if (this.systemLoc === null) {\n this.systemLoc = this.loc.redefaultToSystem();\n }\n const df = this.systemLoc.dtFormatter(dt, Object.assign({}, this.opts, opts));\n return df.format();\n }\n\n formatDateTime(dt, opts = {}) {\n const df = this.loc.dtFormatter(dt, Object.assign({}, this.opts, opts));\n return df.format();\n }\n\n formatDateTimeParts(dt, opts = {}) {\n const df = this.loc.dtFormatter(dt, Object.assign({}, this.opts, opts));\n return df.formatToParts();\n }\n\n resolvedOptions(dt, opts = {}) {\n const df = this.loc.dtFormatter(dt, Object.assign({}, this.opts, opts));\n return df.resolvedOptions();\n }\n\n num(n, p = 0) {\n // we get some perf out of doing this here, annoyingly\n if (this.opts.forceSimple) {\n return padStart(n, p);\n }\n\n const opts = Object.assign({}, this.opts);\n\n if (p > 0) {\n opts.padTo = p;\n }\n\n return this.loc.numberFormatter(opts).format(n);\n }\n\n formatDateTimeFromString(dt, fmt) {\n const knownEnglish = this.loc.listingMode() === \"en\",\n useDateTimeFormatter =\n this.loc.outputCalendar && this.loc.outputCalendar !== \"gregory\" && hasFormatToParts(),\n string = (opts, extract) => this.loc.extract(dt, opts, extract),\n formatOffset = opts => {\n if (dt.isOffsetFixed && dt.offset === 0 && opts.allowZ) {\n return \"Z\";\n }\n\n return dt.isValid ? dt.zone.formatOffset(dt.ts, opts.format) : \"\";\n },\n meridiem = () =>\n knownEnglish\n ? English.meridiemForDateTime(dt)\n : string({ hour: \"numeric\", hour12: true }, \"dayperiod\"),\n month = (length, standalone) =>\n knownEnglish\n ? English.monthForDateTime(dt, length)\n : string(standalone ? { month: length } : { month: length, day: \"numeric\" }, \"month\"),\n weekday = (length, standalone) =>\n knownEnglish\n ? English.weekdayForDateTime(dt, length)\n : string(\n standalone ? { weekday: length } : { weekday: length, month: \"long\", day: \"numeric\" },\n \"weekday\"\n ),\n maybeMacro = token => {\n const formatOpts = Formatter.macroTokenToFormatOpts(token);\n if (formatOpts) {\n return this.formatWithSystemDefault(dt, formatOpts);\n } else {\n return token;\n }\n },\n era = length =>\n knownEnglish ? English.eraForDateTime(dt, length) : string({ era: length }, \"era\"),\n tokenToString = token => {\n // Where possible: http://cldr.unicode.org/translation/date-time-1/date-time#TOC-Standalone-vs.-Format-Styles\n switch (token) {\n // ms\n case \"S\":\n return this.num(dt.millisecond);\n case \"u\":\n // falls through\n case \"SSS\":\n return this.num(dt.millisecond, 3);\n // seconds\n case \"s\":\n return this.num(dt.second);\n case \"ss\":\n return this.num(dt.second, 2);\n // minutes\n case \"m\":\n return this.num(dt.minute);\n case \"mm\":\n return this.num(dt.minute, 2);\n // hours\n case \"h\":\n return this.num(dt.hour % 12 === 0 ? 12 : dt.hour % 12);\n case \"hh\":\n return this.num(dt.hour % 12 === 0 ? 12 : dt.hour % 12, 2);\n case \"H\":\n return this.num(dt.hour);\n case \"HH\":\n return this.num(dt.hour, 2);\n // offset\n case \"Z\":\n // like +6\n return formatOffset({ format: \"narrow\", allowZ: this.opts.allowZ });\n case \"ZZ\":\n // like +06:00\n return formatOffset({ format: \"short\", allowZ: this.opts.allowZ });\n case \"ZZZ\":\n // like +0600\n return formatOffset({ format: \"techie\", allowZ: this.opts.allowZ });\n case \"ZZZZ\":\n // like EST\n return dt.zone.offsetName(dt.ts, { format: \"short\", locale: this.loc.locale });\n case \"ZZZZZ\":\n // like Eastern Standard Time\n return dt.zone.offsetName(dt.ts, { format: \"long\", locale: this.loc.locale });\n // zone\n case \"z\":\n // like America/New_York\n return dt.zoneName;\n // meridiems\n case \"a\":\n return meridiem();\n // dates\n case \"d\":\n return useDateTimeFormatter ? string({ day: \"numeric\" }, \"day\") : this.num(dt.day);\n case \"dd\":\n return useDateTimeFormatter ? string({ day: \"2-digit\" }, \"day\") : this.num(dt.day, 2);\n // weekdays - standalone\n case \"c\":\n // like 1\n return this.num(dt.weekday);\n case \"ccc\":\n // like 'Tues'\n return weekday(\"short\", true);\n case \"cccc\":\n // like 'Tuesday'\n return weekday(\"long\", true);\n case \"ccccc\":\n // like 'T'\n return weekday(\"narrow\", true);\n // weekdays - format\n case \"E\":\n // like 1\n return this.num(dt.weekday);\n case \"EEE\":\n // like 'Tues'\n return weekday(\"short\", false);\n case \"EEEE\":\n // like 'Tuesday'\n return weekday(\"long\", false);\n case \"EEEEE\":\n // like 'T'\n return weekday(\"narrow\", false);\n // months - standalone\n case \"L\":\n // like 1\n return useDateTimeFormatter\n ? string({ month: \"numeric\", day: \"numeric\" }, \"month\")\n : this.num(dt.month);\n case \"LL\":\n // like 01, doesn't seem to work\n return useDateTimeFormatter\n ? string({ month: \"2-digit\", day: \"numeric\" }, \"month\")\n : this.num(dt.month, 2);\n case \"LLL\":\n // like Jan\n return month(\"short\", true);\n case \"LLLL\":\n // like January\n return month(\"long\", true);\n case \"LLLLL\":\n // like J\n return month(\"narrow\", true);\n // months - format\n case \"M\":\n // like 1\n return useDateTimeFormatter\n ? string({ month: \"numeric\" }, \"month\")\n : this.num(dt.month);\n case \"MM\":\n // like 01\n return useDateTimeFormatter\n ? string({ month: \"2-digit\" }, \"month\")\n : this.num(dt.month, 2);\n case \"MMM\":\n // like Jan\n return month(\"short\", false);\n case \"MMMM\":\n // like January\n return month(\"long\", false);\n case \"MMMMM\":\n // like J\n return month(\"narrow\", false);\n // years\n case \"y\":\n // like 2014\n return useDateTimeFormatter ? string({ year: \"numeric\" }, \"year\") : this.num(dt.year);\n case \"yy\":\n // like 14\n return useDateTimeFormatter\n ? string({ year: \"2-digit\" }, \"year\")\n : this.num(dt.year.toString().slice(-2), 2);\n case \"yyyy\":\n // like 0012\n return useDateTimeFormatter\n ? string({ year: \"numeric\" }, \"year\")\n : this.num(dt.year, 4);\n case \"yyyyyy\":\n // like 000012\n return useDateTimeFormatter\n ? string({ year: \"numeric\" }, \"year\")\n : this.num(dt.year, 6);\n // eras\n case \"G\":\n // like AD\n return era(\"short\");\n case \"GG\":\n // like Anno Domini\n return era(\"long\");\n case \"GGGGG\":\n return era(\"narrow\");\n case \"kk\":\n return this.num(dt.weekYear.toString().slice(-2), 2);\n case \"kkkk\":\n return this.num(dt.weekYear, 4);\n case \"W\":\n return this.num(dt.weekNumber);\n case \"WW\":\n return this.num(dt.weekNumber, 2);\n case \"o\":\n return this.num(dt.ordinal);\n case \"ooo\":\n return this.num(dt.ordinal, 3);\n case \"q\":\n // like 1\n return this.num(dt.quarter);\n case \"qq\":\n // like 01\n return this.num(dt.quarter, 2);\n case \"X\":\n return this.num(Math.floor(dt.ts / 1000));\n case \"x\":\n return this.num(dt.ts);\n default:\n return maybeMacro(token);\n }\n };\n\n return stringifyTokens(Formatter.parseFormat(fmt), tokenToString);\n }\n\n formatDurationFromString(dur, fmt) {\n const tokenToField = token => {\n switch (token[0]) {\n case \"S\":\n return \"millisecond\";\n case \"s\":\n return \"second\";\n case \"m\":\n return \"minute\";\n case \"h\":\n return \"hour\";\n case \"d\":\n return \"day\";\n case \"M\":\n return \"month\";\n case \"y\":\n return \"year\";\n default:\n return null;\n }\n },\n tokenToString = lildur => token => {\n const mapped = tokenToField(token);\n if (mapped) {\n return this.num(lildur.get(mapped), token.length);\n } else {\n return token;\n }\n },\n tokens = Formatter.parseFormat(fmt),\n realTokens = tokens.reduce(\n (found, { literal, val }) => (literal ? found : found.concat(val)),\n []\n ),\n collapsed = dur.shiftTo(...realTokens.map(tokenToField).filter(t => t));\n return stringifyTokens(tokens, tokenToString(collapsed));\n }\n}\n","export default class Invalid {\n constructor(reason, explanation) {\n this.reason = reason;\n this.explanation = explanation;\n }\n\n toMessage() {\n if (this.explanation) {\n return `${this.reason}: ${this.explanation}`;\n } else {\n return this.reason;\n }\n }\n}\n","/* eslint no-unused-vars: \"off\" */\nimport { ZoneIsAbstractError } from \"./errors.js\";\n\n/**\n * @interface\n */\nexport default class Zone {\n /**\n * The type of zone\n * @abstract\n * @type {string}\n */\n get type() {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * The name of this zone.\n * @abstract\n * @type {string}\n */\n get name() {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Returns whether the offset is known to be fixed for the whole year.\n * @abstract\n * @type {boolean}\n */\n get universal() {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Returns the offset's common name (such as EST) at the specified timestamp\n * @abstract\n * @param {number} ts - Epoch milliseconds for which to get the name\n * @param {Object} opts - Options to affect the format\n * @param {string} opts.format - What style of offset to return. Accepts 'long' or 'short'.\n * @param {string} opts.locale - What locale to return the offset name in.\n * @return {string}\n */\n offsetName(ts, opts) {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Returns the offset's value as a string\n * @abstract\n * @param {number} ts - Epoch milliseconds for which to get the offset\n * @param {string} format - What style of offset to return.\n * Accepts 'narrow', 'short', or 'techie'. Returning '+6', '+06:00', or '+0600' respectively\n * @return {string}\n */\n formatOffset(ts, format) {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Return the offset in minutes for this zone at the specified timestamp.\n * @abstract\n * @param {number} ts - Epoch milliseconds for which to compute the offset\n * @return {number}\n */\n offset(ts) {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Return whether this Zone is equal to another zone\n * @abstract\n * @param {Zone} otherZone - the zone to compare\n * @return {boolean}\n */\n equals(otherZone) {\n throw new ZoneIsAbstractError();\n }\n\n /**\n * Return whether this Zone is valid.\n * @abstract\n * @type {boolean}\n */\n get isValid() {\n throw new ZoneIsAbstractError();\n }\n}\n","import { formatOffset, parseZoneInfo, hasIntl } from \"../impl/util.js\";\nimport Zone from \"../zone.js\";\n\nlet singleton = null;\n\n/**\n * Represents the local zone for this JavaScript environment.\n * @implements {Zone}\n */\nexport default class LocalZone extends Zone {\n /**\n * Get a singleton instance of the local zone\n * @return {LocalZone}\n */\n static get instance() {\n if (singleton === null) {\n singleton = new LocalZone();\n }\n return singleton;\n }\n\n /** @override **/\n get type() {\n return \"local\";\n }\n\n /** @override **/\n get name() {\n if (hasIntl()) {\n return new Intl.DateTimeFormat().resolvedOptions().timeZone;\n } else return \"local\";\n }\n\n /** @override **/\n get universal() {\n return false;\n }\n\n /** @override **/\n offsetName(ts, { format, locale }) {\n return parseZoneInfo(ts, format, locale);\n }\n\n /** @override **/\n formatOffset(ts, format) {\n return formatOffset(this.offset(ts), format);\n }\n\n /** @override **/\n offset(ts) {\n return -new Date(ts).getTimezoneOffset();\n }\n\n /** @override **/\n equals(otherZone) {\n return otherZone.type === \"local\";\n }\n\n /** @override **/\n get isValid() {\n return true;\n }\n}\n","import { formatOffset, parseZoneInfo, isUndefined, ianaRegex, objToLocalTS } from \"../impl/util.js\";\nimport Zone from \"../zone.js\";\n\nconst matchingRegex = RegExp(`^${ianaRegex.source}$`);\n\nlet dtfCache = {};\nfunction makeDTF(zone) {\n if (!dtfCache[zone]) {\n dtfCache[zone] = new Intl.DateTimeFormat(\"en-US\", {\n hour12: false,\n timeZone: zone,\n year: \"numeric\",\n month: \"2-digit\",\n day: \"2-digit\",\n hour: \"2-digit\",\n minute: \"2-digit\",\n second: \"2-digit\"\n });\n }\n return dtfCache[zone];\n}\n\nconst typeToPos = {\n year: 0,\n month: 1,\n day: 2,\n hour: 3,\n minute: 4,\n second: 5\n};\n\nfunction hackyOffset(dtf, date) {\n const formatted = dtf.format(date).replace(/\\u200E/g, \"\"),\n parsed = /(\\d+)\\/(\\d+)\\/(\\d+),? (\\d+):(\\d+):(\\d+)/.exec(formatted),\n [, fMonth, fDay, fYear, fHour, fMinute, fSecond] = parsed;\n return [fYear, fMonth, fDay, fHour, fMinute, fSecond];\n}\n\nfunction partsOffset(dtf, date) {\n const formatted = dtf.formatToParts(date),\n filled = [];\n for (let i = 0; i < formatted.length; i++) {\n const { type, value } = formatted[i],\n pos = typeToPos[type];\n\n if (!isUndefined(pos)) {\n filled[pos] = parseInt(value, 10);\n }\n }\n return filled;\n}\n\nlet ianaZoneCache = {};\n/**\n * A zone identified by an IANA identifier, like America/New_York\n * @implements {Zone}\n */\nexport default class IANAZone extends Zone {\n /**\n * @param {string} name - Zone name\n * @return {IANAZone}\n */\n static create(name) {\n if (!ianaZoneCache[name]) {\n ianaZoneCache[name] = new IANAZone(name);\n }\n return ianaZoneCache[name];\n }\n\n /**\n * Reset local caches. Should only be necessary in testing scenarios.\n * @return {void}\n */\n static resetCache() {\n ianaZoneCache = {};\n dtfCache = {};\n }\n\n /**\n * Returns whether the provided string is a valid specifier. This only checks the string's format, not that the specifier identifies a known zone; see isValidZone for that.\n * @param {string} s - The string to check validity on\n * @example IANAZone.isValidSpecifier(\"America/New_York\") //=> true\n * @example IANAZone.isValidSpecifier(\"Fantasia/Castle\") //=> true\n * @example IANAZone.isValidSpecifier(\"Sport~~blorp\") //=> false\n * @return {boolean}\n */\n static isValidSpecifier(s) {\n return !!(s && s.match(matchingRegex));\n }\n\n /**\n * Returns whether the provided string identifies a real zone\n * @param {string} zone - The string to check\n * @example IANAZone.isValidZone(\"America/New_York\") //=> true\n * @example IANAZone.isValidZone(\"Fantasia/Castle\") //=> false\n * @example IANAZone.isValidZone(\"Sport~~blorp\") //=> false\n * @return {boolean}\n */\n static isValidZone(zone) {\n try {\n new Intl.DateTimeFormat(\"en-US\", { timeZone: zone }).format();\n return true;\n } catch (e) {\n return false;\n }\n }\n\n // Etc/GMT+8 -> -480\n /** @ignore */\n static parseGMTOffset(specifier) {\n if (specifier) {\n const match = specifier.match(/^Etc\\/GMT(0|[+-]\\d{1,2})$/i);\n if (match) {\n return -60 * parseInt(match[1]);\n }\n }\n return null;\n }\n\n constructor(name) {\n super();\n /** @private **/\n this.zoneName = name;\n /** @private **/\n this.valid = IANAZone.isValidZone(name);\n }\n\n /** @override **/\n get type() {\n return \"iana\";\n }\n\n /** @override **/\n get name() {\n return this.zoneName;\n }\n\n /** @override **/\n get universal() {\n return false;\n }\n\n /** @override **/\n offsetName(ts, { format, locale }) {\n return parseZoneInfo(ts, format, locale, this.name);\n }\n\n /** @override **/\n formatOffset(ts, format) {\n return formatOffset(this.offset(ts), format);\n }\n\n /** @override **/\n offset(ts) {\n const date = new Date(ts);\n\n if (isNaN(date)) return NaN;\n\n const dtf = makeDTF(this.name),\n [year, month, day, hour, minute, second] = dtf.formatToParts\n ? partsOffset(dtf, date)\n : hackyOffset(dtf, date),\n // work around https://bugs.chromium.org/p/chromium/issues/detail?id=1025564&can=2&q=%2224%3A00%22%20datetimeformat\n adjustedHour = hour === 24 ? 0 : hour;\n\n const asUTC = objToLocalTS({\n year,\n month,\n day,\n hour: adjustedHour,\n minute,\n second,\n millisecond: 0\n });\n\n let asTS = +date;\n const over = asTS % 1000;\n asTS -= over >= 0 ? over : 1000 + over;\n return (asUTC - asTS) / (60 * 1000);\n }\n\n /** @override **/\n equals(otherZone) {\n return otherZone.type === \"iana\" && otherZone.name === this.name;\n }\n\n /** @override **/\n get isValid() {\n return this.valid;\n }\n}\n","import { formatOffset, signedOffset } from \"../impl/util.js\";\nimport Zone from \"../zone.js\";\n\nlet singleton = null;\n\n/**\n * A zone with a fixed offset (meaning no DST)\n * @implements {Zone}\n */\nexport default class FixedOffsetZone extends Zone {\n /**\n * Get a singleton instance of UTC\n * @return {FixedOffsetZone}\n */\n static get utcInstance() {\n if (singleton === null) {\n singleton = new FixedOffsetZone(0);\n }\n return singleton;\n }\n\n /**\n * Get an instance with a specified offset\n * @param {number} offset - The offset in minutes\n * @return {FixedOffsetZone}\n */\n static instance(offset) {\n return offset === 0 ? FixedOffsetZone.utcInstance : new FixedOffsetZone(offset);\n }\n\n /**\n * Get an instance of FixedOffsetZone from a UTC offset string, like \"UTC+6\"\n * @param {string} s - The offset string to parse\n * @example FixedOffsetZone.parseSpecifier(\"UTC+6\")\n * @example FixedOffsetZone.parseSpecifier(\"UTC+06\")\n * @example FixedOffsetZone.parseSpecifier(\"UTC-6:00\")\n * @return {FixedOffsetZone}\n */\n static parseSpecifier(s) {\n if (s) {\n const r = s.match(/^utc(?:([+-]\\d{1,2})(?::(\\d{2}))?)?$/i);\n if (r) {\n return new FixedOffsetZone(signedOffset(r[1], r[2]));\n }\n }\n return null;\n }\n\n constructor(offset) {\n super();\n /** @private **/\n this.fixed = offset;\n }\n\n /** @override **/\n get type() {\n return \"fixed\";\n }\n\n /** @override **/\n get name() {\n return this.fixed === 0 ? \"UTC\" : `UTC${formatOffset(this.fixed, \"narrow\")}`;\n }\n\n /** @override **/\n offsetName() {\n return this.name;\n }\n\n /** @override **/\n formatOffset(ts, format) {\n return formatOffset(this.fixed, format);\n }\n\n /** @override **/\n get universal() {\n return true;\n }\n\n /** @override **/\n offset() {\n return this.fixed;\n }\n\n /** @override **/\n equals(otherZone) {\n return otherZone.type === \"fixed\" && otherZone.fixed === this.fixed;\n }\n\n /** @override **/\n get isValid() {\n return true;\n }\n}\n","import Zone from \"../zone.js\";\n\n/**\n * A zone that failed to parse. You should never need to instantiate this.\n * @implements {Zone}\n */\nexport default class InvalidZone extends Zone {\n constructor(zoneName) {\n super();\n /** @private */\n this.zoneName = zoneName;\n }\n\n /** @override **/\n get type() {\n return \"invalid\";\n }\n\n /** @override **/\n get name() {\n return this.zoneName;\n }\n\n /** @override **/\n get universal() {\n return false;\n }\n\n /** @override **/\n offsetName() {\n return null;\n }\n\n /** @override **/\n formatOffset() {\n return \"\";\n }\n\n /** @override **/\n offset() {\n return NaN;\n }\n\n /** @override **/\n equals() {\n return false;\n }\n\n /** @override **/\n get isValid() {\n return false;\n }\n}\n","/**\n * @private\n */\n\nimport Zone from \"../zone.js\";\nimport IANAZone from \"../zones/IANAZone.js\";\nimport FixedOffsetZone from \"../zones/fixedOffsetZone.js\";\nimport InvalidZone from \"../zones/invalidZone.js\";\n\nimport { isUndefined, isString, isNumber } from \"./util.js\";\n\nexport function normalizeZone(input, defaultZone) {\n let offset;\n if (isUndefined(input) || input === null) {\n return defaultZone;\n } else if (input instanceof Zone) {\n return input;\n } else if (isString(input)) {\n const lowered = input.toLowerCase();\n if (lowered === \"local\") return defaultZone;\n else if (lowered === \"utc\" || lowered === \"gmt\") return FixedOffsetZone.utcInstance;\n else if ((offset = IANAZone.parseGMTOffset(input)) != null) {\n // handle Etc/GMT-4, which V8 chokes on\n return FixedOffsetZone.instance(offset);\n } else if (IANAZone.isValidSpecifier(lowered)) return IANAZone.create(input);\n else return FixedOffsetZone.parseSpecifier(lowered) || new InvalidZone(input);\n } else if (isNumber(input)) {\n return FixedOffsetZone.instance(input);\n } else if (typeof input === \"object\" && input.offset && typeof input.offset === \"number\") {\n // This is dumb, but the instanceof check above doesn't seem to really work\n // so we're duck checking it\n return input;\n } else {\n return new InvalidZone(input);\n }\n}\n","import LocalZone from \"./zones/localZone.js\";\nimport IANAZone from \"./zones/IANAZone.js\";\nimport Locale from \"./impl/locale.js\";\n\nimport { normalizeZone } from \"./impl/zoneUtil.js\";\n\nlet now = () => Date.now(),\n defaultZone = null, // not setting this directly to LocalZone.instance bc loading order issues\n defaultLocale = null,\n defaultNumberingSystem = null,\n defaultOutputCalendar = null,\n throwOnInvalid = false;\n\n/**\n * Settings contains static getters and setters that control Luxon's overall behavior. Luxon is a simple library with few options, but the ones it does have live here.\n */\nexport default class Settings {\n /**\n * Get the callback for returning the current timestamp.\n * @type {function}\n */\n static get now() {\n return now;\n }\n\n /**\n * Set the callback for returning the current timestamp.\n * The function should return a number, which will be interpreted as an Epoch millisecond count\n * @type {function}\n * @example Settings.now = () => Date.now() + 3000 // pretend it is 3 seconds in the future\n * @example Settings.now = () => 0 // always pretend it's Jan 1, 1970 at midnight in UTC time\n */\n static set now(n) {\n now = n;\n }\n\n /**\n * Get the default time zone to create DateTimes in.\n * @type {string}\n */\n static get defaultZoneName() {\n return Settings.defaultZone.name;\n }\n\n /**\n * Set the default time zone to create DateTimes in. Does not affect existing instances.\n * @type {string}\n */\n static set defaultZoneName(z) {\n if (!z) {\n defaultZone = null;\n } else {\n defaultZone = normalizeZone(z);\n }\n }\n\n /**\n * Get the default time zone object to create DateTimes in. Does not affect existing instances.\n * @type {Zone}\n */\n static get defaultZone() {\n return defaultZone || LocalZone.instance;\n }\n\n /**\n * Get the default locale to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static get defaultLocale() {\n return defaultLocale;\n }\n\n /**\n * Set the default locale to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static set defaultLocale(locale) {\n defaultLocale = locale;\n }\n\n /**\n * Get the default numbering system to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static get defaultNumberingSystem() {\n return defaultNumberingSystem;\n }\n\n /**\n * Set the default numbering system to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static set defaultNumberingSystem(numberingSystem) {\n defaultNumberingSystem = numberingSystem;\n }\n\n /**\n * Get the default output calendar to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static get defaultOutputCalendar() {\n return defaultOutputCalendar;\n }\n\n /**\n * Set the default output calendar to create DateTimes with. Does not affect existing instances.\n * @type {string}\n */\n static set defaultOutputCalendar(outputCalendar) {\n defaultOutputCalendar = outputCalendar;\n }\n\n /**\n * Get whether Luxon will throw when it encounters invalid DateTimes, Durations, or Intervals\n * @type {boolean}\n */\n static get throwOnInvalid() {\n return throwOnInvalid;\n }\n\n /**\n * Set whether Luxon will throw when it encounters invalid DateTimes, Durations, or Intervals\n * @type {boolean}\n */\n static set throwOnInvalid(t) {\n throwOnInvalid = t;\n }\n\n /**\n * Reset Luxon's global caches. Should only be necessary in testing scenarios.\n * @return {void}\n */\n static resetCaches() {\n Locale.resetCache();\n IANAZone.resetCache();\n }\n}\n","import { hasFormatToParts, hasIntl, padStart, roundTo, hasRelative } from \"./util.js\";\nimport * as English from \"./english.js\";\nimport Settings from \"../settings.js\";\nimport DateTime from \"../datetime.js\";\nimport Formatter from \"./formatter.js\";\nimport IANAZone from \"../zones/IANAZone.js\";\n\nlet intlDTCache = {};\nfunction getCachedDTF(locString, opts = {}) {\n const key = JSON.stringify([locString, opts]);\n let dtf = intlDTCache[key];\n if (!dtf) {\n dtf = new Intl.DateTimeFormat(locString, opts);\n intlDTCache[key] = dtf;\n }\n return dtf;\n}\n\nlet intlNumCache = {};\nfunction getCachedINF(locString, opts = {}) {\n const key = JSON.stringify([locString, opts]);\n let inf = intlNumCache[key];\n if (!inf) {\n inf = new Intl.NumberFormat(locString, opts);\n intlNumCache[key] = inf;\n }\n return inf;\n}\n\nlet intlRelCache = {};\nfunction getCachedRTF(locString, opts = {}) {\n const { base, ...cacheKeyOpts } = opts; // exclude `base` from the options\n const key = JSON.stringify([locString, cacheKeyOpts]);\n let inf = intlRelCache[key];\n if (!inf) {\n inf = new Intl.RelativeTimeFormat(locString, opts);\n intlRelCache[key] = inf;\n }\n return inf;\n}\n\nlet sysLocaleCache = null;\nfunction systemLocale() {\n if (sysLocaleCache) {\n return sysLocaleCache;\n } else if (hasIntl()) {\n const computedSys = new Intl.DateTimeFormat().resolvedOptions().locale;\n // node sometimes defaults to \"und\". Override that because that is dumb\n sysLocaleCache = !computedSys || computedSys === \"und\" ? \"en-US\" : computedSys;\n return sysLocaleCache;\n } else {\n sysLocaleCache = \"en-US\";\n return sysLocaleCache;\n }\n}\n\nfunction parseLocaleString(localeStr) {\n // I really want to avoid writing a BCP 47 parser\n // see, e.g. https://github.com/wooorm/bcp-47\n // Instead, we'll do this:\n\n // a) if the string has no -u extensions, just leave it alone\n // b) if it does, use Intl to resolve everything\n // c) if Intl fails, try again without the -u\n\n const uIndex = localeStr.indexOf(\"-u-\");\n if (uIndex === -1) {\n return [localeStr];\n } else {\n let options;\n const smaller = localeStr.substring(0, uIndex);\n try {\n options = getCachedDTF(localeStr).resolvedOptions();\n } catch (e) {\n options = getCachedDTF(smaller).resolvedOptions();\n }\n\n const { numberingSystem, calendar } = options;\n // return the smaller one so that we can append the calendar and numbering overrides to it\n return [smaller, numberingSystem, calendar];\n }\n}\n\nfunction intlConfigString(localeStr, numberingSystem, outputCalendar) {\n if (hasIntl()) {\n if (outputCalendar || numberingSystem) {\n localeStr += \"-u\";\n\n if (outputCalendar) {\n localeStr += `-ca-${outputCalendar}`;\n }\n\n if (numberingSystem) {\n localeStr += `-nu-${numberingSystem}`;\n }\n return localeStr;\n } else {\n return localeStr;\n }\n } else {\n return [];\n }\n}\n\nfunction mapMonths(f) {\n const ms = [];\n for (let i = 1; i <= 12; i++) {\n const dt = DateTime.utc(2016, i, 1);\n ms.push(f(dt));\n }\n return ms;\n}\n\nfunction mapWeekdays(f) {\n const ms = [];\n for (let i = 1; i <= 7; i++) {\n const dt = DateTime.utc(2016, 11, 13 + i);\n ms.push(f(dt));\n }\n return ms;\n}\n\nfunction listStuff(loc, length, defaultOK, englishFn, intlFn) {\n const mode = loc.listingMode(defaultOK);\n\n if (mode === \"error\") {\n return null;\n } else if (mode === \"en\") {\n return englishFn(length);\n } else {\n return intlFn(length);\n }\n}\n\nfunction supportsFastNumbers(loc) {\n if (loc.numberingSystem && loc.numberingSystem !== \"latn\") {\n return false;\n } else {\n return (\n loc.numberingSystem === \"latn\" ||\n !loc.locale ||\n loc.locale.startsWith(\"en\") ||\n (hasIntl() && new Intl.DateTimeFormat(loc.intl).resolvedOptions().numberingSystem === \"latn\")\n );\n }\n}\n\n/**\n * @private\n */\n\nclass PolyNumberFormatter {\n constructor(intl, forceSimple, opts) {\n this.padTo = opts.padTo || 0;\n this.floor = opts.floor || false;\n\n if (!forceSimple && hasIntl()) {\n const intlOpts = { useGrouping: false };\n if (opts.padTo > 0) intlOpts.minimumIntegerDigits = opts.padTo;\n this.inf = getCachedINF(intl, intlOpts);\n }\n }\n\n format(i) {\n if (this.inf) {\n const fixed = this.floor ? Math.floor(i) : i;\n return this.inf.format(fixed);\n } else {\n // to match the browser's numberformatter defaults\n const fixed = this.floor ? Math.floor(i) : roundTo(i, 3);\n return padStart(fixed, this.padTo);\n }\n }\n}\n\n/**\n * @private\n */\n\nclass PolyDateFormatter {\n constructor(dt, intl, opts) {\n this.opts = opts;\n this.hasIntl = hasIntl();\n\n let z;\n if (dt.zone.universal && this.hasIntl) {\n // UTC-8 or Etc/UTC-8 are not part of tzdata, only Etc/GMT+8 and the like.\n // That is why fixed-offset TZ is set to that unless it is:\n // 1. Representing offset 0 when UTC is used to maintain previous behavior and does not become GMT.\n // 2. Unsupported by the browser:\n // - some do not support Etc/\n // - < Etc/GMT-14, > Etc/GMT+12, and 30-minute or 45-minute offsets are not part of tzdata\n const gmtOffset = -1 * (dt.offset / 60);\n const offsetZ = gmtOffset >= 0 ? `Etc/GMT+${gmtOffset}` : `Etc/GMT${gmtOffset}`;\n const isOffsetZoneSupported = IANAZone.isValidZone(offsetZ);\n if (dt.offset !== 0 && isOffsetZoneSupported) {\n z = offsetZ;\n this.dt = dt;\n } else {\n // Not all fixed-offset zones like Etc/+4:30 are present in tzdata.\n // So we have to make do. Two cases:\n // 1. The format options tell us to show the zone. We can't do that, so the best\n // we can do is format the date in UTC.\n // 2. The format options don't tell us to show the zone. Then we can adjust them\n // the time and tell the formatter to show it to us in UTC, so that the time is right\n // and the bad zone doesn't show up.\n z = \"UTC\";\n if (opts.timeZoneName) {\n this.dt = dt;\n } else {\n this.dt = dt.offset === 0 ? dt : DateTime.fromMillis(dt.ts + dt.offset * 60 * 1000);\n }\n }\n } else if (dt.zone.type === \"local\") {\n this.dt = dt;\n } else {\n this.dt = dt;\n z = dt.zone.name;\n }\n\n if (this.hasIntl) {\n const intlOpts = Object.assign({}, this.opts);\n if (z) {\n intlOpts.timeZone = z;\n }\n this.dtf = getCachedDTF(intl, intlOpts);\n }\n }\n\n format() {\n if (this.hasIntl) {\n return this.dtf.format(this.dt.toJSDate());\n } else {\n const tokenFormat = English.formatString(this.opts),\n loc = Locale.create(\"en-US\");\n return Formatter.create(loc).formatDateTimeFromString(this.dt, tokenFormat);\n }\n }\n\n formatToParts() {\n if (this.hasIntl && hasFormatToParts()) {\n return this.dtf.formatToParts(this.dt.toJSDate());\n } else {\n // This is kind of a cop out. We actually could do this for English. However, we couldn't do it for intl strings\n // and IMO it's too weird to have an uncanny valley like that\n return [];\n }\n }\n\n resolvedOptions() {\n if (this.hasIntl) {\n return this.dtf.resolvedOptions();\n } else {\n return {\n locale: \"en-US\",\n numberingSystem: \"latn\",\n outputCalendar: \"gregory\"\n };\n }\n }\n}\n\n/**\n * @private\n */\nclass PolyRelFormatter {\n constructor(intl, isEnglish, opts) {\n this.opts = Object.assign({ style: \"long\" }, opts);\n if (!isEnglish && hasRelative()) {\n this.rtf = getCachedRTF(intl, opts);\n }\n }\n\n format(count, unit) {\n if (this.rtf) {\n return this.rtf.format(count, unit);\n } else {\n return English.formatRelativeTime(unit, count, this.opts.numeric, this.opts.style !== \"long\");\n }\n }\n\n formatToParts(count, unit) {\n if (this.rtf) {\n return this.rtf.formatToParts(count, unit);\n } else {\n return [];\n }\n }\n}\n\n/**\n * @private\n */\n\nexport default class Locale {\n static fromOpts(opts) {\n return Locale.create(opts.locale, opts.numberingSystem, opts.outputCalendar, opts.defaultToEN);\n }\n\n static create(locale, numberingSystem, outputCalendar, defaultToEN = false) {\n const specifiedLocale = locale || Settings.defaultLocale,\n // the system locale is useful for human readable strings but annoying for parsing/formatting known formats\n localeR = specifiedLocale || (defaultToEN ? \"en-US\" : systemLocale()),\n numberingSystemR = numberingSystem || Settings.defaultNumberingSystem,\n outputCalendarR = outputCalendar || Settings.defaultOutputCalendar;\n return new Locale(localeR, numberingSystemR, outputCalendarR, specifiedLocale);\n }\n\n static resetCache() {\n sysLocaleCache = null;\n intlDTCache = {};\n intlNumCache = {};\n intlRelCache = {};\n }\n\n static fromObject({ locale, numberingSystem, outputCalendar } = {}) {\n return Locale.create(locale, numberingSystem, outputCalendar);\n }\n\n constructor(locale, numbering, outputCalendar, specifiedLocale) {\n const [parsedLocale, parsedNumberingSystem, parsedOutputCalendar] = parseLocaleString(locale);\n\n this.locale = parsedLocale;\n this.numberingSystem = numbering || parsedNumberingSystem || null;\n this.outputCalendar = outputCalendar || parsedOutputCalendar || null;\n this.intl = intlConfigString(this.locale, this.numberingSystem, this.outputCalendar);\n\n this.weekdaysCache = { format: {}, standalone: {} };\n this.monthsCache = { format: {}, standalone: {} };\n this.meridiemCache = null;\n this.eraCache = {};\n\n this.specifiedLocale = specifiedLocale;\n this.fastNumbersCached = null;\n }\n\n get fastNumbers() {\n if (this.fastNumbersCached == null) {\n this.fastNumbersCached = supportsFastNumbers(this);\n }\n\n return this.fastNumbersCached;\n }\n\n listingMode(defaultOK = true) {\n const intl = hasIntl(),\n hasFTP = intl && hasFormatToParts(),\n isActuallyEn = this.isEnglish(),\n hasNoWeirdness =\n (this.numberingSystem === null || this.numberingSystem === \"latn\") &&\n (this.outputCalendar === null || this.outputCalendar === \"gregory\");\n\n if (!hasFTP && !(isActuallyEn && hasNoWeirdness) && !defaultOK) {\n return \"error\";\n } else if (!hasFTP || (isActuallyEn && hasNoWeirdness)) {\n return \"en\";\n } else {\n return \"intl\";\n }\n }\n\n clone(alts) {\n if (!alts || Object.getOwnPropertyNames(alts).length === 0) {\n return this;\n } else {\n return Locale.create(\n alts.locale || this.specifiedLocale,\n alts.numberingSystem || this.numberingSystem,\n alts.outputCalendar || this.outputCalendar,\n alts.defaultToEN || false\n );\n }\n }\n\n redefaultToEN(alts = {}) {\n return this.clone(Object.assign({}, alts, { defaultToEN: true }));\n }\n\n redefaultToSystem(alts = {}) {\n return this.clone(Object.assign({}, alts, { defaultToEN: false }));\n }\n\n months(length, format = false, defaultOK = true) {\n return listStuff(this, length, defaultOK, English.months, () => {\n const intl = format ? { month: length, day: \"numeric\" } : { month: length },\n formatStr = format ? \"format\" : \"standalone\";\n if (!this.monthsCache[formatStr][length]) {\n this.monthsCache[formatStr][length] = mapMonths(dt => this.extract(dt, intl, \"month\"));\n }\n return this.monthsCache[formatStr][length];\n });\n }\n\n weekdays(length, format = false, defaultOK = true) {\n return listStuff(this, length, defaultOK, English.weekdays, () => {\n const intl = format\n ? { weekday: length, year: \"numeric\", month: \"long\", day: \"numeric\" }\n : { weekday: length },\n formatStr = format ? \"format\" : \"standalone\";\n if (!this.weekdaysCache[formatStr][length]) {\n this.weekdaysCache[formatStr][length] = mapWeekdays(dt =>\n this.extract(dt, intl, \"weekday\")\n );\n }\n return this.weekdaysCache[formatStr][length];\n });\n }\n\n meridiems(defaultOK = true) {\n return listStuff(\n this,\n undefined,\n defaultOK,\n () => English.meridiems,\n () => {\n // In theory there could be aribitrary day periods. We're gonna assume there are exactly two\n // for AM and PM. This is probably wrong, but it's makes parsing way easier.\n if (!this.meridiemCache) {\n const intl = { hour: \"numeric\", hour12: true };\n this.meridiemCache = [DateTime.utc(2016, 11, 13, 9), DateTime.utc(2016, 11, 13, 19)].map(\n dt => this.extract(dt, intl, \"dayperiod\")\n );\n }\n\n return this.meridiemCache;\n }\n );\n }\n\n eras(length, defaultOK = true) {\n return listStuff(this, length, defaultOK, English.eras, () => {\n const intl = { era: length };\n\n // This is problematic. Different calendars are going to define eras totally differently. What I need is the minimum set of dates\n // to definitely enumerate them.\n if (!this.eraCache[length]) {\n this.eraCache[length] = [DateTime.utc(-40, 1, 1), DateTime.utc(2017, 1, 1)].map(dt =>\n this.extract(dt, intl, \"era\")\n );\n }\n\n return this.eraCache[length];\n });\n }\n\n extract(dt, intlOpts, field) {\n const df = this.dtFormatter(dt, intlOpts),\n results = df.formatToParts(),\n matching = results.find(m => m.type.toLowerCase() === field);\n return matching ? matching.value : null;\n }\n\n numberFormatter(opts = {}) {\n // this forcesimple option is never used (the only caller short-circuits on it, but it seems safer to leave)\n // (in contrast, the rest of the condition is used heavily)\n return new PolyNumberFormatter(this.intl, opts.forceSimple || this.fastNumbers, opts);\n }\n\n dtFormatter(dt, intlOpts = {}) {\n return new PolyDateFormatter(dt, this.intl, intlOpts);\n }\n\n relFormatter(opts = {}) {\n return new PolyRelFormatter(this.intl, this.isEnglish(), opts);\n }\n\n isEnglish() {\n return (\n this.locale === \"en\" ||\n this.locale.toLowerCase() === \"en-us\" ||\n (hasIntl() && new Intl.DateTimeFormat(this.intl).resolvedOptions().locale.startsWith(\"en-us\"))\n );\n }\n\n equals(other) {\n return (\n this.locale === other.locale &&\n this.numberingSystem === other.numberingSystem &&\n this.outputCalendar === other.outputCalendar\n );\n }\n}\n","import {\n untruncateYear,\n signedOffset,\n parseInteger,\n parseMillis,\n ianaRegex,\n isUndefined\n} from \"./util.js\";\nimport * as English from \"./english.js\";\nimport FixedOffsetZone from \"../zones/fixedOffsetZone.js\";\nimport IANAZone from \"../zones/IANAZone.js\";\n\n/*\n * This file handles parsing for well-specified formats. Here's how it works:\n * Two things go into parsing: a regex to match with and an extractor to take apart the groups in the match.\n * An extractor is just a function that takes a regex match array and returns a { year: ..., month: ... } object\n * parse() does the work of executing the regex and applying the extractor. It takes multiple regex/extractor pairs to try in sequence.\n * Extractors can take a \"cursor\" representing the offset in the match to look at. This makes it easy to combine extractors.\n * combineExtractors() does the work of combining them, keeping track of the cursor through multiple extractions.\n * Some extractions are super dumb and simpleParse and fromStrings help DRY them.\n */\n\nfunction combineRegexes(...regexes) {\n const full = regexes.reduce((f, r) => f + r.source, \"\");\n return RegExp(`^${full}$`);\n}\n\nfunction combineExtractors(...extractors) {\n return m =>\n extractors\n .reduce(\n ([mergedVals, mergedZone, cursor], ex) => {\n const [val, zone, next] = ex(m, cursor);\n return [Object.assign(mergedVals, val), mergedZone || zone, next];\n },\n [{}, null, 1]\n )\n .slice(0, 2);\n}\n\nfunction parse(s, ...patterns) {\n if (s == null) {\n return [null, null];\n }\n\n for (const [regex, extractor] of patterns) {\n const m = regex.exec(s);\n if (m) {\n return extractor(m);\n }\n }\n return [null, null];\n}\n\nfunction simpleParse(...keys) {\n return (match, cursor) => {\n const ret = {};\n let i;\n\n for (i = 0; i < keys.length; i++) {\n ret[keys[i]] = parseInteger(match[cursor + i]);\n }\n return [ret, null, cursor + i];\n };\n}\n\n// ISO and SQL parsing\nconst offsetRegex = /(?:(Z)|([+-]\\d\\d)(?::?(\\d\\d))?)/,\n isoTimeBaseRegex = /(\\d\\d)(?::?(\\d\\d)(?::?(\\d\\d)(?:[.,](\\d{1,30}))?)?)?/,\n isoTimeRegex = RegExp(`${isoTimeBaseRegex.source}${offsetRegex.source}?`),\n isoTimeExtensionRegex = RegExp(`(?:T${isoTimeRegex.source})?`),\n isoYmdRegex = /([+-]\\d{6}|\\d{4})(?:-?(\\d\\d)(?:-?(\\d\\d))?)?/,\n isoWeekRegex = /(\\d{4})-?W(\\d\\d)(?:-?(\\d))?/,\n isoOrdinalRegex = /(\\d{4})-?(\\d{3})/,\n extractISOWeekData = simpleParse(\"weekYear\", \"weekNumber\", \"weekDay\"),\n extractISOOrdinalData = simpleParse(\"year\", \"ordinal\"),\n sqlYmdRegex = /(\\d{4})-(\\d\\d)-(\\d\\d)/, // dumbed-down version of the ISO one\n sqlTimeRegex = RegExp(\n `${isoTimeBaseRegex.source} ?(?:${offsetRegex.source}|(${ianaRegex.source}))?`\n ),\n sqlTimeExtensionRegex = RegExp(`(?: ${sqlTimeRegex.source})?`);\n\nfunction int(match, pos, fallback) {\n const m = match[pos];\n return isUndefined(m) ? fallback : parseInteger(m);\n}\n\nfunction extractISOYmd(match, cursor) {\n const item = {\n year: int(match, cursor),\n month: int(match, cursor + 1, 1),\n day: int(match, cursor + 2, 1)\n };\n\n return [item, null, cursor + 3];\n}\n\nfunction extractISOTime(match, cursor) {\n const item = {\n hours: int(match, cursor, 0),\n minutes: int(match, cursor + 1, 0),\n seconds: int(match, cursor + 2, 0),\n milliseconds: parseMillis(match[cursor + 3])\n };\n\n return [item, null, cursor + 4];\n}\n\nfunction extractISOOffset(match, cursor) {\n const local = !match[cursor] && !match[cursor + 1],\n fullOffset = signedOffset(match[cursor + 1], match[cursor + 2]),\n zone = local ? null : FixedOffsetZone.instance(fullOffset);\n return [{}, zone, cursor + 3];\n}\n\nfunction extractIANAZone(match, cursor) {\n const zone = match[cursor] ? IANAZone.create(match[cursor]) : null;\n return [{}, zone, cursor + 1];\n}\n\n// ISO time parsing\n\nconst isoTimeOnly = RegExp(`^T?${isoTimeBaseRegex.source}$`);\n\n// ISO duration parsing\n\nconst isoDuration = /^-?P(?:(?:(-?\\d{1,9})Y)?(?:(-?\\d{1,9})M)?(?:(-?\\d{1,9})W)?(?:(-?\\d{1,9})D)?(?:T(?:(-?\\d{1,9})H)?(?:(-?\\d{1,9})M)?(?:(-?\\d{1,20})(?:[.,](-?\\d{1,9}))?S)?)?)$/;\n\nfunction extractISODuration(match) {\n const [\n s,\n yearStr,\n monthStr,\n weekStr,\n dayStr,\n hourStr,\n minuteStr,\n secondStr,\n millisecondsStr\n ] = match;\n\n const hasNegativePrefix = s[0] === \"-\";\n const negativeSeconds = secondStr && secondStr[0] === \"-\";\n\n const maybeNegate = (num, force = false) =>\n num !== undefined && (force || (num && hasNegativePrefix)) ? -num : num;\n\n return [\n {\n years: maybeNegate(parseInteger(yearStr)),\n months: maybeNegate(parseInteger(monthStr)),\n weeks: maybeNegate(parseInteger(weekStr)),\n days: maybeNegate(parseInteger(dayStr)),\n hours: maybeNegate(parseInteger(hourStr)),\n minutes: maybeNegate(parseInteger(minuteStr)),\n seconds: maybeNegate(parseInteger(secondStr), secondStr === \"-0\"),\n milliseconds: maybeNegate(parseMillis(millisecondsStr), negativeSeconds)\n }\n ];\n}\n\n// These are a little braindead. EDT *should* tell us that we're in, say, America/New_York\n// and not just that we're in -240 *right now*. But since I don't think these are used that often\n// I'm just going to ignore that\nconst obsOffsets = {\n GMT: 0,\n EDT: -4 * 60,\n EST: -5 * 60,\n CDT: -5 * 60,\n CST: -6 * 60,\n MDT: -6 * 60,\n MST: -7 * 60,\n PDT: -7 * 60,\n PST: -8 * 60\n};\n\nfunction fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr) {\n const result = {\n year: yearStr.length === 2 ? untruncateYear(parseInteger(yearStr)) : parseInteger(yearStr),\n month: English.monthsShort.indexOf(monthStr) + 1,\n day: parseInteger(dayStr),\n hour: parseInteger(hourStr),\n minute: parseInteger(minuteStr)\n };\n\n if (secondStr) result.second = parseInteger(secondStr);\n if (weekdayStr) {\n result.weekday =\n weekdayStr.length > 3\n ? English.weekdaysLong.indexOf(weekdayStr) + 1\n : English.weekdaysShort.indexOf(weekdayStr) + 1;\n }\n\n return result;\n}\n\n// RFC 2822/5322\nconst rfc2822 = /^(?:(Mon|Tue|Wed|Thu|Fri|Sat|Sun),\\s)?(\\d{1,2})\\s(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)\\s(\\d{2,4})\\s(\\d\\d):(\\d\\d)(?::(\\d\\d))?\\s(?:(UT|GMT|[ECMP][SD]T)|([Zz])|(?:([+-]\\d\\d)(\\d\\d)))$/;\n\nfunction extractRFC2822(match) {\n const [\n ,\n weekdayStr,\n dayStr,\n monthStr,\n yearStr,\n hourStr,\n minuteStr,\n secondStr,\n obsOffset,\n milOffset,\n offHourStr,\n offMinuteStr\n ] = match,\n result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);\n\n let offset;\n if (obsOffset) {\n offset = obsOffsets[obsOffset];\n } else if (milOffset) {\n offset = 0;\n } else {\n offset = signedOffset(offHourStr, offMinuteStr);\n }\n\n return [result, new FixedOffsetZone(offset)];\n}\n\nfunction preprocessRFC2822(s) {\n // Remove comments and folding whitespace and replace multiple-spaces with a single space\n return s\n .replace(/\\([^)]*\\)|[\\n\\t]/g, \" \")\n .replace(/(\\s\\s+)/g, \" \")\n .trim();\n}\n\n// http date\n\nconst rfc1123 = /^(Mon|Tue|Wed|Thu|Fri|Sat|Sun), (\\d\\d) (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec) (\\d{4}) (\\d\\d):(\\d\\d):(\\d\\d) GMT$/,\n rfc850 = /^(Monday|Tuesday|Wedsday|Thursday|Friday|Saturday|Sunday), (\\d\\d)-(Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec)-(\\d\\d) (\\d\\d):(\\d\\d):(\\d\\d) GMT$/,\n ascii = /^(Mon|Tue|Wed|Thu|Fri|Sat|Sun) (Jan|Feb|Mar|Apr|May|Jun|Jul|Aug|Sep|Oct|Nov|Dec) ( \\d|\\d\\d) (\\d\\d):(\\d\\d):(\\d\\d) (\\d{4})$/;\n\nfunction extractRFC1123Or850(match) {\n const [, weekdayStr, dayStr, monthStr, yearStr, hourStr, minuteStr, secondStr] = match,\n result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);\n return [result, FixedOffsetZone.utcInstance];\n}\n\nfunction extractASCII(match) {\n const [, weekdayStr, monthStr, dayStr, hourStr, minuteStr, secondStr, yearStr] = match,\n result = fromStrings(weekdayStr, yearStr, monthStr, dayStr, hourStr, minuteStr, secondStr);\n return [result, FixedOffsetZone.utcInstance];\n}\n\nconst isoYmdWithTimeExtensionRegex = combineRegexes(isoYmdRegex, isoTimeExtensionRegex);\nconst isoWeekWithTimeExtensionRegex = combineRegexes(isoWeekRegex, isoTimeExtensionRegex);\nconst isoOrdinalWithTimeExtensionRegex = combineRegexes(isoOrdinalRegex, isoTimeExtensionRegex);\nconst isoTimeCombinedRegex = combineRegexes(isoTimeRegex);\n\nconst extractISOYmdTimeAndOffset = combineExtractors(\n extractISOYmd,\n extractISOTime,\n extractISOOffset\n);\nconst extractISOWeekTimeAndOffset = combineExtractors(\n extractISOWeekData,\n extractISOTime,\n extractISOOffset\n);\nconst extractISOOrdinalDateAndTime = combineExtractors(\n extractISOOrdinalData,\n extractISOTime,\n extractISOOffset\n);\nconst extractISOTimeAndOffset = combineExtractors(extractISOTime, extractISOOffset);\n\n/**\n * @private\n */\n\nexport function parseISODate(s) {\n return parse(\n s,\n [isoYmdWithTimeExtensionRegex, extractISOYmdTimeAndOffset],\n [isoWeekWithTimeExtensionRegex, extractISOWeekTimeAndOffset],\n [isoOrdinalWithTimeExtensionRegex, extractISOOrdinalDateAndTime],\n [isoTimeCombinedRegex, extractISOTimeAndOffset]\n );\n}\n\nexport function parseRFC2822Date(s) {\n return parse(preprocessRFC2822(s), [rfc2822, extractRFC2822]);\n}\n\nexport function parseHTTPDate(s) {\n return parse(\n s,\n [rfc1123, extractRFC1123Or850],\n [rfc850, extractRFC1123Or850],\n [ascii, extractASCII]\n );\n}\n\nexport function parseISODuration(s) {\n return parse(s, [isoDuration, extractISODuration]);\n}\n\nconst extractISOTimeOnly = combineExtractors(extractISOTime);\n\nexport function parseISOTimeOnly(s) {\n return parse(s, [isoTimeOnly, extractISOTimeOnly]);\n}\n\nconst sqlYmdWithTimeExtensionRegex = combineRegexes(sqlYmdRegex, sqlTimeExtensionRegex);\nconst sqlTimeCombinedRegex = combineRegexes(sqlTimeRegex);\n\nconst extractISOYmdTimeOffsetAndIANAZone = combineExtractors(\n extractISOYmd,\n extractISOTime,\n extractISOOffset,\n extractIANAZone\n);\nconst extractISOTimeOffsetAndIANAZone = combineExtractors(\n extractISOTime,\n extractISOOffset,\n extractIANAZone\n);\n\nexport function parseSQL(s) {\n return parse(\n s,\n [sqlYmdWithTimeExtensionRegex, extractISOYmdTimeOffsetAndIANAZone],\n [sqlTimeCombinedRegex, extractISOTimeOffsetAndIANAZone]\n );\n}\n","import { InvalidArgumentError, InvalidDurationError, InvalidUnitError } from \"./errors.js\";\nimport Formatter from \"./impl/formatter.js\";\nimport Invalid from \"./impl/invalid.js\";\nimport Locale from \"./impl/locale.js\";\nimport { parseISODuration, parseISOTimeOnly } from \"./impl/regexParser.js\";\nimport {\n asNumber,\n hasOwnProperty,\n isNumber,\n isUndefined,\n normalizeObject,\n roundTo\n} from \"./impl/util.js\";\nimport Settings from \"./settings.js\";\n\nconst INVALID = \"Invalid Duration\";\n\n// unit conversion constants\nconst lowOrderMatrix = {\n weeks: {\n days: 7,\n hours: 7 * 24,\n minutes: 7 * 24 * 60,\n seconds: 7 * 24 * 60 * 60,\n milliseconds: 7 * 24 * 60 * 60 * 1000\n },\n days: {\n hours: 24,\n minutes: 24 * 60,\n seconds: 24 * 60 * 60,\n milliseconds: 24 * 60 * 60 * 1000\n },\n hours: { minutes: 60, seconds: 60 * 60, milliseconds: 60 * 60 * 1000 },\n minutes: { seconds: 60, milliseconds: 60 * 1000 },\n seconds: { milliseconds: 1000 }\n },\n casualMatrix = Object.assign(\n {\n years: {\n quarters: 4,\n months: 12,\n weeks: 52,\n days: 365,\n hours: 365 * 24,\n minutes: 365 * 24 * 60,\n seconds: 365 * 24 * 60 * 60,\n milliseconds: 365 * 24 * 60 * 60 * 1000\n },\n quarters: {\n months: 3,\n weeks: 13,\n days: 91,\n hours: 91 * 24,\n minutes: 91 * 24 * 60,\n seconds: 91 * 24 * 60 * 60,\n milliseconds: 91 * 24 * 60 * 60 * 1000\n },\n months: {\n weeks: 4,\n days: 30,\n hours: 30 * 24,\n minutes: 30 * 24 * 60,\n seconds: 30 * 24 * 60 * 60,\n milliseconds: 30 * 24 * 60 * 60 * 1000\n }\n },\n lowOrderMatrix\n ),\n daysInYearAccurate = 146097.0 / 400,\n daysInMonthAccurate = 146097.0 / 4800,\n accurateMatrix = Object.assign(\n {\n years: {\n quarters: 4,\n months: 12,\n weeks: daysInYearAccurate / 7,\n days: daysInYearAccurate,\n hours: daysInYearAccurate * 24,\n minutes: daysInYearAccurate * 24 * 60,\n seconds: daysInYearAccurate * 24 * 60 * 60,\n milliseconds: daysInYearAccurate * 24 * 60 * 60 * 1000\n },\n quarters: {\n months: 3,\n weeks: daysInYearAccurate / 28,\n days: daysInYearAccurate / 4,\n hours: (daysInYearAccurate * 24) / 4,\n minutes: (daysInYearAccurate * 24 * 60) / 4,\n seconds: (daysInYearAccurate * 24 * 60 * 60) / 4,\n milliseconds: (daysInYearAccurate * 24 * 60 * 60 * 1000) / 4\n },\n months: {\n weeks: daysInMonthAccurate / 7,\n days: daysInMonthAccurate,\n hours: daysInMonthAccurate * 24,\n minutes: daysInMonthAccurate * 24 * 60,\n seconds: daysInMonthAccurate * 24 * 60 * 60,\n milliseconds: daysInMonthAccurate * 24 * 60 * 60 * 1000\n }\n },\n lowOrderMatrix\n );\n\n// units ordered by size\nconst orderedUnits = [\n \"years\",\n \"quarters\",\n \"months\",\n \"weeks\",\n \"days\",\n \"hours\",\n \"minutes\",\n \"seconds\",\n \"milliseconds\"\n];\n\nconst reverseUnits = orderedUnits.slice(0).reverse();\n\n// clone really means \"create another instance just like this one, but with these changes\"\nfunction clone(dur, alts, clear = false) {\n // deep merge for vals\n const conf = {\n values: clear ? alts.values : Object.assign({}, dur.values, alts.values || {}),\n loc: dur.loc.clone(alts.loc),\n conversionAccuracy: alts.conversionAccuracy || dur.conversionAccuracy\n };\n return new Duration(conf);\n}\n\nfunction antiTrunc(n) {\n return n < 0 ? Math.floor(n) : Math.ceil(n);\n}\n\n// NB: mutates parameters\nfunction convert(matrix, fromMap, fromUnit, toMap, toUnit) {\n const conv = matrix[toUnit][fromUnit],\n raw = fromMap[fromUnit] / conv,\n sameSign = Math.sign(raw) === Math.sign(toMap[toUnit]),\n // ok, so this is wild, but see the matrix in the tests\n added =\n !sameSign && toMap[toUnit] !== 0 && Math.abs(raw) <= 1 ? antiTrunc(raw) : Math.trunc(raw);\n toMap[toUnit] += added;\n fromMap[fromUnit] -= added * conv;\n}\n\n// NB: mutates parameters\nfunction normalizeValues(matrix, vals) {\n reverseUnits.reduce((previous, current) => {\n if (!isUndefined(vals[current])) {\n if (previous) {\n convert(matrix, vals, previous, vals, current);\n }\n return current;\n } else {\n return previous;\n }\n }, null);\n}\n\n/**\n * A Duration object represents a period of time, like \"2 months\" or \"1 day, 1 hour\". Conceptually, it's just a map of units to their quantities, accompanied by some additional configuration and methods for creating, parsing, interrogating, transforming, and formatting them. They can be used on their own or in conjunction with other Luxon types; for example, you can use {@link DateTime.plus} to add a Duration object to a DateTime, producing another DateTime.\n *\n * Here is a brief overview of commonly used methods and getters in Duration:\n *\n * * **Creation** To create a Duration, use {@link Duration.fromMillis}, {@link Duration.fromObject}, or {@link Duration.fromISO}.\n * * **Unit values** See the {@link Duration.years}, {@link Duration.months}, {@link Duration.weeks}, {@link Duration.days}, {@link Duration.hours}, {@link Duration.minutes}, {@link Duration.seconds}, {@link Duration.milliseconds} accessors.\n * * **Configuration** See {@link Duration.locale} and {@link Duration.numberingSystem} accessors.\n * * **Transformation** To create new Durations out of old ones use {@link Duration.plus}, {@link Duration.minus}, {@link Duration.normalize}, {@link Duration.set}, {@link Duration.reconfigure}, {@link Duration.shiftTo}, and {@link Duration.negate}.\n * * **Output** To convert the Duration into other representations, see {@link Duration.as}, {@link Duration.toISO}, {@link Duration.toFormat}, and {@link Duration.toJSON}\n *\n * There's are more methods documented below. In addition, for more information on subtler topics like internationalization and validity, see the external documentation.\n */\nexport default class Duration {\n /**\n * @private\n */\n constructor(config) {\n const accurate = config.conversionAccuracy === \"longterm\" || false;\n /**\n * @access private\n */\n this.values = config.values;\n /**\n * @access private\n */\n this.loc = config.loc || Locale.create();\n /**\n * @access private\n */\n this.conversionAccuracy = accurate ? \"longterm\" : \"casual\";\n /**\n * @access private\n */\n this.invalid = config.invalid || null;\n /**\n * @access private\n */\n this.matrix = accurate ? accurateMatrix : casualMatrix;\n /**\n * @access private\n */\n this.isLuxonDuration = true;\n }\n\n /**\n * Create Duration from a number of milliseconds.\n * @param {number} count of milliseconds\n * @param {Object} opts - options for parsing\n * @param {string} [opts.locale='en-US'] - the locale to use\n * @param {string} opts.numberingSystem - the numbering system to use\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @return {Duration}\n */\n static fromMillis(count, opts) {\n return Duration.fromObject(Object.assign({ milliseconds: count }, opts));\n }\n\n /**\n * Create a Duration from a JavaScript object with keys like 'years' and 'hours'.\n * If this object is empty then a zero milliseconds duration is returned.\n * @param {Object} obj - the object to create the DateTime from\n * @param {number} obj.years\n * @param {number} obj.quarters\n * @param {number} obj.months\n * @param {number} obj.weeks\n * @param {number} obj.days\n * @param {number} obj.hours\n * @param {number} obj.minutes\n * @param {number} obj.seconds\n * @param {number} obj.milliseconds\n * @param {string} [obj.locale='en-US'] - the locale to use\n * @param {string} obj.numberingSystem - the numbering system to use\n * @param {string} [obj.conversionAccuracy='casual'] - the conversion system to use\n * @return {Duration}\n */\n static fromObject(obj) {\n if (obj == null || typeof obj !== \"object\") {\n throw new InvalidArgumentError(\n `Duration.fromObject: argument expected to be an object, got ${\n obj === null ? \"null\" : typeof obj\n }`\n );\n }\n return new Duration({\n values: normalizeObject(obj, Duration.normalizeUnit, [\n \"locale\",\n \"numberingSystem\",\n \"conversionAccuracy\",\n \"zone\" // a bit of debt; it's super inconvenient internally not to be able to blindly pass this\n ]),\n loc: Locale.fromObject(obj),\n conversionAccuracy: obj.conversionAccuracy\n });\n }\n\n /**\n * Create a Duration from an ISO 8601 duration string.\n * @param {string} text - text to parse\n * @param {Object} opts - options for parsing\n * @param {string} [opts.locale='en-US'] - the locale to use\n * @param {string} opts.numberingSystem - the numbering system to use\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @see https://en.wikipedia.org/wiki/ISO_8601#Durations\n * @example Duration.fromISO('P3Y6M1W4DT12H30M5S').toObject() //=> { years: 3, months: 6, weeks: 1, days: 4, hours: 12, minutes: 30, seconds: 5 }\n * @example Duration.fromISO('PT23H').toObject() //=> { hours: 23 }\n * @example Duration.fromISO('P5Y3M').toObject() //=> { years: 5, months: 3 }\n * @return {Duration}\n */\n static fromISO(text, opts) {\n const [parsed] = parseISODuration(text);\n if (parsed) {\n const obj = Object.assign(parsed, opts);\n return Duration.fromObject(obj);\n } else {\n return Duration.invalid(\"unparsable\", `the input \"${text}\" can't be parsed as ISO 8601`);\n }\n }\n\n /**\n * Create a Duration from an ISO 8601 time string.\n * @param {string} text - text to parse\n * @param {Object} opts - options for parsing\n * @param {string} [opts.locale='en-US'] - the locale to use\n * @param {string} opts.numberingSystem - the numbering system to use\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @see https://en.wikipedia.org/wiki/ISO_8601#Times\n * @example Duration.fromISOTime('11:22:33.444').toObject() //=> { hours: 11, minutes: 22, seconds: 33, milliseconds: 444 }\n * @example Duration.fromISOTime('11:00').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }\n * @example Duration.fromISOTime('T11:00').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }\n * @example Duration.fromISOTime('1100').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }\n * @example Duration.fromISOTime('T1100').toObject() //=> { hours: 11, minutes: 0, seconds: 0 }\n * @return {Duration}\n */\n static fromISOTime(text, opts) {\n const [parsed] = parseISOTimeOnly(text);\n if (parsed) {\n const obj = Object.assign(parsed, opts);\n return Duration.fromObject(obj);\n } else {\n return Duration.invalid(\"unparsable\", `the input \"${text}\" can't be parsed as ISO 8601`);\n }\n }\n\n /**\n * Create an invalid Duration.\n * @param {string} reason - simple string of why this datetime is invalid. Should not contain parameters or anything else data-dependent\n * @param {string} [explanation=null] - longer explanation, may include parameters and other useful debugging information\n * @return {Duration}\n */\n static invalid(reason, explanation = null) {\n if (!reason) {\n throw new InvalidArgumentError(\"need to specify a reason the Duration is invalid\");\n }\n\n const invalid = reason instanceof Invalid ? reason : new Invalid(reason, explanation);\n\n if (Settings.throwOnInvalid) {\n throw new InvalidDurationError(invalid);\n } else {\n return new Duration({ invalid });\n }\n }\n\n /**\n * @private\n */\n static normalizeUnit(unit) {\n const normalized = {\n year: \"years\",\n years: \"years\",\n quarter: \"quarters\",\n quarters: \"quarters\",\n month: \"months\",\n months: \"months\",\n week: \"weeks\",\n weeks: \"weeks\",\n day: \"days\",\n days: \"days\",\n hour: \"hours\",\n hours: \"hours\",\n minute: \"minutes\",\n minutes: \"minutes\",\n second: \"seconds\",\n seconds: \"seconds\",\n millisecond: \"milliseconds\",\n milliseconds: \"milliseconds\"\n }[unit ? unit.toLowerCase() : unit];\n\n if (!normalized) throw new InvalidUnitError(unit);\n\n return normalized;\n }\n\n /**\n * Check if an object is a Duration. Works across context boundaries\n * @param {object} o\n * @return {boolean}\n */\n static isDuration(o) {\n return (o && o.isLuxonDuration) || false;\n }\n\n /**\n * Get the locale of a Duration, such 'en-GB'\n * @type {string}\n */\n get locale() {\n return this.isValid ? this.loc.locale : null;\n }\n\n /**\n * Get the numbering system of a Duration, such 'beng'. The numbering system is used when formatting the Duration\n *\n * @type {string}\n */\n get numberingSystem() {\n return this.isValid ? this.loc.numberingSystem : null;\n }\n\n /**\n * Returns a string representation of this Duration formatted according to the specified format string. You may use these tokens:\n * * `S` for milliseconds\n * * `s` for seconds\n * * `m` for minutes\n * * `h` for hours\n * * `d` for days\n * * `M` for months\n * * `y` for years\n * Notes:\n * * Add padding by repeating the token, e.g. \"yy\" pads the years to two digits, \"hhhh\" pads the hours out to four digits\n * * The duration will be converted to the set of units in the format string using {@link Duration.shiftTo} and the Durations's conversion accuracy setting.\n * @param {string} fmt - the format string\n * @param {Object} opts - options\n * @param {boolean} [opts.floor=true] - floor numerical values\n * @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat(\"y d s\") //=> \"1 6 2\"\n * @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat(\"yy dd sss\") //=> \"01 06 002\"\n * @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toFormat(\"M S\") //=> \"12 518402000\"\n * @return {string}\n */\n toFormat(fmt, opts = {}) {\n // reverse-compat since 1.2; we always round down now, never up, and we do it by default\n const fmtOpts = Object.assign({}, opts, {\n floor: opts.round !== false && opts.floor !== false\n });\n return this.isValid\n ? Formatter.create(this.loc, fmtOpts).formatDurationFromString(this, fmt)\n : INVALID;\n }\n\n /**\n * Returns a JavaScript object with this Duration's values.\n * @param opts - options for generating the object\n * @param {boolean} [opts.includeConfig=false] - include configuration attributes in the output\n * @example Duration.fromObject({ years: 1, days: 6, seconds: 2 }).toObject() //=> { years: 1, days: 6, seconds: 2 }\n * @return {Object}\n */\n toObject(opts = {}) {\n if (!this.isValid) return {};\n\n const base = Object.assign({}, this.values);\n\n if (opts.includeConfig) {\n base.conversionAccuracy = this.conversionAccuracy;\n base.numberingSystem = this.loc.numberingSystem;\n base.locale = this.loc.locale;\n }\n return base;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this Duration.\n * @see https://en.wikipedia.org/wiki/ISO_8601#Durations\n * @example Duration.fromObject({ years: 3, seconds: 45 }).toISO() //=> 'P3YT45S'\n * @example Duration.fromObject({ months: 4, seconds: 45 }).toISO() //=> 'P4MT45S'\n * @example Duration.fromObject({ months: 5 }).toISO() //=> 'P5M'\n * @example Duration.fromObject({ minutes: 5 }).toISO() //=> 'PT5M'\n * @example Duration.fromObject({ milliseconds: 6 }).toISO() //=> 'PT0.006S'\n * @return {string}\n */\n toISO() {\n // we could use the formatter, but this is an easier way to get the minimum string\n if (!this.isValid) return null;\n\n let s = \"P\";\n if (this.years !== 0) s += this.years + \"Y\";\n if (this.months !== 0 || this.quarters !== 0) s += this.months + this.quarters * 3 + \"M\";\n if (this.weeks !== 0) s += this.weeks + \"W\";\n if (this.days !== 0) s += this.days + \"D\";\n if (this.hours !== 0 || this.minutes !== 0 || this.seconds !== 0 || this.milliseconds !== 0)\n s += \"T\";\n if (this.hours !== 0) s += this.hours + \"H\";\n if (this.minutes !== 0) s += this.minutes + \"M\";\n if (this.seconds !== 0 || this.milliseconds !== 0)\n // this will handle \"floating point madness\" by removing extra decimal places\n // https://stackoverflow.com/questions/588004/is-floating-point-math-broken\n s += roundTo(this.seconds + this.milliseconds / 1000, 3) + \"S\";\n if (s === \"P\") s += \"T0S\";\n return s;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this Duration, formatted as a time of day.\n * Note that this will return null if the duration is invalid, negative, or equal to or greater than 24 hours.\n * @see https://en.wikipedia.org/wiki/ISO_8601#Times\n * @param {Object} opts - options\n * @param {boolean} [opts.suppressMilliseconds=false] - exclude milliseconds from the format if they're 0\n * @param {boolean} [opts.suppressSeconds=false] - exclude seconds from the format if they're 0\n * @param {boolean} [opts.includePrefix=false] - include the `T` prefix\n * @param {string} [opts.format='extended'] - choose between the basic and extended format\n * @example Duration.fromObject({ hours: 11 }).toISOTime() //=> '11:00:00.000'\n * @example Duration.fromObject({ hours: 11 }).toISOTime({ suppressMilliseconds: true }) //=> '11:00:00'\n * @example Duration.fromObject({ hours: 11 }).toISOTime({ suppressSeconds: true }) //=> '11:00'\n * @example Duration.fromObject({ hours: 11 }).toISOTime({ includePrefix: true }) //=> 'T11:00:00.000'\n * @example Duration.fromObject({ hours: 11 }).toISOTime({ format: 'basic' }) //=> '110000.000'\n * @return {string}\n */\n toISOTime(opts = {}) {\n if (!this.isValid) return null;\n\n const millis = this.toMillis();\n if (millis < 0 || millis >= 86400000) return null;\n\n opts = Object.assign(\n {\n suppressMilliseconds: false,\n suppressSeconds: false,\n includePrefix: false,\n format: \"extended\"\n },\n opts\n );\n\n const value = this.shiftTo(\"hours\", \"minutes\", \"seconds\", \"milliseconds\");\n\n let fmt = opts.format === \"basic\" ? \"hhmm\" : \"hh:mm\";\n\n if (!opts.suppressSeconds || value.seconds !== 0 || value.milliseconds !== 0) {\n fmt += opts.format === \"basic\" ? \"ss\" : \":ss\";\n if (!opts.suppressMilliseconds || value.milliseconds !== 0) {\n fmt += \".SSS\";\n }\n }\n\n let str = value.toFormat(fmt);\n\n if (opts.includePrefix) {\n str = \"T\" + str;\n }\n\n return str;\n }\n\n /**\n * Returns an ISO 8601 representation of this Duration appropriate for use in JSON.\n * @return {string}\n */\n toJSON() {\n return this.toISO();\n }\n\n /**\n * Returns an ISO 8601 representation of this Duration appropriate for use in debugging.\n * @return {string}\n */\n toString() {\n return this.toISO();\n }\n\n /**\n * Returns an milliseconds value of this Duration.\n * @return {number}\n */\n toMillis() {\n return this.as(\"milliseconds\");\n }\n\n /**\n * Returns an milliseconds value of this Duration. Alias of {@link toMillis}\n * @return {number}\n */\n valueOf() {\n return this.toMillis();\n }\n\n /**\n * Make this Duration longer by the specified amount. Return a newly-constructed Duration.\n * @param {Duration|Object|number} duration - The amount to add. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()\n * @return {Duration}\n */\n plus(duration) {\n if (!this.isValid) return this;\n\n const dur = friendlyDuration(duration),\n result = {};\n\n for (const k of orderedUnits) {\n if (hasOwnProperty(dur.values, k) || hasOwnProperty(this.values, k)) {\n result[k] = dur.get(k) + this.get(k);\n }\n }\n\n return clone(this, { values: result }, true);\n }\n\n /**\n * Make this Duration shorter by the specified amount. Return a newly-constructed Duration.\n * @param {Duration|Object|number} duration - The amount to subtract. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()\n * @return {Duration}\n */\n minus(duration) {\n if (!this.isValid) return this;\n\n const dur = friendlyDuration(duration);\n return this.plus(dur.negate());\n }\n\n /**\n * Scale this Duration by the specified amount. Return a newly-constructed Duration.\n * @param {function} fn - The function to apply to each unit. Arity is 1 or 2: the value of the unit and, optionally, the unit name. Must return a number.\n * @example Duration.fromObject({ hours: 1, minutes: 30 }).mapUnit(x => x * 2) //=> { hours: 2, minutes: 60 }\n * @example Duration.fromObject({ hours: 1, minutes: 30 }).mapUnit((x, u) => u === \"hour\" ? x * 2 : x) //=> { hours: 2, minutes: 30 }\n * @return {Duration}\n */\n mapUnits(fn) {\n if (!this.isValid) return this;\n const result = {};\n for (const k of Object.keys(this.values)) {\n result[k] = asNumber(fn(this.values[k], k));\n }\n return clone(this, { values: result }, true);\n }\n\n /**\n * Get the value of unit.\n * @param {string} unit - a unit such as 'minute' or 'day'\n * @example Duration.fromObject({years: 2, days: 3}).get('years') //=> 2\n * @example Duration.fromObject({years: 2, days: 3}).get('months') //=> 0\n * @example Duration.fromObject({years: 2, days: 3}).get('days') //=> 3\n * @return {number}\n */\n get(unit) {\n return this[Duration.normalizeUnit(unit)];\n }\n\n /**\n * \"Set\" the values of specified units. Return a newly-constructed Duration.\n * @param {Object} values - a mapping of units to numbers\n * @example dur.set({ years: 2017 })\n * @example dur.set({ hours: 8, minutes: 30 })\n * @return {Duration}\n */\n set(values) {\n if (!this.isValid) return this;\n\n const mixed = Object.assign(this.values, normalizeObject(values, Duration.normalizeUnit, []));\n return clone(this, { values: mixed });\n }\n\n /**\n * \"Set\" the locale and/or numberingSystem. Returns a newly-constructed Duration.\n * @example dur.reconfigure({ locale: 'en-GB' })\n * @return {Duration}\n */\n reconfigure({ locale, numberingSystem, conversionAccuracy } = {}) {\n const loc = this.loc.clone({ locale, numberingSystem }),\n opts = { loc };\n\n if (conversionAccuracy) {\n opts.conversionAccuracy = conversionAccuracy;\n }\n\n return clone(this, opts);\n }\n\n /**\n * Return the length of the duration in the specified unit.\n * @param {string} unit - a unit such as 'minutes' or 'days'\n * @example Duration.fromObject({years: 1}).as('days') //=> 365\n * @example Duration.fromObject({years: 1}).as('months') //=> 12\n * @example Duration.fromObject({hours: 60}).as('days') //=> 2.5\n * @return {number}\n */\n as(unit) {\n return this.isValid ? this.shiftTo(unit).get(unit) : NaN;\n }\n\n /**\n * Reduce this Duration to its canonical representation in its current units.\n * @example Duration.fromObject({ years: 2, days: 5000 }).normalize().toObject() //=> { years: 15, days: 255 }\n * @example Duration.fromObject({ hours: 12, minutes: -45 }).normalize().toObject() //=> { hours: 11, minutes: 15 }\n * @return {Duration}\n */\n normalize() {\n if (!this.isValid) return this;\n const vals = this.toObject();\n normalizeValues(this.matrix, vals);\n return clone(this, { values: vals }, true);\n }\n\n /**\n * Convert this Duration into its representation in a different set of units.\n * @example Duration.fromObject({ hours: 1, seconds: 30 }).shiftTo('minutes', 'milliseconds').toObject() //=> { minutes: 60, milliseconds: 30000 }\n * @return {Duration}\n */\n shiftTo(...units) {\n if (!this.isValid) return this;\n\n if (units.length === 0) {\n return this;\n }\n\n units = units.map(u => Duration.normalizeUnit(u));\n\n const built = {},\n accumulated = {},\n vals = this.toObject();\n let lastUnit;\n\n for (const k of orderedUnits) {\n if (units.indexOf(k) >= 0) {\n lastUnit = k;\n\n let own = 0;\n\n // anything we haven't boiled down yet should get boiled to this unit\n for (const ak in accumulated) {\n own += this.matrix[ak][k] * accumulated[ak];\n accumulated[ak] = 0;\n }\n\n // plus anything that's already in this unit\n if (isNumber(vals[k])) {\n own += vals[k];\n }\n\n const i = Math.trunc(own);\n built[k] = i;\n accumulated[k] = own - i; // we'd like to absorb these fractions in another unit\n\n // plus anything further down the chain that should be rolled up in to this\n for (const down in vals) {\n if (orderedUnits.indexOf(down) > orderedUnits.indexOf(k)) {\n convert(this.matrix, vals, down, built, k);\n }\n }\n // otherwise, keep it in the wings to boil it later\n } else if (isNumber(vals[k])) {\n accumulated[k] = vals[k];\n }\n }\n\n // anything leftover becomes the decimal for the last unit\n // lastUnit must be defined since units is not empty\n for (const key in accumulated) {\n if (accumulated[key] !== 0) {\n built[lastUnit] +=\n key === lastUnit ? accumulated[key] : accumulated[key] / this.matrix[lastUnit][key];\n }\n }\n\n return clone(this, { values: built }, true).normalize();\n }\n\n /**\n * Return the negative of this Duration.\n * @example Duration.fromObject({ hours: 1, seconds: 30 }).negate().toObject() //=> { hours: -1, seconds: -30 }\n * @return {Duration}\n */\n negate() {\n if (!this.isValid) return this;\n const negated = {};\n for (const k of Object.keys(this.values)) {\n negated[k] = -this.values[k];\n }\n return clone(this, { values: negated }, true);\n }\n\n /**\n * Get the years.\n * @type {number}\n */\n get years() {\n return this.isValid ? this.values.years || 0 : NaN;\n }\n\n /**\n * Get the quarters.\n * @type {number}\n */\n get quarters() {\n return this.isValid ? this.values.quarters || 0 : NaN;\n }\n\n /**\n * Get the months.\n * @type {number}\n */\n get months() {\n return this.isValid ? this.values.months || 0 : NaN;\n }\n\n /**\n * Get the weeks\n * @type {number}\n */\n get weeks() {\n return this.isValid ? this.values.weeks || 0 : NaN;\n }\n\n /**\n * Get the days.\n * @type {number}\n */\n get days() {\n return this.isValid ? this.values.days || 0 : NaN;\n }\n\n /**\n * Get the hours.\n * @type {number}\n */\n get hours() {\n return this.isValid ? this.values.hours || 0 : NaN;\n }\n\n /**\n * Get the minutes.\n * @type {number}\n */\n get minutes() {\n return this.isValid ? this.values.minutes || 0 : NaN;\n }\n\n /**\n * Get the seconds.\n * @return {number}\n */\n get seconds() {\n return this.isValid ? this.values.seconds || 0 : NaN;\n }\n\n /**\n * Get the milliseconds.\n * @return {number}\n */\n get milliseconds() {\n return this.isValid ? this.values.milliseconds || 0 : NaN;\n }\n\n /**\n * Returns whether the Duration is invalid. Invalid durations are returned by diff operations\n * on invalid DateTimes or Intervals.\n * @return {boolean}\n */\n get isValid() {\n return this.invalid === null;\n }\n\n /**\n * Returns an error code if this Duration became invalid, or null if the Duration is valid\n * @return {string}\n */\n get invalidReason() {\n return this.invalid ? this.invalid.reason : null;\n }\n\n /**\n * Returns an explanation of why this Duration became invalid, or null if the Duration is valid\n * @type {string}\n */\n get invalidExplanation() {\n return this.invalid ? this.invalid.explanation : null;\n }\n\n /**\n * Equality check\n * Two Durations are equal iff they have the same units and the same values for each unit.\n * @param {Duration} other\n * @return {boolean}\n */\n equals(other) {\n if (!this.isValid || !other.isValid) {\n return false;\n }\n\n if (!this.loc.equals(other.loc)) {\n return false;\n }\n\n function eq(v1, v2) {\n // Consider 0 and undefined as equal\n if (v1 === undefined || v1 === 0) return v2 === undefined || v2 === 0;\n return v1 === v2;\n }\n\n for (const u of orderedUnits) {\n if (!eq(this.values[u], other.values[u])) {\n return false;\n }\n }\n return true;\n }\n}\n\n/**\n * @private\n */\nexport function friendlyDuration(durationish) {\n if (isNumber(durationish)) {\n return Duration.fromMillis(durationish);\n } else if (Duration.isDuration(durationish)) {\n return durationish;\n } else if (typeof durationish === \"object\") {\n return Duration.fromObject(durationish);\n } else {\n throw new InvalidArgumentError(\n `Unknown duration argument ${durationish} of type ${typeof durationish}`\n );\n }\n}\n","import DateTime, { friendlyDateTime } from \"./datetime.js\";\nimport Duration, { friendlyDuration } from \"./duration.js\";\nimport Settings from \"./settings.js\";\nimport { InvalidArgumentError, InvalidIntervalError } from \"./errors.js\";\nimport Invalid from \"./impl/invalid.js\";\n\nconst INVALID = \"Invalid Interval\";\n\n// checks if the start is equal to or before the end\nfunction validateStartEnd(start, end) {\n if (!start || !start.isValid) {\n return Interval.invalid(\"missing or invalid start\");\n } else if (!end || !end.isValid) {\n return Interval.invalid(\"missing or invalid end\");\n } else if (end < start) {\n return Interval.invalid(\n \"end before start\",\n `The end of an interval must be after its start, but you had start=${start.toISO()} and end=${end.toISO()}`\n );\n } else {\n return null;\n }\n}\n\n/**\n * An Interval object represents a half-open interval of time, where each endpoint is a {@link DateTime}. Conceptually, it's a container for those two endpoints, accompanied by methods for creating, parsing, interrogating, comparing, transforming, and formatting them.\n *\n * Here is a brief overview of the most commonly used methods and getters in Interval:\n *\n * * **Creation** To create an Interval, use {@link fromDateTimes}, {@link after}, {@link before}, or {@link fromISO}.\n * * **Accessors** Use {@link start} and {@link end} to get the start and end.\n * * **Interrogation** To analyze the Interval, use {@link count}, {@link length}, {@link hasSame}, {@link contains}, {@link isAfter}, or {@link isBefore}.\n * * **Transformation** To create other Intervals out of this one, use {@link set}, {@link splitAt}, {@link splitBy}, {@link divideEqually}, {@link merge}, {@link xor}, {@link union}, {@link intersection}, or {@link difference}.\n * * **Comparison** To compare this Interval to another one, use {@link equals}, {@link overlaps}, {@link abutsStart}, {@link abutsEnd}, {@link engulfs}.\n * * **Output** To convert the Interval into other representations, see {@link toString}, {@link toISO}, {@link toISODate}, {@link toISOTime}, {@link toFormat}, and {@link toDuration}.\n */\nexport default class Interval {\n /**\n * @private\n */\n constructor(config) {\n /**\n * @access private\n */\n this.s = config.start;\n /**\n * @access private\n */\n this.e = config.end;\n /**\n * @access private\n */\n this.invalid = config.invalid || null;\n /**\n * @access private\n */\n this.isLuxonInterval = true;\n }\n\n /**\n * Create an invalid Interval.\n * @param {string} reason - simple string of why this Interval is invalid. Should not contain parameters or anything else data-dependent\n * @param {string} [explanation=null] - longer explanation, may include parameters and other useful debugging information\n * @return {Interval}\n */\n static invalid(reason, explanation = null) {\n if (!reason) {\n throw new InvalidArgumentError(\"need to specify a reason the Interval is invalid\");\n }\n\n const invalid = reason instanceof Invalid ? reason : new Invalid(reason, explanation);\n\n if (Settings.throwOnInvalid) {\n throw new InvalidIntervalError(invalid);\n } else {\n return new Interval({ invalid });\n }\n }\n\n /**\n * Create an Interval from a start DateTime and an end DateTime. Inclusive of the start but not the end.\n * @param {DateTime|Date|Object} start\n * @param {DateTime|Date|Object} end\n * @return {Interval}\n */\n static fromDateTimes(start, end) {\n const builtStart = friendlyDateTime(start),\n builtEnd = friendlyDateTime(end);\n\n const validateError = validateStartEnd(builtStart, builtEnd);\n\n if (validateError == null) {\n return new Interval({\n start: builtStart,\n end: builtEnd\n });\n } else {\n return validateError;\n }\n }\n\n /**\n * Create an Interval from a start DateTime and a Duration to extend to.\n * @param {DateTime|Date|Object} start\n * @param {Duration|Object|number} duration - the length of the Interval.\n * @return {Interval}\n */\n static after(start, duration) {\n const dur = friendlyDuration(duration),\n dt = friendlyDateTime(start);\n return Interval.fromDateTimes(dt, dt.plus(dur));\n }\n\n /**\n * Create an Interval from an end DateTime and a Duration to extend backwards to.\n * @param {DateTime|Date|Object} end\n * @param {Duration|Object|number} duration - the length of the Interval.\n * @return {Interval}\n */\n static before(end, duration) {\n const dur = friendlyDuration(duration),\n dt = friendlyDateTime(end);\n return Interval.fromDateTimes(dt.minus(dur), dt);\n }\n\n /**\n * Create an Interval from an ISO 8601 string.\n * Accepts `/`, `/`, and `/` formats.\n * @param {string} text - the ISO string to parse\n * @param {Object} [opts] - options to pass {@link DateTime.fromISO} and optionally {@link Duration.fromISO}\n * @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals\n * @return {Interval}\n */\n static fromISO(text, opts) {\n const [s, e] = (text || \"\").split(\"/\", 2);\n if (s && e) {\n let start, startIsValid;\n try {\n start = DateTime.fromISO(s, opts);\n startIsValid = start.isValid;\n } catch (e) {\n startIsValid = false;\n }\n\n let end, endIsValid;\n try {\n end = DateTime.fromISO(e, opts);\n endIsValid = end.isValid;\n } catch (e) {\n endIsValid = false;\n }\n\n if (startIsValid && endIsValid) {\n return Interval.fromDateTimes(start, end);\n }\n\n if (startIsValid) {\n const dur = Duration.fromISO(e, opts);\n if (dur.isValid) {\n return Interval.after(start, dur);\n }\n } else if (endIsValid) {\n const dur = Duration.fromISO(s, opts);\n if (dur.isValid) {\n return Interval.before(end, dur);\n }\n }\n }\n return Interval.invalid(\"unparsable\", `the input \"${text}\" can't be parsed as ISO 8601`);\n }\n\n /**\n * Check if an object is an Interval. Works across context boundaries\n * @param {object} o\n * @return {boolean}\n */\n static isInterval(o) {\n return (o && o.isLuxonInterval) || false;\n }\n\n /**\n * Returns the start of the Interval\n * @type {DateTime}\n */\n get start() {\n return this.isValid ? this.s : null;\n }\n\n /**\n * Returns the end of the Interval\n * @type {DateTime}\n */\n get end() {\n return this.isValid ? this.e : null;\n }\n\n /**\n * Returns whether this Interval's end is at least its start, meaning that the Interval isn't 'backwards'.\n * @type {boolean}\n */\n get isValid() {\n return this.invalidReason === null;\n }\n\n /**\n * Returns an error code if this Interval is invalid, or null if the Interval is valid\n * @type {string}\n */\n get invalidReason() {\n return this.invalid ? this.invalid.reason : null;\n }\n\n /**\n * Returns an explanation of why this Interval became invalid, or null if the Interval is valid\n * @type {string}\n */\n get invalidExplanation() {\n return this.invalid ? this.invalid.explanation : null;\n }\n\n /**\n * Returns the length of the Interval in the specified unit.\n * @param {string} unit - the unit (such as 'hours' or 'days') to return the length in.\n * @return {number}\n */\n length(unit = \"milliseconds\") {\n return this.isValid ? this.toDuration(...[unit]).get(unit) : NaN;\n }\n\n /**\n * Returns the count of minutes, hours, days, months, or years included in the Interval, even in part.\n * Unlike {@link length} this counts sections of the calendar, not periods of time, e.g. specifying 'day'\n * asks 'what dates are included in this interval?', not 'how many days long is this interval?'\n * @param {string} [unit='milliseconds'] - the unit of time to count.\n * @return {number}\n */\n count(unit = \"milliseconds\") {\n if (!this.isValid) return NaN;\n const start = this.start.startOf(unit),\n end = this.end.startOf(unit);\n return Math.floor(end.diff(start, unit).get(unit)) + 1;\n }\n\n /**\n * Returns whether this Interval's start and end are both in the same unit of time\n * @param {string} unit - the unit of time to check sameness on\n * @return {boolean}\n */\n hasSame(unit) {\n return this.isValid ? this.isEmpty() || this.e.minus(1).hasSame(this.s, unit) : false;\n }\n\n /**\n * Return whether this Interval has the same start and end DateTimes.\n * @return {boolean}\n */\n isEmpty() {\n return this.s.valueOf() === this.e.valueOf();\n }\n\n /**\n * Return whether this Interval's start is after the specified DateTime.\n * @param {DateTime} dateTime\n * @return {boolean}\n */\n isAfter(dateTime) {\n if (!this.isValid) return false;\n return this.s > dateTime;\n }\n\n /**\n * Return whether this Interval's end is before the specified DateTime.\n * @param {DateTime} dateTime\n * @return {boolean}\n */\n isBefore(dateTime) {\n if (!this.isValid) return false;\n return this.e <= dateTime;\n }\n\n /**\n * Return whether this Interval contains the specified DateTime.\n * @param {DateTime} dateTime\n * @return {boolean}\n */\n contains(dateTime) {\n if (!this.isValid) return false;\n return this.s <= dateTime && this.e > dateTime;\n }\n\n /**\n * \"Sets\" the start and/or end dates. Returns a newly-constructed Interval.\n * @param {Object} values - the values to set\n * @param {DateTime} values.start - the starting DateTime\n * @param {DateTime} values.end - the ending DateTime\n * @return {Interval}\n */\n set({ start, end } = {}) {\n if (!this.isValid) return this;\n return Interval.fromDateTimes(start || this.s, end || this.e);\n }\n\n /**\n * Split this Interval at each of the specified DateTimes\n * @param {...[DateTime]} dateTimes - the unit of time to count.\n * @return {[Interval]}\n */\n splitAt(...dateTimes) {\n if (!this.isValid) return [];\n const sorted = dateTimes\n .map(friendlyDateTime)\n .filter(d => this.contains(d))\n .sort(),\n results = [];\n let { s } = this,\n i = 0;\n\n while (s < this.e) {\n const added = sorted[i] || this.e,\n next = +added > +this.e ? this.e : added;\n results.push(Interval.fromDateTimes(s, next));\n s = next;\n i += 1;\n }\n\n return results;\n }\n\n /**\n * Split this Interval into smaller Intervals, each of the specified length.\n * Left over time is grouped into a smaller interval\n * @param {Duration|Object|number} duration - The length of each resulting interval.\n * @return {[Interval]}\n */\n splitBy(duration) {\n const dur = friendlyDuration(duration);\n\n if (!this.isValid || !dur.isValid || dur.as(\"milliseconds\") === 0) {\n return [];\n }\n\n let { s } = this,\n idx = 1,\n next;\n\n const results = [];\n while (s < this.e) {\n const added = this.start.plus(dur.mapUnits(x => x * idx));\n next = +added > +this.e ? this.e : added;\n results.push(Interval.fromDateTimes(s, next));\n s = next;\n idx += 1;\n }\n\n return results;\n }\n\n /**\n * Split this Interval into the specified number of smaller intervals.\n * @param {number} numberOfParts - The number of Intervals to divide the Interval into.\n * @return {[Interval]}\n */\n divideEqually(numberOfParts) {\n if (!this.isValid) return [];\n return this.splitBy(this.length() / numberOfParts).slice(0, numberOfParts);\n }\n\n /**\n * Return whether this Interval overlaps with the specified Interval\n * @param {Interval} other\n * @return {boolean}\n */\n overlaps(other) {\n return this.e > other.s && this.s < other.e;\n }\n\n /**\n * Return whether this Interval's end is adjacent to the specified Interval's start.\n * @param {Interval} other\n * @return {boolean}\n */\n abutsStart(other) {\n if (!this.isValid) return false;\n return +this.e === +other.s;\n }\n\n /**\n * Return whether this Interval's start is adjacent to the specified Interval's end.\n * @param {Interval} other\n * @return {boolean}\n */\n abutsEnd(other) {\n if (!this.isValid) return false;\n return +other.e === +this.s;\n }\n\n /**\n * Return whether this Interval engulfs the start and end of the specified Interval.\n * @param {Interval} other\n * @return {boolean}\n */\n engulfs(other) {\n if (!this.isValid) return false;\n return this.s <= other.s && this.e >= other.e;\n }\n\n /**\n * Return whether this Interval has the same start and end as the specified Interval.\n * @param {Interval} other\n * @return {boolean}\n */\n equals(other) {\n if (!this.isValid || !other.isValid) {\n return false;\n }\n\n return this.s.equals(other.s) && this.e.equals(other.e);\n }\n\n /**\n * Return an Interval representing the intersection of this Interval and the specified Interval.\n * Specifically, the resulting Interval has the maximum start time and the minimum end time of the two Intervals.\n * Returns null if the intersection is empty, meaning, the intervals don't intersect.\n * @param {Interval} other\n * @return {Interval}\n */\n intersection(other) {\n if (!this.isValid) return this;\n const s = this.s > other.s ? this.s : other.s,\n e = this.e < other.e ? this.e : other.e;\n\n if (s >= e) {\n return null;\n } else {\n return Interval.fromDateTimes(s, e);\n }\n }\n\n /**\n * Return an Interval representing the union of this Interval and the specified Interval.\n * Specifically, the resulting Interval has the minimum start time and the maximum end time of the two Intervals.\n * @param {Interval} other\n * @return {Interval}\n */\n union(other) {\n if (!this.isValid) return this;\n const s = this.s < other.s ? this.s : other.s,\n e = this.e > other.e ? this.e : other.e;\n return Interval.fromDateTimes(s, e);\n }\n\n /**\n * Merge an array of Intervals into a equivalent minimal set of Intervals.\n * Combines overlapping and adjacent Intervals.\n * @param {[Interval]} intervals\n * @return {[Interval]}\n */\n static merge(intervals) {\n const [found, final] = intervals.sort((a, b) => a.s - b.s).reduce(\n ([sofar, current], item) => {\n if (!current) {\n return [sofar, item];\n } else if (current.overlaps(item) || current.abutsStart(item)) {\n return [sofar, current.union(item)];\n } else {\n return [sofar.concat([current]), item];\n }\n },\n [[], null]\n );\n if (final) {\n found.push(final);\n }\n return found;\n }\n\n /**\n * Return an array of Intervals representing the spans of time that only appear in one of the specified Intervals.\n * @param {[Interval]} intervals\n * @return {[Interval]}\n */\n static xor(intervals) {\n let start = null,\n currentCount = 0;\n const results = [],\n ends = intervals.map(i => [{ time: i.s, type: \"s\" }, { time: i.e, type: \"e\" }]),\n flattened = Array.prototype.concat(...ends),\n arr = flattened.sort((a, b) => a.time - b.time);\n\n for (const i of arr) {\n currentCount += i.type === \"s\" ? 1 : -1;\n\n if (currentCount === 1) {\n start = i.time;\n } else {\n if (start && +start !== +i.time) {\n results.push(Interval.fromDateTimes(start, i.time));\n }\n\n start = null;\n }\n }\n\n return Interval.merge(results);\n }\n\n /**\n * Return an Interval representing the span of time in this Interval that doesn't overlap with any of the specified Intervals.\n * @param {...Interval} intervals\n * @return {[Interval]}\n */\n difference(...intervals) {\n return Interval.xor([this].concat(intervals))\n .map(i => this.intersection(i))\n .filter(i => i && !i.isEmpty());\n }\n\n /**\n * Returns a string representation of this Interval appropriate for debugging.\n * @return {string}\n */\n toString() {\n if (!this.isValid) return INVALID;\n return `[${this.s.toISO()} – ${this.e.toISO()})`;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this Interval.\n * @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals\n * @param {Object} opts - The same options as {@link DateTime.toISO}\n * @return {string}\n */\n toISO(opts) {\n if (!this.isValid) return INVALID;\n return `${this.s.toISO(opts)}/${this.e.toISO(opts)}`;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of date of this Interval.\n * The time components are ignored.\n * @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals\n * @return {string}\n */\n toISODate() {\n if (!this.isValid) return INVALID;\n return `${this.s.toISODate()}/${this.e.toISODate()}`;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of time of this Interval.\n * The date components are ignored.\n * @see https://en.wikipedia.org/wiki/ISO_8601#Time_intervals\n * @param {Object} opts - The same options as {@link DateTime.toISO}\n * @return {string}\n */\n toISOTime(opts) {\n if (!this.isValid) return INVALID;\n return `${this.s.toISOTime(opts)}/${this.e.toISOTime(opts)}`;\n }\n\n /**\n * Returns a string representation of this Interval formatted according to the specified format string.\n * @param {string} dateFormat - the format string. This string formats the start and end time. See {@link DateTime.toFormat} for details.\n * @param {Object} opts - options\n * @param {string} [opts.separator = ' – '] - a separator to place between the start and end representations\n * @return {string}\n */\n toFormat(dateFormat, { separator = \" – \" } = {}) {\n if (!this.isValid) return INVALID;\n return `${this.s.toFormat(dateFormat)}${separator}${this.e.toFormat(dateFormat)}`;\n }\n\n /**\n * Return a Duration representing the time spanned by this interval.\n * @param {string|string[]} [unit=['milliseconds']] - the unit or units (such as 'hours' or 'days') to include in the duration.\n * @param {Object} opts - options that affect the creation of the Duration\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @example Interval.fromDateTimes(dt1, dt2).toDuration().toObject() //=> { milliseconds: 88489257 }\n * @example Interval.fromDateTimes(dt1, dt2).toDuration('days').toObject() //=> { days: 1.0241812152777778 }\n * @example Interval.fromDateTimes(dt1, dt2).toDuration(['hours', 'minutes']).toObject() //=> { hours: 24, minutes: 34.82095 }\n * @example Interval.fromDateTimes(dt1, dt2).toDuration(['hours', 'minutes', 'seconds']).toObject() //=> { hours: 24, minutes: 34, seconds: 49.257 }\n * @example Interval.fromDateTimes(dt1, dt2).toDuration('seconds').toObject() //=> { seconds: 88489.257 }\n * @return {Duration}\n */\n toDuration(unit, opts) {\n if (!this.isValid) {\n return Duration.invalid(this.invalidReason);\n }\n return this.e.diff(this.s, unit, opts);\n }\n\n /**\n * Run mapFn on the interval start and end, returning a new Interval from the resulting DateTimes\n * @param {function} mapFn\n * @return {Interval}\n * @example Interval.fromDateTimes(dt1, dt2).mapEndpoints(endpoint => endpoint.toUTC())\n * @example Interval.fromDateTimes(dt1, dt2).mapEndpoints(endpoint => endpoint.plus({ hours: 2 }))\n */\n mapEndpoints(mapFn) {\n return Interval.fromDateTimes(mapFn(this.s), mapFn(this.e));\n }\n}\n","import DateTime from \"./datetime.js\";\nimport Settings from \"./settings.js\";\nimport Locale from \"./impl/locale.js\";\nimport IANAZone from \"./zones/IANAZone.js\";\nimport { normalizeZone } from \"./impl/zoneUtil.js\";\n\nimport { hasFormatToParts, hasIntl, hasRelative } from \"./impl/util.js\";\n\n/**\n * The Info class contains static methods for retrieving general time and date related data. For example, it has methods for finding out if a time zone has a DST, for listing the months in any supported locale, and for discovering which of Luxon features are available in the current environment.\n */\nexport default class Info {\n /**\n * Return whether the specified zone contains a DST.\n * @param {string|Zone} [zone='local'] - Zone to check. Defaults to the environment's local zone.\n * @return {boolean}\n */\n static hasDST(zone = Settings.defaultZone) {\n const proto = DateTime.now()\n .setZone(zone)\n .set({ month: 12 });\n\n return !zone.universal && proto.offset !== proto.set({ month: 6 }).offset;\n }\n\n /**\n * Return whether the specified zone is a valid IANA specifier.\n * @param {string} zone - Zone to check\n * @return {boolean}\n */\n static isValidIANAZone(zone) {\n return IANAZone.isValidSpecifier(zone) && IANAZone.isValidZone(zone);\n }\n\n /**\n * Converts the input into a {@link Zone} instance.\n *\n * * If `input` is already a Zone instance, it is returned unchanged.\n * * If `input` is a string containing a valid time zone name, a Zone instance\n * with that name is returned.\n * * If `input` is a string that doesn't refer to a known time zone, a Zone\n * instance with {@link Zone.isValid} == false is returned.\n * * If `input is a number, a Zone instance with the specified fixed offset\n * in minutes is returned.\n * * If `input` is `null` or `undefined`, the default zone is returned.\n * @param {string|Zone|number} [input] - the value to be converted\n * @return {Zone}\n */\n static normalizeZone(input) {\n return normalizeZone(input, Settings.defaultZone);\n }\n\n /**\n * Return an array of standalone month names.\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat\n * @param {string} [length='long'] - the length of the month representation, such as \"numeric\", \"2-digit\", \"narrow\", \"short\", \"long\"\n * @param {Object} opts - options\n * @param {string} [opts.locale] - the locale code\n * @param {string} [opts.numberingSystem=null] - the numbering system\n * @param {string} [opts.locObj=null] - an existing locale object to use\n * @param {string} [opts.outputCalendar='gregory'] - the calendar\n * @example Info.months()[0] //=> 'January'\n * @example Info.months('short')[0] //=> 'Jan'\n * @example Info.months('numeric')[0] //=> '1'\n * @example Info.months('short', { locale: 'fr-CA' } )[0] //=> 'janv.'\n * @example Info.months('numeric', { locale: 'ar' })[0] //=> '١'\n * @example Info.months('long', { outputCalendar: 'islamic' })[0] //=> 'Rabiʻ I'\n * @return {[string]}\n */\n static months(\n length = \"long\",\n { locale = null, numberingSystem = null, locObj = null, outputCalendar = \"gregory\" } = {}\n ) {\n return (locObj || Locale.create(locale, numberingSystem, outputCalendar)).months(length);\n }\n\n /**\n * Return an array of format month names.\n * Format months differ from standalone months in that they're meant to appear next to the day of the month. In some languages, that\n * changes the string.\n * See {@link months}\n * @param {string} [length='long'] - the length of the month representation, such as \"numeric\", \"2-digit\", \"narrow\", \"short\", \"long\"\n * @param {Object} opts - options\n * @param {string} [opts.locale] - the locale code\n * @param {string} [opts.numberingSystem=null] - the numbering system\n * @param {string} [opts.locObj=null] - an existing locale object to use\n * @param {string} [opts.outputCalendar='gregory'] - the calendar\n * @return {[string]}\n */\n static monthsFormat(\n length = \"long\",\n { locale = null, numberingSystem = null, locObj = null, outputCalendar = \"gregory\" } = {}\n ) {\n return (locObj || Locale.create(locale, numberingSystem, outputCalendar)).months(length, true);\n }\n\n /**\n * Return an array of standalone week names.\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat\n * @param {string} [length='long'] - the length of the weekday representation, such as \"narrow\", \"short\", \"long\".\n * @param {Object} opts - options\n * @param {string} [opts.locale] - the locale code\n * @param {string} [opts.numberingSystem=null] - the numbering system\n * @param {string} [opts.locObj=null] - an existing locale object to use\n * @example Info.weekdays()[0] //=> 'Monday'\n * @example Info.weekdays('short')[0] //=> 'Mon'\n * @example Info.weekdays('short', { locale: 'fr-CA' })[0] //=> 'lun.'\n * @example Info.weekdays('short', { locale: 'ar' })[0] //=> 'الاثنين'\n * @return {[string]}\n */\n static weekdays(length = \"long\", { locale = null, numberingSystem = null, locObj = null } = {}) {\n return (locObj || Locale.create(locale, numberingSystem, null)).weekdays(length);\n }\n\n /**\n * Return an array of format week names.\n * Format weekdays differ from standalone weekdays in that they're meant to appear next to more date information. In some languages, that\n * changes the string.\n * See {@link weekdays}\n * @param {string} [length='long'] - the length of the weekday representation, such as \"narrow\", \"short\", \"long\".\n * @param {Object} opts - options\n * @param {string} [opts.locale=null] - the locale code\n * @param {string} [opts.numberingSystem=null] - the numbering system\n * @param {string} [opts.locObj=null] - an existing locale object to use\n * @return {[string]}\n */\n static weekdaysFormat(\n length = \"long\",\n { locale = null, numberingSystem = null, locObj = null } = {}\n ) {\n return (locObj || Locale.create(locale, numberingSystem, null)).weekdays(length, true);\n }\n\n /**\n * Return an array of meridiems.\n * @param {Object} opts - options\n * @param {string} [opts.locale] - the locale code\n * @example Info.meridiems() //=> [ 'AM', 'PM' ]\n * @example Info.meridiems({ locale: 'my' }) //=> [ 'နံနက်', 'ညနေ' ]\n * @return {[string]}\n */\n static meridiems({ locale = null } = {}) {\n return Locale.create(locale).meridiems();\n }\n\n /**\n * Return an array of eras, such as ['BC', 'AD']. The locale can be specified, but the calendar system is always Gregorian.\n * @param {string} [length='short'] - the length of the era representation, such as \"short\" or \"long\".\n * @param {Object} opts - options\n * @param {string} [opts.locale] - the locale code\n * @example Info.eras() //=> [ 'BC', 'AD' ]\n * @example Info.eras('long') //=> [ 'Before Christ', 'Anno Domini' ]\n * @example Info.eras('long', { locale: 'fr' }) //=> [ 'avant Jésus-Christ', 'après Jésus-Christ' ]\n * @return {[string]}\n */\n static eras(length = \"short\", { locale = null } = {}) {\n return Locale.create(locale, null, \"gregory\").eras(length);\n }\n\n /**\n * Return the set of available features in this environment.\n * Some features of Luxon are not available in all environments. For example, on older browsers, timezone support is not available. Use this function to figure out if that's the case.\n * Keys:\n * * `zones`: whether this environment supports IANA timezones\n * * `intlTokens`: whether this environment supports internationalized token-based formatting/parsing\n * * `intl`: whether this environment supports general internationalization\n * * `relative`: whether this environment supports relative time formatting\n * @example Info.features() //=> { intl: true, intlTokens: false, zones: true, relative: false }\n * @return {Object}\n */\n static features() {\n let intl = false,\n intlTokens = false,\n zones = false,\n relative = false;\n\n if (hasIntl()) {\n intl = true;\n intlTokens = hasFormatToParts();\n relative = hasRelative();\n\n try {\n zones =\n new Intl.DateTimeFormat(\"en\", { timeZone: \"America/New_York\" }).resolvedOptions()\n .timeZone === \"America/New_York\";\n } catch (e) {\n zones = false;\n }\n }\n\n return { intl, intlTokens, zones, relative };\n }\n}\n","import Duration from \"../duration.js\";\n\nfunction dayDiff(earlier, later) {\n const utcDayStart = dt =>\n dt\n .toUTC(0, { keepLocalTime: true })\n .startOf(\"day\")\n .valueOf(),\n ms = utcDayStart(later) - utcDayStart(earlier);\n return Math.floor(Duration.fromMillis(ms).as(\"days\"));\n}\n\nfunction highOrderDiffs(cursor, later, units) {\n const differs = [\n [\"years\", (a, b) => b.year - a.year],\n [\"quarters\", (a, b) => b.quarter - a.quarter],\n [\"months\", (a, b) => b.month - a.month + (b.year - a.year) * 12],\n [\n \"weeks\",\n (a, b) => {\n const days = dayDiff(a, b);\n return (days - (days % 7)) / 7;\n }\n ],\n [\"days\", dayDiff]\n ];\n\n const results = {};\n let lowestOrder, highWater;\n\n for (const [unit, differ] of differs) {\n if (units.indexOf(unit) >= 0) {\n lowestOrder = unit;\n\n let delta = differ(cursor, later);\n highWater = cursor.plus({ [unit]: delta });\n\n if (highWater > later) {\n cursor = cursor.plus({ [unit]: delta - 1 });\n delta -= 1;\n } else {\n cursor = highWater;\n }\n\n results[unit] = delta;\n }\n }\n\n return [cursor, results, highWater, lowestOrder];\n}\n\nexport default function(earlier, later, units, opts) {\n let [cursor, results, highWater, lowestOrder] = highOrderDiffs(earlier, later, units);\n\n const remainingMillis = later - cursor;\n\n const lowerOrderUnits = units.filter(\n u => [\"hours\", \"minutes\", \"seconds\", \"milliseconds\"].indexOf(u) >= 0\n );\n\n if (lowerOrderUnits.length === 0) {\n if (highWater < later) {\n highWater = cursor.plus({ [lowestOrder]: 1 });\n }\n\n if (highWater !== cursor) {\n results[lowestOrder] = (results[lowestOrder] || 0) + remainingMillis / (highWater - cursor);\n }\n }\n\n const duration = Duration.fromObject(Object.assign(results, opts));\n\n if (lowerOrderUnits.length > 0) {\n return Duration.fromMillis(remainingMillis, opts)\n .shiftTo(...lowerOrderUnits)\n .plus(duration);\n } else {\n return duration;\n }\n}\n","const numberingSystems = {\n arab: \"[\\u0660-\\u0669]\",\n arabext: \"[\\u06F0-\\u06F9]\",\n bali: \"[\\u1B50-\\u1B59]\",\n beng: \"[\\u09E6-\\u09EF]\",\n deva: \"[\\u0966-\\u096F]\",\n fullwide: \"[\\uFF10-\\uFF19]\",\n gujr: \"[\\u0AE6-\\u0AEF]\",\n hanidec: \"[〇|一|二|三|四|五|六|七|八|九]\",\n khmr: \"[\\u17E0-\\u17E9]\",\n knda: \"[\\u0CE6-\\u0CEF]\",\n laoo: \"[\\u0ED0-\\u0ED9]\",\n limb: \"[\\u1946-\\u194F]\",\n mlym: \"[\\u0D66-\\u0D6F]\",\n mong: \"[\\u1810-\\u1819]\",\n mymr: \"[\\u1040-\\u1049]\",\n orya: \"[\\u0B66-\\u0B6F]\",\n tamldec: \"[\\u0BE6-\\u0BEF]\",\n telu: \"[\\u0C66-\\u0C6F]\",\n thai: \"[\\u0E50-\\u0E59]\",\n tibt: \"[\\u0F20-\\u0F29]\",\n latn: \"\\\\d\"\n};\n\nconst numberingSystemsUTF16 = {\n arab: [1632, 1641],\n arabext: [1776, 1785],\n bali: [6992, 7001],\n beng: [2534, 2543],\n deva: [2406, 2415],\n fullwide: [65296, 65303],\n gujr: [2790, 2799],\n khmr: [6112, 6121],\n knda: [3302, 3311],\n laoo: [3792, 3801],\n limb: [6470, 6479],\n mlym: [3430, 3439],\n mong: [6160, 6169],\n mymr: [4160, 4169],\n orya: [2918, 2927],\n tamldec: [3046, 3055],\n telu: [3174, 3183],\n thai: [3664, 3673],\n tibt: [3872, 3881]\n};\n\n// eslint-disable-next-line\nconst hanidecChars = numberingSystems.hanidec.replace(/[\\[|\\]]/g, \"\").split(\"\");\n\nexport function parseDigits(str) {\n let value = parseInt(str, 10);\n if (isNaN(value)) {\n value = \"\";\n for (let i = 0; i < str.length; i++) {\n const code = str.charCodeAt(i);\n\n if (str[i].search(numberingSystems.hanidec) !== -1) {\n value += hanidecChars.indexOf(str[i]);\n } else {\n for (const key in numberingSystemsUTF16) {\n const [min, max] = numberingSystemsUTF16[key];\n if (code >= min && code <= max) {\n value += code - min;\n }\n }\n }\n }\n return parseInt(value, 10);\n } else {\n return value;\n }\n}\n\nexport function digitRegex({ numberingSystem }, append = \"\") {\n return new RegExp(`${numberingSystems[numberingSystem || \"latn\"]}${append}`);\n}\n","import { parseMillis, isUndefined, untruncateYear, signedOffset, hasOwnProperty } from \"./util.js\";\nimport Formatter from \"./formatter.js\";\nimport FixedOffsetZone from \"../zones/fixedOffsetZone.js\";\nimport IANAZone from \"../zones/IANAZone.js\";\nimport DateTime from \"../datetime.js\";\nimport { digitRegex, parseDigits } from \"./digits.js\";\nimport { ConflictingSpecificationError } from \"../errors.js\";\n\nconst MISSING_FTP = \"missing Intl.DateTimeFormat.formatToParts support\";\n\nfunction intUnit(regex, post = i => i) {\n return { regex, deser: ([s]) => post(parseDigits(s)) };\n}\n\nconst NBSP = String.fromCharCode(160);\nconst spaceOrNBSP = `( |${NBSP})`;\nconst spaceOrNBSPRegExp = new RegExp(spaceOrNBSP, \"g\");\n\nfunction fixListRegex(s) {\n // make dots optional and also make them literal\n // make space and non breakable space characters interchangeable\n return s.replace(/\\./g, \"\\\\.?\").replace(spaceOrNBSPRegExp, spaceOrNBSP);\n}\n\nfunction stripInsensitivities(s) {\n return s\n .replace(/\\./g, \"\") // ignore dots that were made optional\n .replace(spaceOrNBSPRegExp, \" \") // interchange space and nbsp\n .toLowerCase();\n}\n\nfunction oneOf(strings, startIndex) {\n if (strings === null) {\n return null;\n } else {\n return {\n regex: RegExp(strings.map(fixListRegex).join(\"|\")),\n deser: ([s]) =>\n strings.findIndex(i => stripInsensitivities(s) === stripInsensitivities(i)) + startIndex\n };\n }\n}\n\nfunction offset(regex, groups) {\n return { regex, deser: ([, h, m]) => signedOffset(h, m), groups };\n}\n\nfunction simple(regex) {\n return { regex, deser: ([s]) => s };\n}\n\nfunction escapeToken(value) {\n // eslint-disable-next-line no-useless-escape\n return value.replace(/[\\-\\[\\]{}()*+?.,\\\\\\^$|#\\s]/g, \"\\\\$&\");\n}\n\nfunction unitForToken(token, loc) {\n const one = digitRegex(loc),\n two = digitRegex(loc, \"{2}\"),\n three = digitRegex(loc, \"{3}\"),\n four = digitRegex(loc, \"{4}\"),\n six = digitRegex(loc, \"{6}\"),\n oneOrTwo = digitRegex(loc, \"{1,2}\"),\n oneToThree = digitRegex(loc, \"{1,3}\"),\n oneToSix = digitRegex(loc, \"{1,6}\"),\n oneToNine = digitRegex(loc, \"{1,9}\"),\n twoToFour = digitRegex(loc, \"{2,4}\"),\n fourToSix = digitRegex(loc, \"{4,6}\"),\n literal = t => ({ regex: RegExp(escapeToken(t.val)), deser: ([s]) => s, literal: true }),\n unitate = t => {\n if (token.literal) {\n return literal(t);\n }\n switch (t.val) {\n // era\n case \"G\":\n return oneOf(loc.eras(\"short\", false), 0);\n case \"GG\":\n return oneOf(loc.eras(\"long\", false), 0);\n // years\n case \"y\":\n return intUnit(oneToSix);\n case \"yy\":\n return intUnit(twoToFour, untruncateYear);\n case \"yyyy\":\n return intUnit(four);\n case \"yyyyy\":\n return intUnit(fourToSix);\n case \"yyyyyy\":\n return intUnit(six);\n // months\n case \"M\":\n return intUnit(oneOrTwo);\n case \"MM\":\n return intUnit(two);\n case \"MMM\":\n return oneOf(loc.months(\"short\", true, false), 1);\n case \"MMMM\":\n return oneOf(loc.months(\"long\", true, false), 1);\n case \"L\":\n return intUnit(oneOrTwo);\n case \"LL\":\n return intUnit(two);\n case \"LLL\":\n return oneOf(loc.months(\"short\", false, false), 1);\n case \"LLLL\":\n return oneOf(loc.months(\"long\", false, false), 1);\n // dates\n case \"d\":\n return intUnit(oneOrTwo);\n case \"dd\":\n return intUnit(two);\n // ordinals\n case \"o\":\n return intUnit(oneToThree);\n case \"ooo\":\n return intUnit(three);\n // time\n case \"HH\":\n return intUnit(two);\n case \"H\":\n return intUnit(oneOrTwo);\n case \"hh\":\n return intUnit(two);\n case \"h\":\n return intUnit(oneOrTwo);\n case \"mm\":\n return intUnit(two);\n case \"m\":\n return intUnit(oneOrTwo);\n case \"q\":\n return intUnit(oneOrTwo);\n case \"qq\":\n return intUnit(two);\n case \"s\":\n return intUnit(oneOrTwo);\n case \"ss\":\n return intUnit(two);\n case \"S\":\n return intUnit(oneToThree);\n case \"SSS\":\n return intUnit(three);\n case \"u\":\n return simple(oneToNine);\n // meridiem\n case \"a\":\n return oneOf(loc.meridiems(), 0);\n // weekYear (k)\n case \"kkkk\":\n return intUnit(four);\n case \"kk\":\n return intUnit(twoToFour, untruncateYear);\n // weekNumber (W)\n case \"W\":\n return intUnit(oneOrTwo);\n case \"WW\":\n return intUnit(two);\n // weekdays\n case \"E\":\n case \"c\":\n return intUnit(one);\n case \"EEE\":\n return oneOf(loc.weekdays(\"short\", false, false), 1);\n case \"EEEE\":\n return oneOf(loc.weekdays(\"long\", false, false), 1);\n case \"ccc\":\n return oneOf(loc.weekdays(\"short\", true, false), 1);\n case \"cccc\":\n return oneOf(loc.weekdays(\"long\", true, false), 1);\n // offset/zone\n case \"Z\":\n case \"ZZ\":\n return offset(new RegExp(`([+-]${oneOrTwo.source})(?::(${two.source}))?`), 2);\n case \"ZZZ\":\n return offset(new RegExp(`([+-]${oneOrTwo.source})(${two.source})?`), 2);\n // we don't support ZZZZ (PST) or ZZZZZ (Pacific Standard Time) in parsing\n // because we don't have any way to figure out what they are\n case \"z\":\n return simple(/[a-z_+-/]{1,256}?/i);\n default:\n return literal(t);\n }\n };\n\n const unit = unitate(token) || {\n invalidReason: MISSING_FTP\n };\n\n unit.token = token;\n\n return unit;\n}\n\nconst partTypeStyleToTokenVal = {\n year: {\n \"2-digit\": \"yy\",\n numeric: \"yyyyy\"\n },\n month: {\n numeric: \"M\",\n \"2-digit\": \"MM\",\n short: \"MMM\",\n long: \"MMMM\"\n },\n day: {\n numeric: \"d\",\n \"2-digit\": \"dd\"\n },\n weekday: {\n short: \"EEE\",\n long: \"EEEE\"\n },\n dayperiod: \"a\",\n dayPeriod: \"a\",\n hour: {\n numeric: \"h\",\n \"2-digit\": \"hh\"\n },\n minute: {\n numeric: \"m\",\n \"2-digit\": \"mm\"\n },\n second: {\n numeric: \"s\",\n \"2-digit\": \"ss\"\n }\n};\n\nfunction tokenForPart(part, locale, formatOpts) {\n const { type, value } = part;\n\n if (type === \"literal\") {\n return {\n literal: true,\n val: value\n };\n }\n\n const style = formatOpts[type];\n\n let val = partTypeStyleToTokenVal[type];\n if (typeof val === \"object\") {\n val = val[style];\n }\n\n if (val) {\n return {\n literal: false,\n val\n };\n }\n\n return undefined;\n}\n\nfunction buildRegex(units) {\n const re = units.map(u => u.regex).reduce((f, r) => `${f}(${r.source})`, \"\");\n return [`^${re}$`, units];\n}\n\nfunction match(input, regex, handlers) {\n const matches = input.match(regex);\n\n if (matches) {\n const all = {};\n let matchIndex = 1;\n for (const i in handlers) {\n if (hasOwnProperty(handlers, i)) {\n const h = handlers[i],\n groups = h.groups ? h.groups + 1 : 1;\n if (!h.literal && h.token) {\n all[h.token.val[0]] = h.deser(matches.slice(matchIndex, matchIndex + groups));\n }\n matchIndex += groups;\n }\n }\n return [matches, all];\n } else {\n return [matches, {}];\n }\n}\n\nfunction dateTimeFromMatches(matches) {\n const toField = token => {\n switch (token) {\n case \"S\":\n return \"millisecond\";\n case \"s\":\n return \"second\";\n case \"m\":\n return \"minute\";\n case \"h\":\n case \"H\":\n return \"hour\";\n case \"d\":\n return \"day\";\n case \"o\":\n return \"ordinal\";\n case \"L\":\n case \"M\":\n return \"month\";\n case \"y\":\n return \"year\";\n case \"E\":\n case \"c\":\n return \"weekday\";\n case \"W\":\n return \"weekNumber\";\n case \"k\":\n return \"weekYear\";\n case \"q\":\n return \"quarter\";\n default:\n return null;\n }\n };\n\n let zone;\n if (!isUndefined(matches.Z)) {\n zone = new FixedOffsetZone(matches.Z);\n } else if (!isUndefined(matches.z)) {\n zone = IANAZone.create(matches.z);\n } else {\n zone = null;\n }\n\n if (!isUndefined(matches.q)) {\n matches.M = (matches.q - 1) * 3 + 1;\n }\n\n if (!isUndefined(matches.h)) {\n if (matches.h < 12 && matches.a === 1) {\n matches.h += 12;\n } else if (matches.h === 12 && matches.a === 0) {\n matches.h = 0;\n }\n }\n\n if (matches.G === 0 && matches.y) {\n matches.y = -matches.y;\n }\n\n if (!isUndefined(matches.u)) {\n matches.S = parseMillis(matches.u);\n }\n\n const vals = Object.keys(matches).reduce((r, k) => {\n const f = toField(k);\n if (f) {\n r[f] = matches[k];\n }\n\n return r;\n }, {});\n\n return [vals, zone];\n}\n\nlet dummyDateTimeCache = null;\n\nfunction getDummyDateTime() {\n if (!dummyDateTimeCache) {\n dummyDateTimeCache = DateTime.fromMillis(1555555555555);\n }\n\n return dummyDateTimeCache;\n}\n\nfunction maybeExpandMacroToken(token, locale) {\n if (token.literal) {\n return token;\n }\n\n const formatOpts = Formatter.macroTokenToFormatOpts(token.val);\n\n if (!formatOpts) {\n return token;\n }\n\n const formatter = Formatter.create(locale, formatOpts);\n const parts = formatter.formatDateTimeParts(getDummyDateTime());\n\n const tokens = parts.map(p => tokenForPart(p, locale, formatOpts));\n\n if (tokens.includes(undefined)) {\n return token;\n }\n\n return tokens;\n}\n\nfunction expandMacroTokens(tokens, locale) {\n return Array.prototype.concat(...tokens.map(t => maybeExpandMacroToken(t, locale)));\n}\n\n/**\n * @private\n */\n\nexport function explainFromTokens(locale, input, format) {\n const tokens = expandMacroTokens(Formatter.parseFormat(format), locale),\n units = tokens.map(t => unitForToken(t, locale)),\n disqualifyingUnit = units.find(t => t.invalidReason);\n\n if (disqualifyingUnit) {\n return { input, tokens, invalidReason: disqualifyingUnit.invalidReason };\n } else {\n const [regexString, handlers] = buildRegex(units),\n regex = RegExp(regexString, \"i\"),\n [rawMatches, matches] = match(input, regex, handlers),\n [result, zone] = matches ? dateTimeFromMatches(matches) : [null, null];\n if (hasOwnProperty(matches, \"a\") && hasOwnProperty(matches, \"H\")) {\n throw new ConflictingSpecificationError(\n \"Can't include meridiem when specifying 24-hour format\"\n );\n }\n return { input, tokens, regex, rawMatches, matches, result, zone };\n }\n}\n\nexport function parseFromTokens(locale, input, format) {\n const { result, zone, invalidReason } = explainFromTokens(locale, input, format);\n return [result, zone, invalidReason];\n}\n","import {\n integerBetween,\n isLeapYear,\n timeObject,\n daysInYear,\n daysInMonth,\n weeksInWeekYear,\n isInteger\n} from \"./util.js\";\nimport Invalid from \"./invalid.js\";\n\nconst nonLeapLadder = [0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334],\n leapLadder = [0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335];\n\nfunction unitOutOfRange(unit, value) {\n return new Invalid(\n \"unit out of range\",\n `you specified ${value} (of type ${typeof value}) as a ${unit}, which is invalid`\n );\n}\n\nfunction dayOfWeek(year, month, day) {\n const js = new Date(Date.UTC(year, month - 1, day)).getUTCDay();\n return js === 0 ? 7 : js;\n}\n\nfunction computeOrdinal(year, month, day) {\n return day + (isLeapYear(year) ? leapLadder : nonLeapLadder)[month - 1];\n}\n\nfunction uncomputeOrdinal(year, ordinal) {\n const table = isLeapYear(year) ? leapLadder : nonLeapLadder,\n month0 = table.findIndex(i => i < ordinal),\n day = ordinal - table[month0];\n return { month: month0 + 1, day };\n}\n\n/**\n * @private\n */\n\nexport function gregorianToWeek(gregObj) {\n const { year, month, day } = gregObj,\n ordinal = computeOrdinal(year, month, day),\n weekday = dayOfWeek(year, month, day);\n\n let weekNumber = Math.floor((ordinal - weekday + 10) / 7),\n weekYear;\n\n if (weekNumber < 1) {\n weekYear = year - 1;\n weekNumber = weeksInWeekYear(weekYear);\n } else if (weekNumber > weeksInWeekYear(year)) {\n weekYear = year + 1;\n weekNumber = 1;\n } else {\n weekYear = year;\n }\n\n return Object.assign({ weekYear, weekNumber, weekday }, timeObject(gregObj));\n}\n\nexport function weekToGregorian(weekData) {\n const { weekYear, weekNumber, weekday } = weekData,\n weekdayOfJan4 = dayOfWeek(weekYear, 1, 4),\n yearInDays = daysInYear(weekYear);\n\n let ordinal = weekNumber * 7 + weekday - weekdayOfJan4 - 3,\n year;\n\n if (ordinal < 1) {\n year = weekYear - 1;\n ordinal += daysInYear(year);\n } else if (ordinal > yearInDays) {\n year = weekYear + 1;\n ordinal -= daysInYear(weekYear);\n } else {\n year = weekYear;\n }\n\n const { month, day } = uncomputeOrdinal(year, ordinal);\n\n return Object.assign({ year, month, day }, timeObject(weekData));\n}\n\nexport function gregorianToOrdinal(gregData) {\n const { year, month, day } = gregData,\n ordinal = computeOrdinal(year, month, day);\n\n return Object.assign({ year, ordinal }, timeObject(gregData));\n}\n\nexport function ordinalToGregorian(ordinalData) {\n const { year, ordinal } = ordinalData,\n { month, day } = uncomputeOrdinal(year, ordinal);\n\n return Object.assign({ year, month, day }, timeObject(ordinalData));\n}\n\nexport function hasInvalidWeekData(obj) {\n const validYear = isInteger(obj.weekYear),\n validWeek = integerBetween(obj.weekNumber, 1, weeksInWeekYear(obj.weekYear)),\n validWeekday = integerBetween(obj.weekday, 1, 7);\n\n if (!validYear) {\n return unitOutOfRange(\"weekYear\", obj.weekYear);\n } else if (!validWeek) {\n return unitOutOfRange(\"week\", obj.week);\n } else if (!validWeekday) {\n return unitOutOfRange(\"weekday\", obj.weekday);\n } else return false;\n}\n\nexport function hasInvalidOrdinalData(obj) {\n const validYear = isInteger(obj.year),\n validOrdinal = integerBetween(obj.ordinal, 1, daysInYear(obj.year));\n\n if (!validYear) {\n return unitOutOfRange(\"year\", obj.year);\n } else if (!validOrdinal) {\n return unitOutOfRange(\"ordinal\", obj.ordinal);\n } else return false;\n}\n\nexport function hasInvalidGregorianData(obj) {\n const validYear = isInteger(obj.year),\n validMonth = integerBetween(obj.month, 1, 12),\n validDay = integerBetween(obj.day, 1, daysInMonth(obj.year, obj.month));\n\n if (!validYear) {\n return unitOutOfRange(\"year\", obj.year);\n } else if (!validMonth) {\n return unitOutOfRange(\"month\", obj.month);\n } else if (!validDay) {\n return unitOutOfRange(\"day\", obj.day);\n } else return false;\n}\n\nexport function hasInvalidTimeData(obj) {\n const { hour, minute, second, millisecond } = obj;\n const validHour =\n integerBetween(hour, 0, 23) ||\n (hour === 24 && minute === 0 && second === 0 && millisecond === 0),\n validMinute = integerBetween(minute, 0, 59),\n validSecond = integerBetween(second, 0, 59),\n validMillisecond = integerBetween(millisecond, 0, 999);\n\n if (!validHour) {\n return unitOutOfRange(\"hour\", hour);\n } else if (!validMinute) {\n return unitOutOfRange(\"minute\", minute);\n } else if (!validSecond) {\n return unitOutOfRange(\"second\", second);\n } else if (!validMillisecond) {\n return unitOutOfRange(\"millisecond\", millisecond);\n } else return false;\n}\n","import Duration, { friendlyDuration } from \"./duration.js\";\nimport Interval from \"./interval.js\";\nimport Settings from \"./settings.js\";\nimport Info from \"./info.js\";\nimport Formatter from \"./impl/formatter.js\";\nimport FixedOffsetZone from \"./zones/fixedOffsetZone.js\";\nimport Locale from \"./impl/locale.js\";\nimport {\n isUndefined,\n maybeArray,\n isDate,\n isNumber,\n bestBy,\n daysInMonth,\n daysInYear,\n isLeapYear,\n weeksInWeekYear,\n normalizeObject,\n roundTo,\n objToLocalTS\n} from \"./impl/util.js\";\nimport { normalizeZone } from \"./impl/zoneUtil.js\";\nimport diff from \"./impl/diff.js\";\nimport { parseRFC2822Date, parseISODate, parseHTTPDate, parseSQL } from \"./impl/regexParser.js\";\nimport { parseFromTokens, explainFromTokens } from \"./impl/tokenParser.js\";\nimport {\n gregorianToWeek,\n weekToGregorian,\n gregorianToOrdinal,\n ordinalToGregorian,\n hasInvalidGregorianData,\n hasInvalidWeekData,\n hasInvalidOrdinalData,\n hasInvalidTimeData\n} from \"./impl/conversions.js\";\nimport * as Formats from \"./impl/formats.js\";\nimport {\n InvalidArgumentError,\n ConflictingSpecificationError,\n InvalidUnitError,\n InvalidDateTimeError\n} from \"./errors.js\";\nimport Invalid from \"./impl/invalid.js\";\n\nconst INVALID = \"Invalid DateTime\";\nconst MAX_DATE = 8.64e15;\n\nfunction unsupportedZone(zone) {\n return new Invalid(\"unsupported zone\", `the zone \"${zone.name}\" is not supported`);\n}\n\n// we cache week data on the DT object and this intermediates the cache\nfunction possiblyCachedWeekData(dt) {\n if (dt.weekData === null) {\n dt.weekData = gregorianToWeek(dt.c);\n }\n return dt.weekData;\n}\n\n// clone really means, \"make a new object with these modifications\". all \"setters\" really use this\n// to create a new object while only changing some of the properties\nfunction clone(inst, alts) {\n const current = {\n ts: inst.ts,\n zone: inst.zone,\n c: inst.c,\n o: inst.o,\n loc: inst.loc,\n invalid: inst.invalid\n };\n return new DateTime(Object.assign({}, current, alts, { old: current }));\n}\n\n// find the right offset a given local time. The o input is our guess, which determines which\n// offset we'll pick in ambiguous cases (e.g. there are two 3 AMs b/c Fallback DST)\nfunction fixOffset(localTS, o, tz) {\n // Our UTC time is just a guess because our offset is just a guess\n let utcGuess = localTS - o * 60 * 1000;\n\n // Test whether the zone matches the offset for this ts\n const o2 = tz.offset(utcGuess);\n\n // If so, offset didn't change and we're done\n if (o === o2) {\n return [utcGuess, o];\n }\n\n // If not, change the ts by the difference in the offset\n utcGuess -= (o2 - o) * 60 * 1000;\n\n // If that gives us the local time we want, we're done\n const o3 = tz.offset(utcGuess);\n if (o2 === o3) {\n return [utcGuess, o2];\n }\n\n // If it's different, we're in a hole time. The offset has changed, but the we don't adjust the time\n return [localTS - Math.min(o2, o3) * 60 * 1000, Math.max(o2, o3)];\n}\n\n// convert an epoch timestamp into a calendar object with the given offset\nfunction tsToObj(ts, offset) {\n ts += offset * 60 * 1000;\n\n const d = new Date(ts);\n\n return {\n year: d.getUTCFullYear(),\n month: d.getUTCMonth() + 1,\n day: d.getUTCDate(),\n hour: d.getUTCHours(),\n minute: d.getUTCMinutes(),\n second: d.getUTCSeconds(),\n millisecond: d.getUTCMilliseconds()\n };\n}\n\n// convert a calendar object to a epoch timestamp\nfunction objToTS(obj, offset, zone) {\n return fixOffset(objToLocalTS(obj), offset, zone);\n}\n\n// create a new DT instance by adding a duration, adjusting for DSTs\nfunction adjustTime(inst, dur) {\n const oPre = inst.o,\n year = inst.c.year + Math.trunc(dur.years),\n month = inst.c.month + Math.trunc(dur.months) + Math.trunc(dur.quarters) * 3,\n c = Object.assign({}, inst.c, {\n year,\n month,\n day:\n Math.min(inst.c.day, daysInMonth(year, month)) +\n Math.trunc(dur.days) +\n Math.trunc(dur.weeks) * 7\n }),\n millisToAdd = Duration.fromObject({\n years: dur.years - Math.trunc(dur.years),\n quarters: dur.quarters - Math.trunc(dur.quarters),\n months: dur.months - Math.trunc(dur.months),\n weeks: dur.weeks - Math.trunc(dur.weeks),\n days: dur.days - Math.trunc(dur.days),\n hours: dur.hours,\n minutes: dur.minutes,\n seconds: dur.seconds,\n milliseconds: dur.milliseconds\n }).as(\"milliseconds\"),\n localTS = objToLocalTS(c);\n\n let [ts, o] = fixOffset(localTS, oPre, inst.zone);\n\n if (millisToAdd !== 0) {\n ts += millisToAdd;\n // that could have changed the offset by going over a DST, but we want to keep the ts the same\n o = inst.zone.offset(ts);\n }\n\n return { ts, o };\n}\n\n// helper useful in turning the results of parsing into real dates\n// by handling the zone options\nfunction parseDataToDateTime(parsed, parsedZone, opts, format, text) {\n const { setZone, zone } = opts;\n if (parsed && Object.keys(parsed).length !== 0) {\n const interpretationZone = parsedZone || zone,\n inst = DateTime.fromObject(\n Object.assign(parsed, opts, {\n zone: interpretationZone,\n // setZone is a valid option in the calling methods, but not in fromObject\n setZone: undefined\n })\n );\n return setZone ? inst : inst.setZone(zone);\n } else {\n return DateTime.invalid(\n new Invalid(\"unparsable\", `the input \"${text}\" can't be parsed as ${format}`)\n );\n }\n}\n\n// if you want to output a technical format (e.g. RFC 2822), this helper\n// helps handle the details\nfunction toTechFormat(dt, format, allowZ = true) {\n return dt.isValid\n ? Formatter.create(Locale.create(\"en-US\"), {\n allowZ,\n forceSimple: true\n }).formatDateTimeFromString(dt, format)\n : null;\n}\n\n// technical time formats (e.g. the time part of ISO 8601), take some options\n// and this commonizes their handling\nfunction toTechTimeFormat(\n dt,\n {\n suppressSeconds = false,\n suppressMilliseconds = false,\n includeOffset,\n includePrefix = false,\n includeZone = false,\n spaceZone = false,\n format = \"extended\"\n }\n) {\n let fmt = format === \"basic\" ? \"HHmm\" : \"HH:mm\";\n\n if (!suppressSeconds || dt.second !== 0 || dt.millisecond !== 0) {\n fmt += format === \"basic\" ? \"ss\" : \":ss\";\n if (!suppressMilliseconds || dt.millisecond !== 0) {\n fmt += \".SSS\";\n }\n }\n\n if ((includeZone || includeOffset) && spaceZone) {\n fmt += \" \";\n }\n\n if (includeZone) {\n fmt += \"z\";\n } else if (includeOffset) {\n fmt += format === \"basic\" ? \"ZZZ\" : \"ZZ\";\n }\n\n let str = toTechFormat(dt, fmt);\n\n if (includePrefix) {\n str = \"T\" + str;\n }\n\n return str;\n}\n\n// defaults for unspecified units in the supported calendars\nconst defaultUnitValues = {\n month: 1,\n day: 1,\n hour: 0,\n minute: 0,\n second: 0,\n millisecond: 0\n },\n defaultWeekUnitValues = {\n weekNumber: 1,\n weekday: 1,\n hour: 0,\n minute: 0,\n second: 0,\n millisecond: 0\n },\n defaultOrdinalUnitValues = {\n ordinal: 1,\n hour: 0,\n minute: 0,\n second: 0,\n millisecond: 0\n };\n\n// Units in the supported calendars, sorted by bigness\nconst orderedUnits = [\"year\", \"month\", \"day\", \"hour\", \"minute\", \"second\", \"millisecond\"],\n orderedWeekUnits = [\n \"weekYear\",\n \"weekNumber\",\n \"weekday\",\n \"hour\",\n \"minute\",\n \"second\",\n \"millisecond\"\n ],\n orderedOrdinalUnits = [\"year\", \"ordinal\", \"hour\", \"minute\", \"second\", \"millisecond\"];\n\n// standardize case and plurality in units\nfunction normalizeUnit(unit) {\n const normalized = {\n year: \"year\",\n years: \"year\",\n month: \"month\",\n months: \"month\",\n day: \"day\",\n days: \"day\",\n hour: \"hour\",\n hours: \"hour\",\n minute: \"minute\",\n minutes: \"minute\",\n quarter: \"quarter\",\n quarters: \"quarter\",\n second: \"second\",\n seconds: \"second\",\n millisecond: \"millisecond\",\n milliseconds: \"millisecond\",\n weekday: \"weekday\",\n weekdays: \"weekday\",\n weeknumber: \"weekNumber\",\n weeksnumber: \"weekNumber\",\n weeknumbers: \"weekNumber\",\n weekyear: \"weekYear\",\n weekyears: \"weekYear\",\n ordinal: \"ordinal\"\n }[unit.toLowerCase()];\n\n if (!normalized) throw new InvalidUnitError(unit);\n\n return normalized;\n}\n\n// this is a dumbed down version of fromObject() that runs about 60% faster\n// but doesn't do any validation, makes a bunch of assumptions about what units\n// are present, and so on.\nfunction quickDT(obj, zone) {\n // assume we have the higher-order units\n for (const u of orderedUnits) {\n if (isUndefined(obj[u])) {\n obj[u] = defaultUnitValues[u];\n }\n }\n\n const invalid = hasInvalidGregorianData(obj) || hasInvalidTimeData(obj);\n if (invalid) {\n return DateTime.invalid(invalid);\n }\n\n const tsNow = Settings.now(),\n offsetProvis = zone.offset(tsNow),\n [ts, o] = objToTS(obj, offsetProvis, zone);\n\n return new DateTime({\n ts,\n zone,\n o\n });\n}\n\nfunction diffRelative(start, end, opts) {\n const round = isUndefined(opts.round) ? true : opts.round,\n format = (c, unit) => {\n c = roundTo(c, round || opts.calendary ? 0 : 2, true);\n const formatter = end.loc.clone(opts).relFormatter(opts);\n return formatter.format(c, unit);\n },\n differ = unit => {\n if (opts.calendary) {\n if (!end.hasSame(start, unit)) {\n return end\n .startOf(unit)\n .diff(start.startOf(unit), unit)\n .get(unit);\n } else return 0;\n } else {\n return end.diff(start, unit).get(unit);\n }\n };\n\n if (opts.unit) {\n return format(differ(opts.unit), opts.unit);\n }\n\n for (const unit of opts.units) {\n const count = differ(unit);\n if (Math.abs(count) >= 1) {\n return format(count, unit);\n }\n }\n return format(start > end ? -0 : 0, opts.units[opts.units.length - 1]);\n}\n\n/**\n * A DateTime is an immutable data structure representing a specific date and time and accompanying methods. It contains class and instance methods for creating, parsing, interrogating, transforming, and formatting them.\n *\n * A DateTime comprises of:\n * * A timestamp. Each DateTime instance refers to a specific millisecond of the Unix epoch.\n * * A time zone. Each instance is considered in the context of a specific zone (by default the local system's zone).\n * * Configuration properties that effect how output strings are formatted, such as `locale`, `numberingSystem`, and `outputCalendar`.\n *\n * Here is a brief overview of the most commonly used functionality it provides:\n *\n * * **Creation**: To create a DateTime from its components, use one of its factory class methods: {@link local}, {@link utc}, and (most flexibly) {@link fromObject}. To create one from a standard string format, use {@link fromISO}, {@link fromHTTP}, and {@link fromRFC2822}. To create one from a custom string format, use {@link fromFormat}. To create one from a native JS date, use {@link fromJSDate}.\n * * **Gregorian calendar and time**: To examine the Gregorian properties of a DateTime individually (i.e as opposed to collectively through {@link toObject}), use the {@link year}, {@link month},\n * {@link day}, {@link hour}, {@link minute}, {@link second}, {@link millisecond} accessors.\n * * **Week calendar**: For ISO week calendar attributes, see the {@link weekYear}, {@link weekNumber}, and {@link weekday} accessors.\n * * **Configuration** See the {@link locale} and {@link numberingSystem} accessors.\n * * **Transformation**: To transform the DateTime into other DateTimes, use {@link set}, {@link reconfigure}, {@link setZone}, {@link setLocale}, {@link plus}, {@link minus}, {@link endOf}, {@link startOf}, {@link toUTC}, and {@link toLocal}.\n * * **Output**: To convert the DateTime to other representations, use the {@link toRelative}, {@link toRelativeCalendar}, {@link toJSON}, {@link toISO}, {@link toHTTP}, {@link toObject}, {@link toRFC2822}, {@link toString}, {@link toLocaleString}, {@link toFormat}, {@link toMillis} and {@link toJSDate}.\n *\n * There's plenty others documented below. In addition, for more information on subtler topics like internationalization, time zones, alternative calendars, validity, and so on, see the external documentation.\n */\nexport default class DateTime {\n /**\n * @access private\n */\n constructor(config) {\n const zone = config.zone || Settings.defaultZone;\n\n let invalid =\n config.invalid ||\n (Number.isNaN(config.ts) ? new Invalid(\"invalid input\") : null) ||\n (!zone.isValid ? unsupportedZone(zone) : null);\n /**\n * @access private\n */\n this.ts = isUndefined(config.ts) ? Settings.now() : config.ts;\n\n let c = null,\n o = null;\n if (!invalid) {\n const unchanged = config.old && config.old.ts === this.ts && config.old.zone.equals(zone);\n\n if (unchanged) {\n [c, o] = [config.old.c, config.old.o];\n } else {\n const ot = zone.offset(this.ts);\n c = tsToObj(this.ts, ot);\n invalid = Number.isNaN(c.year) ? new Invalid(\"invalid input\") : null;\n c = invalid ? null : c;\n o = invalid ? null : ot;\n }\n }\n\n /**\n * @access private\n */\n this._zone = zone;\n /**\n * @access private\n */\n this.loc = config.loc || Locale.create();\n /**\n * @access private\n */\n this.invalid = invalid;\n /**\n * @access private\n */\n this.weekData = null;\n /**\n * @access private\n */\n this.c = c;\n /**\n * @access private\n */\n this.o = o;\n /**\n * @access private\n */\n this.isLuxonDateTime = true;\n }\n\n // CONSTRUCT\n\n /**\n * Create a DateTime for the current instant, in the system's time zone.\n *\n * Use Settings to override these default values if needed.\n * @example DateTime.now().toISO() //~> now in the ISO format\n * @return {DateTime}\n */\n static now() {\n return new DateTime({});\n }\n\n /**\n * Create a local DateTime\n * @param {number} [year] - The calendar year. If omitted (as in, call `local()` with no arguments), the current time will be used\n * @param {number} [month=1] - The month, 1-indexed\n * @param {number} [day=1] - The day of the month, 1-indexed\n * @param {number} [hour=0] - The hour of the day, in 24-hour time\n * @param {number} [minute=0] - The minute of the hour, meaning a number between 0 and 59\n * @param {number} [second=0] - The second of the minute, meaning a number between 0 and 59\n * @param {number} [millisecond=0] - The millisecond of the second, meaning a number between 0 and 999\n * @example DateTime.local() //~> now\n * @example DateTime.local(2017) //~> 2017-01-01T00:00:00\n * @example DateTime.local(2017, 3) //~> 2017-03-01T00:00:00\n * @example DateTime.local(2017, 3, 12) //~> 2017-03-12T00:00:00\n * @example DateTime.local(2017, 3, 12, 5) //~> 2017-03-12T05:00:00\n * @example DateTime.local(2017, 3, 12, 5, 45) //~> 2017-03-12T05:45:00\n * @example DateTime.local(2017, 3, 12, 5, 45, 10) //~> 2017-03-12T05:45:10\n * @example DateTime.local(2017, 3, 12, 5, 45, 10, 765) //~> 2017-03-12T05:45:10.765\n * @return {DateTime}\n */\n static local(year, month, day, hour, minute, second, millisecond) {\n if (isUndefined(year)) {\n return DateTime.now();\n } else {\n return quickDT(\n {\n year,\n month,\n day,\n hour,\n minute,\n second,\n millisecond\n },\n Settings.defaultZone\n );\n }\n }\n\n /**\n * Create a DateTime in UTC\n * @param {number} [year] - The calendar year. If omitted (as in, call `utc()` with no arguments), the current time will be used\n * @param {number} [month=1] - The month, 1-indexed\n * @param {number} [day=1] - The day of the month\n * @param {number} [hour=0] - The hour of the day, in 24-hour time\n * @param {number} [minute=0] - The minute of the hour, meaning a number between 0 and 59\n * @param {number} [second=0] - The second of the minute, meaning a number between 0 and 59\n * @param {number} [millisecond=0] - The millisecond of the second, meaning a number between 0 and 999\n * @example DateTime.utc() //~> now\n * @example DateTime.utc(2017) //~> 2017-01-01T00:00:00Z\n * @example DateTime.utc(2017, 3) //~> 2017-03-01T00:00:00Z\n * @example DateTime.utc(2017, 3, 12) //~> 2017-03-12T00:00:00Z\n * @example DateTime.utc(2017, 3, 12, 5) //~> 2017-03-12T05:00:00Z\n * @example DateTime.utc(2017, 3, 12, 5, 45) //~> 2017-03-12T05:45:00Z\n * @example DateTime.utc(2017, 3, 12, 5, 45, 10) //~> 2017-03-12T05:45:10Z\n * @example DateTime.utc(2017, 3, 12, 5, 45, 10, 765) //~> 2017-03-12T05:45:10.765Z\n * @return {DateTime}\n */\n static utc(year, month, day, hour, minute, second, millisecond) {\n if (isUndefined(year)) {\n return new DateTime({\n ts: Settings.now(),\n zone: FixedOffsetZone.utcInstance\n });\n } else {\n return quickDT(\n {\n year,\n month,\n day,\n hour,\n minute,\n second,\n millisecond\n },\n FixedOffsetZone.utcInstance\n );\n }\n }\n\n /**\n * Create a DateTime from a JavaScript Date object. Uses the default zone.\n * @param {Date} date - a JavaScript Date object\n * @param {Object} options - configuration options for the DateTime\n * @param {string|Zone} [options.zone='local'] - the zone to place the DateTime into\n * @return {DateTime}\n */\n static fromJSDate(date, options = {}) {\n const ts = isDate(date) ? date.valueOf() : NaN;\n if (Number.isNaN(ts)) {\n return DateTime.invalid(\"invalid input\");\n }\n\n const zoneToUse = normalizeZone(options.zone, Settings.defaultZone);\n if (!zoneToUse.isValid) {\n return DateTime.invalid(unsupportedZone(zoneToUse));\n }\n\n return new DateTime({\n ts: ts,\n zone: zoneToUse,\n loc: Locale.fromObject(options)\n });\n }\n\n /**\n * Create a DateTime from a number of milliseconds since the epoch (meaning since 1 January 1970 00:00:00 UTC). Uses the default zone.\n * @param {number} milliseconds - a number of milliseconds since 1970 UTC\n * @param {Object} options - configuration options for the DateTime\n * @param {string|Zone} [options.zone='local'] - the zone to place the DateTime into\n * @param {string} [options.locale] - a locale to set on the resulting DateTime instance\n * @param {string} options.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @param {string} options.numberingSystem - the numbering system to set on the resulting DateTime instance\n * @return {DateTime}\n */\n static fromMillis(milliseconds, options = {}) {\n if (!isNumber(milliseconds)) {\n throw new InvalidArgumentError(\n `fromMillis requires a numerical input, but received a ${typeof milliseconds} with value ${milliseconds}`\n );\n } else if (milliseconds < -MAX_DATE || milliseconds > MAX_DATE) {\n // this isn't perfect because because we can still end up out of range because of additional shifting, but it's a start\n return DateTime.invalid(\"Timestamp out of range\");\n } else {\n return new DateTime({\n ts: milliseconds,\n zone: normalizeZone(options.zone, Settings.defaultZone),\n loc: Locale.fromObject(options)\n });\n }\n }\n\n /**\n * Create a DateTime from a number of seconds since the epoch (meaning since 1 January 1970 00:00:00 UTC). Uses the default zone.\n * @param {number} seconds - a number of seconds since 1970 UTC\n * @param {Object} options - configuration options for the DateTime\n * @param {string|Zone} [options.zone='local'] - the zone to place the DateTime into\n * @param {string} [options.locale] - a locale to set on the resulting DateTime instance\n * @param {string} options.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @param {string} options.numberingSystem - the numbering system to set on the resulting DateTime instance\n * @return {DateTime}\n */\n static fromSeconds(seconds, options = {}) {\n if (!isNumber(seconds)) {\n throw new InvalidArgumentError(\"fromSeconds requires a numerical input\");\n } else {\n return new DateTime({\n ts: seconds * 1000,\n zone: normalizeZone(options.zone, Settings.defaultZone),\n loc: Locale.fromObject(options)\n });\n }\n }\n\n /**\n * Create a DateTime from a JavaScript object with keys like 'year' and 'hour' with reasonable defaults.\n * @param {Object} obj - the object to create the DateTime from\n * @param {number} obj.year - a year, such as 1987\n * @param {number} obj.month - a month, 1-12\n * @param {number} obj.day - a day of the month, 1-31, depending on the month\n * @param {number} obj.ordinal - day of the year, 1-365 or 366\n * @param {number} obj.weekYear - an ISO week year\n * @param {number} obj.weekNumber - an ISO week number, between 1 and 52 or 53, depending on the year\n * @param {number} obj.weekday - an ISO weekday, 1-7, where 1 is Monday and 7 is Sunday\n * @param {number} obj.hour - hour of the day, 0-23\n * @param {number} obj.minute - minute of the hour, 0-59\n * @param {number} obj.second - second of the minute, 0-59\n * @param {number} obj.millisecond - millisecond of the second, 0-999\n * @param {string|Zone} [obj.zone='local'] - interpret the numbers in the context of a particular zone. Can take any value taken as the first argument to setZone()\n * @param {string} [obj.locale='system's locale'] - a locale to set on the resulting DateTime instance\n * @param {string} obj.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @param {string} obj.numberingSystem - the numbering system to set on the resulting DateTime instance\n * @example DateTime.fromObject({ year: 1982, month: 5, day: 25}).toISODate() //=> '1982-05-25'\n * @example DateTime.fromObject({ year: 1982 }).toISODate() //=> '1982-01-01'\n * @example DateTime.fromObject({ hour: 10, minute: 26, second: 6 }) //~> today at 10:26:06\n * @example DateTime.fromObject({ hour: 10, minute: 26, second: 6, zone: 'utc' }),\n * @example DateTime.fromObject({ hour: 10, minute: 26, second: 6, zone: 'local' })\n * @example DateTime.fromObject({ hour: 10, minute: 26, second: 6, zone: 'America/New_York' })\n * @example DateTime.fromObject({ weekYear: 2016, weekNumber: 2, weekday: 3 }).toISODate() //=> '2016-01-13'\n * @return {DateTime}\n */\n static fromObject(obj) {\n const zoneToUse = normalizeZone(obj.zone, Settings.defaultZone);\n if (!zoneToUse.isValid) {\n return DateTime.invalid(unsupportedZone(zoneToUse));\n }\n\n const tsNow = Settings.now(),\n offsetProvis = zoneToUse.offset(tsNow),\n normalized = normalizeObject(obj, normalizeUnit, [\n \"zone\",\n \"locale\",\n \"outputCalendar\",\n \"numberingSystem\"\n ]),\n containsOrdinal = !isUndefined(normalized.ordinal),\n containsGregorYear = !isUndefined(normalized.year),\n containsGregorMD = !isUndefined(normalized.month) || !isUndefined(normalized.day),\n containsGregor = containsGregorYear || containsGregorMD,\n definiteWeekDef = normalized.weekYear || normalized.weekNumber,\n loc = Locale.fromObject(obj);\n\n // cases:\n // just a weekday -> this week's instance of that weekday, no worries\n // (gregorian data or ordinal) + (weekYear or weekNumber) -> error\n // (gregorian month or day) + ordinal -> error\n // otherwise just use weeks or ordinals or gregorian, depending on what's specified\n\n if ((containsGregor || containsOrdinal) && definiteWeekDef) {\n throw new ConflictingSpecificationError(\n \"Can't mix weekYear/weekNumber units with year/month/day or ordinals\"\n );\n }\n\n if (containsGregorMD && containsOrdinal) {\n throw new ConflictingSpecificationError(\"Can't mix ordinal dates with month/day\");\n }\n\n const useWeekData = definiteWeekDef || (normalized.weekday && !containsGregor);\n\n // configure ourselves to deal with gregorian dates or week stuff\n let units,\n defaultValues,\n objNow = tsToObj(tsNow, offsetProvis);\n if (useWeekData) {\n units = orderedWeekUnits;\n defaultValues = defaultWeekUnitValues;\n objNow = gregorianToWeek(objNow);\n } else if (containsOrdinal) {\n units = orderedOrdinalUnits;\n defaultValues = defaultOrdinalUnitValues;\n objNow = gregorianToOrdinal(objNow);\n } else {\n units = orderedUnits;\n defaultValues = defaultUnitValues;\n }\n\n // set default values for missing stuff\n let foundFirst = false;\n for (const u of units) {\n const v = normalized[u];\n if (!isUndefined(v)) {\n foundFirst = true;\n } else if (foundFirst) {\n normalized[u] = defaultValues[u];\n } else {\n normalized[u] = objNow[u];\n }\n }\n\n // make sure the values we have are in range\n const higherOrderInvalid = useWeekData\n ? hasInvalidWeekData(normalized)\n : containsOrdinal\n ? hasInvalidOrdinalData(normalized)\n : hasInvalidGregorianData(normalized),\n invalid = higherOrderInvalid || hasInvalidTimeData(normalized);\n\n if (invalid) {\n return DateTime.invalid(invalid);\n }\n\n // compute the actual time\n const gregorian = useWeekData\n ? weekToGregorian(normalized)\n : containsOrdinal\n ? ordinalToGregorian(normalized)\n : normalized,\n [tsFinal, offsetFinal] = objToTS(gregorian, offsetProvis, zoneToUse),\n inst = new DateTime({\n ts: tsFinal,\n zone: zoneToUse,\n o: offsetFinal,\n loc\n });\n\n // gregorian data + weekday serves only to validate\n if (normalized.weekday && containsGregor && obj.weekday !== inst.weekday) {\n return DateTime.invalid(\n \"mismatched weekday\",\n `you can't specify both a weekday of ${normalized.weekday} and a date of ${inst.toISO()}`\n );\n }\n\n return inst;\n }\n\n /**\n * Create a DateTime from an ISO 8601 string\n * @param {string} text - the ISO string\n * @param {Object} opts - options to affect the creation\n * @param {string|Zone} [opts.zone='local'] - use this zone if no offset is specified in the input string itself. Will also convert the time to this zone\n * @param {boolean} [opts.setZone=false] - override the zone with a fixed-offset zone specified in the string itself, if it specifies one\n * @param {string} [opts.locale='system's locale'] - a locale to set on the resulting DateTime instance\n * @param {string} [opts.outputCalendar] - the output calendar to set on the resulting DateTime instance\n * @param {string} [opts.numberingSystem] - the numbering system to set on the resulting DateTime instance\n * @example DateTime.fromISO('2016-05-25T09:08:34.123')\n * @example DateTime.fromISO('2016-05-25T09:08:34.123+06:00')\n * @example DateTime.fromISO('2016-05-25T09:08:34.123+06:00', {setZone: true})\n * @example DateTime.fromISO('2016-05-25T09:08:34.123', {zone: 'utc'})\n * @example DateTime.fromISO('2016-W05-4')\n * @return {DateTime}\n */\n static fromISO(text, opts = {}) {\n const [vals, parsedZone] = parseISODate(text);\n return parseDataToDateTime(vals, parsedZone, opts, \"ISO 8601\", text);\n }\n\n /**\n * Create a DateTime from an RFC 2822 string\n * @param {string} text - the RFC 2822 string\n * @param {Object} opts - options to affect the creation\n * @param {string|Zone} [opts.zone='local'] - convert the time to this zone. Since the offset is always specified in the string itself, this has no effect on the interpretation of string, merely the zone the resulting DateTime is expressed in.\n * @param {boolean} [opts.setZone=false] - override the zone with a fixed-offset zone specified in the string itself, if it specifies one\n * @param {string} [opts.locale='system's locale'] - a locale to set on the resulting DateTime instance\n * @param {string} opts.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @param {string} opts.numberingSystem - the numbering system to set on the resulting DateTime instance\n * @example DateTime.fromRFC2822('25 Nov 2016 13:23:12 GMT')\n * @example DateTime.fromRFC2822('Fri, 25 Nov 2016 13:23:12 +0600')\n * @example DateTime.fromRFC2822('25 Nov 2016 13:23 Z')\n * @return {DateTime}\n */\n static fromRFC2822(text, opts = {}) {\n const [vals, parsedZone] = parseRFC2822Date(text);\n return parseDataToDateTime(vals, parsedZone, opts, \"RFC 2822\", text);\n }\n\n /**\n * Create a DateTime from an HTTP header date\n * @see https://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.3.1\n * @param {string} text - the HTTP header date\n * @param {Object} opts - options to affect the creation\n * @param {string|Zone} [opts.zone='local'] - convert the time to this zone. Since HTTP dates are always in UTC, this has no effect on the interpretation of string, merely the zone the resulting DateTime is expressed in.\n * @param {boolean} [opts.setZone=false] - override the zone with the fixed-offset zone specified in the string. For HTTP dates, this is always UTC, so this option is equivalent to setting the `zone` option to 'utc', but this option is included for consistency with similar methods.\n * @param {string} [opts.locale='system's locale'] - a locale to set on the resulting DateTime instance\n * @param {string} opts.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @param {string} opts.numberingSystem - the numbering system to set on the resulting DateTime instance\n * @example DateTime.fromHTTP('Sun, 06 Nov 1994 08:49:37 GMT')\n * @example DateTime.fromHTTP('Sunday, 06-Nov-94 08:49:37 GMT')\n * @example DateTime.fromHTTP('Sun Nov 6 08:49:37 1994')\n * @return {DateTime}\n */\n static fromHTTP(text, opts = {}) {\n const [vals, parsedZone] = parseHTTPDate(text);\n return parseDataToDateTime(vals, parsedZone, opts, \"HTTP\", opts);\n }\n\n /**\n * Create a DateTime from an input string and format string.\n * Defaults to en-US if no locale has been specified, regardless of the system's locale.\n * @see https://moment.github.io/luxon/docs/manual/parsing.html#table-of-tokens\n * @param {string} text - the string to parse\n * @param {string} fmt - the format the string is expected to be in (see the link below for the formats)\n * @param {Object} opts - options to affect the creation\n * @param {string|Zone} [opts.zone='local'] - use this zone if no offset is specified in the input string itself. Will also convert the DateTime to this zone\n * @param {boolean} [opts.setZone=false] - override the zone with a zone specified in the string itself, if it specifies one\n * @param {string} [opts.locale='en-US'] - a locale string to use when parsing. Will also set the DateTime to this locale\n * @param {string} opts.numberingSystem - the numbering system to use when parsing. Will also set the resulting DateTime to this numbering system\n * @param {string} opts.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @return {DateTime}\n */\n static fromFormat(text, fmt, opts = {}) {\n if (isUndefined(text) || isUndefined(fmt)) {\n throw new InvalidArgumentError(\"fromFormat requires an input string and a format\");\n }\n\n const { locale = null, numberingSystem = null } = opts,\n localeToUse = Locale.fromOpts({\n locale,\n numberingSystem,\n defaultToEN: true\n }),\n [vals, parsedZone, invalid] = parseFromTokens(localeToUse, text, fmt);\n if (invalid) {\n return DateTime.invalid(invalid);\n } else {\n return parseDataToDateTime(vals, parsedZone, opts, `format ${fmt}`, text);\n }\n }\n\n /**\n * @deprecated use fromFormat instead\n */\n static fromString(text, fmt, opts = {}) {\n return DateTime.fromFormat(text, fmt, opts);\n }\n\n /**\n * Create a DateTime from a SQL date, time, or datetime\n * Defaults to en-US if no locale has been specified, regardless of the system's locale\n * @param {string} text - the string to parse\n * @param {Object} opts - options to affect the creation\n * @param {string|Zone} [opts.zone='local'] - use this zone if no offset is specified in the input string itself. Will also convert the DateTime to this zone\n * @param {boolean} [opts.setZone=false] - override the zone with a zone specified in the string itself, if it specifies one\n * @param {string} [opts.locale='en-US'] - a locale string to use when parsing. Will also set the DateTime to this locale\n * @param {string} opts.numberingSystem - the numbering system to use when parsing. Will also set the resulting DateTime to this numbering system\n * @param {string} opts.outputCalendar - the output calendar to set on the resulting DateTime instance\n * @example DateTime.fromSQL('2017-05-15')\n * @example DateTime.fromSQL('2017-05-15 09:12:34')\n * @example DateTime.fromSQL('2017-05-15 09:12:34.342')\n * @example DateTime.fromSQL('2017-05-15 09:12:34.342+06:00')\n * @example DateTime.fromSQL('2017-05-15 09:12:34.342 America/Los_Angeles')\n * @example DateTime.fromSQL('2017-05-15 09:12:34.342 America/Los_Angeles', { setZone: true })\n * @example DateTime.fromSQL('2017-05-15 09:12:34.342', { zone: 'America/Los_Angeles' })\n * @example DateTime.fromSQL('09:12:34.342')\n * @return {DateTime}\n */\n static fromSQL(text, opts = {}) {\n const [vals, parsedZone] = parseSQL(text);\n return parseDataToDateTime(vals, parsedZone, opts, \"SQL\", text);\n }\n\n /**\n * Create an invalid DateTime.\n * @param {string} reason - simple string of why this DateTime is invalid. Should not contain parameters or anything else data-dependent\n * @param {string} [explanation=null] - longer explanation, may include parameters and other useful debugging information\n * @return {DateTime}\n */\n static invalid(reason, explanation = null) {\n if (!reason) {\n throw new InvalidArgumentError(\"need to specify a reason the DateTime is invalid\");\n }\n\n const invalid = reason instanceof Invalid ? reason : new Invalid(reason, explanation);\n\n if (Settings.throwOnInvalid) {\n throw new InvalidDateTimeError(invalid);\n } else {\n return new DateTime({ invalid });\n }\n }\n\n /**\n * Check if an object is a DateTime. Works across context boundaries\n * @param {object} o\n * @return {boolean}\n */\n static isDateTime(o) {\n return (o && o.isLuxonDateTime) || false;\n }\n\n // INFO\n\n /**\n * Get the value of unit.\n * @param {string} unit - a unit such as 'minute' or 'day'\n * @example DateTime.local(2017, 7, 4).get('month'); //=> 7\n * @example DateTime.local(2017, 7, 4).get('day'); //=> 4\n * @return {number}\n */\n get(unit) {\n return this[unit];\n }\n\n /**\n * Returns whether the DateTime is valid. Invalid DateTimes occur when:\n * * The DateTime was created from invalid calendar information, such as the 13th month or February 30\n * * The DateTime was created by an operation on another invalid date\n * @type {boolean}\n */\n get isValid() {\n return this.invalid === null;\n }\n\n /**\n * Returns an error code if this DateTime is invalid, or null if the DateTime is valid\n * @type {string}\n */\n get invalidReason() {\n return this.invalid ? this.invalid.reason : null;\n }\n\n /**\n * Returns an explanation of why this DateTime became invalid, or null if the DateTime is valid\n * @type {string}\n */\n get invalidExplanation() {\n return this.invalid ? this.invalid.explanation : null;\n }\n\n /**\n * Get the locale of a DateTime, such 'en-GB'. The locale is used when formatting the DateTime\n *\n * @type {string}\n */\n get locale() {\n return this.isValid ? this.loc.locale : null;\n }\n\n /**\n * Get the numbering system of a DateTime, such 'beng'. The numbering system is used when formatting the DateTime\n *\n * @type {string}\n */\n get numberingSystem() {\n return this.isValid ? this.loc.numberingSystem : null;\n }\n\n /**\n * Get the output calendar of a DateTime, such 'islamic'. The output calendar is used when formatting the DateTime\n *\n * @type {string}\n */\n get outputCalendar() {\n return this.isValid ? this.loc.outputCalendar : null;\n }\n\n /**\n * Get the time zone associated with this DateTime.\n * @type {Zone}\n */\n get zone() {\n return this._zone;\n }\n\n /**\n * Get the name of the time zone.\n * @type {string}\n */\n get zoneName() {\n return this.isValid ? this.zone.name : null;\n }\n\n /**\n * Get the year\n * @example DateTime.local(2017, 5, 25).year //=> 2017\n * @type {number}\n */\n get year() {\n return this.isValid ? this.c.year : NaN;\n }\n\n /**\n * Get the quarter\n * @example DateTime.local(2017, 5, 25).quarter //=> 2\n * @type {number}\n */\n get quarter() {\n return this.isValid ? Math.ceil(this.c.month / 3) : NaN;\n }\n\n /**\n * Get the month (1-12).\n * @example DateTime.local(2017, 5, 25).month //=> 5\n * @type {number}\n */\n get month() {\n return this.isValid ? this.c.month : NaN;\n }\n\n /**\n * Get the day of the month (1-30ish).\n * @example DateTime.local(2017, 5, 25).day //=> 25\n * @type {number}\n */\n get day() {\n return this.isValid ? this.c.day : NaN;\n }\n\n /**\n * Get the hour of the day (0-23).\n * @example DateTime.local(2017, 5, 25, 9).hour //=> 9\n * @type {number}\n */\n get hour() {\n return this.isValid ? this.c.hour : NaN;\n }\n\n /**\n * Get the minute of the hour (0-59).\n * @example DateTime.local(2017, 5, 25, 9, 30).minute //=> 30\n * @type {number}\n */\n get minute() {\n return this.isValid ? this.c.minute : NaN;\n }\n\n /**\n * Get the second of the minute (0-59).\n * @example DateTime.local(2017, 5, 25, 9, 30, 52).second //=> 52\n * @type {number}\n */\n get second() {\n return this.isValid ? this.c.second : NaN;\n }\n\n /**\n * Get the millisecond of the second (0-999).\n * @example DateTime.local(2017, 5, 25, 9, 30, 52, 654).millisecond //=> 654\n * @type {number}\n */\n get millisecond() {\n return this.isValid ? this.c.millisecond : NaN;\n }\n\n /**\n * Get the week year\n * @see https://en.wikipedia.org/wiki/ISO_week_date\n * @example DateTime.local(2014, 12, 31).weekYear //=> 2015\n * @type {number}\n */\n get weekYear() {\n return this.isValid ? possiblyCachedWeekData(this).weekYear : NaN;\n }\n\n /**\n * Get the week number of the week year (1-52ish).\n * @see https://en.wikipedia.org/wiki/ISO_week_date\n * @example DateTime.local(2017, 5, 25).weekNumber //=> 21\n * @type {number}\n */\n get weekNumber() {\n return this.isValid ? possiblyCachedWeekData(this).weekNumber : NaN;\n }\n\n /**\n * Get the day of the week.\n * 1 is Monday and 7 is Sunday\n * @see https://en.wikipedia.org/wiki/ISO_week_date\n * @example DateTime.local(2014, 11, 31).weekday //=> 4\n * @type {number}\n */\n get weekday() {\n return this.isValid ? possiblyCachedWeekData(this).weekday : NaN;\n }\n\n /**\n * Get the ordinal (meaning the day of the year)\n * @example DateTime.local(2017, 5, 25).ordinal //=> 145\n * @type {number|DateTime}\n */\n get ordinal() {\n return this.isValid ? gregorianToOrdinal(this.c).ordinal : NaN;\n }\n\n /**\n * Get the human readable short month name, such as 'Oct'.\n * Defaults to the system's locale if no locale has been specified\n * @example DateTime.local(2017, 10, 30).monthShort //=> Oct\n * @type {string}\n */\n get monthShort() {\n return this.isValid ? Info.months(\"short\", { locObj: this.loc })[this.month - 1] : null;\n }\n\n /**\n * Get the human readable long month name, such as 'October'.\n * Defaults to the system's locale if no locale has been specified\n * @example DateTime.local(2017, 10, 30).monthLong //=> October\n * @type {string}\n */\n get monthLong() {\n return this.isValid ? Info.months(\"long\", { locObj: this.loc })[this.month - 1] : null;\n }\n\n /**\n * Get the human readable short weekday, such as 'Mon'.\n * Defaults to the system's locale if no locale has been specified\n * @example DateTime.local(2017, 10, 30).weekdayShort //=> Mon\n * @type {string}\n */\n get weekdayShort() {\n return this.isValid ? Info.weekdays(\"short\", { locObj: this.loc })[this.weekday - 1] : null;\n }\n\n /**\n * Get the human readable long weekday, such as 'Monday'.\n * Defaults to the system's locale if no locale has been specified\n * @example DateTime.local(2017, 10, 30).weekdayLong //=> Monday\n * @type {string}\n */\n get weekdayLong() {\n return this.isValid ? Info.weekdays(\"long\", { locObj: this.loc })[this.weekday - 1] : null;\n }\n\n /**\n * Get the UTC offset of this DateTime in minutes\n * @example DateTime.now().offset //=> -240\n * @example DateTime.utc().offset //=> 0\n * @type {number}\n */\n get offset() {\n return this.isValid ? +this.o : NaN;\n }\n\n /**\n * Get the short human name for the zone's current offset, for example \"EST\" or \"EDT\".\n * Defaults to the system's locale if no locale has been specified\n * @type {string}\n */\n get offsetNameShort() {\n if (this.isValid) {\n return this.zone.offsetName(this.ts, {\n format: \"short\",\n locale: this.locale\n });\n } else {\n return null;\n }\n }\n\n /**\n * Get the long human name for the zone's current offset, for example \"Eastern Standard Time\" or \"Eastern Daylight Time\".\n * Defaults to the system's locale if no locale has been specified\n * @type {string}\n */\n get offsetNameLong() {\n if (this.isValid) {\n return this.zone.offsetName(this.ts, {\n format: \"long\",\n locale: this.locale\n });\n } else {\n return null;\n }\n }\n\n /**\n * Get whether this zone's offset ever changes, as in a DST.\n * @type {boolean}\n */\n get isOffsetFixed() {\n return this.isValid ? this.zone.universal : null;\n }\n\n /**\n * Get whether the DateTime is in a DST.\n * @type {boolean}\n */\n get isInDST() {\n if (this.isOffsetFixed) {\n return false;\n } else {\n return (\n this.offset > this.set({ month: 1 }).offset || this.offset > this.set({ month: 5 }).offset\n );\n }\n }\n\n /**\n * Returns true if this DateTime is in a leap year, false otherwise\n * @example DateTime.local(2016).isInLeapYear //=> true\n * @example DateTime.local(2013).isInLeapYear //=> false\n * @type {boolean}\n */\n get isInLeapYear() {\n return isLeapYear(this.year);\n }\n\n /**\n * Returns the number of days in this DateTime's month\n * @example DateTime.local(2016, 2).daysInMonth //=> 29\n * @example DateTime.local(2016, 3).daysInMonth //=> 31\n * @type {number}\n */\n get daysInMonth() {\n return daysInMonth(this.year, this.month);\n }\n\n /**\n * Returns the number of days in this DateTime's year\n * @example DateTime.local(2016).daysInYear //=> 366\n * @example DateTime.local(2013).daysInYear //=> 365\n * @type {number}\n */\n get daysInYear() {\n return this.isValid ? daysInYear(this.year) : NaN;\n }\n\n /**\n * Returns the number of weeks in this DateTime's year\n * @see https://en.wikipedia.org/wiki/ISO_week_date\n * @example DateTime.local(2004).weeksInWeekYear //=> 53\n * @example DateTime.local(2013).weeksInWeekYear //=> 52\n * @type {number}\n */\n get weeksInWeekYear() {\n return this.isValid ? weeksInWeekYear(this.weekYear) : NaN;\n }\n\n /**\n * Returns the resolved Intl options for this DateTime.\n * This is useful in understanding the behavior of formatting methods\n * @param {Object} opts - the same options as toLocaleString\n * @return {Object}\n */\n resolvedLocaleOpts(opts = {}) {\n const { locale, numberingSystem, calendar } = Formatter.create(\n this.loc.clone(opts),\n opts\n ).resolvedOptions(this);\n return { locale, numberingSystem, outputCalendar: calendar };\n }\n\n // TRANSFORM\n\n /**\n * \"Set\" the DateTime's zone to UTC. Returns a newly-constructed DateTime.\n *\n * Equivalent to {@link setZone}('utc')\n * @param {number} [offset=0] - optionally, an offset from UTC in minutes\n * @param {Object} [opts={}] - options to pass to `setZone()`\n * @return {DateTime}\n */\n toUTC(offset = 0, opts = {}) {\n return this.setZone(FixedOffsetZone.instance(offset), opts);\n }\n\n /**\n * \"Set\" the DateTime's zone to the host's local zone. Returns a newly-constructed DateTime.\n *\n * Equivalent to `setZone('local')`\n * @return {DateTime}\n */\n toLocal() {\n return this.setZone(Settings.defaultZone);\n }\n\n /**\n * \"Set\" the DateTime's zone to specified zone. Returns a newly-constructed DateTime.\n *\n * By default, the setter keeps the underlying time the same (as in, the same timestamp), but the new instance will report different local times and consider DSTs when making computations, as with {@link plus}. You may wish to use {@link toLocal} and {@link toUTC} which provide simple convenience wrappers for commonly used zones.\n * @param {string|Zone} [zone='local'] - a zone identifier. As a string, that can be any IANA zone supported by the host environment, or a fixed-offset name of the form 'UTC+3', or the strings 'local' or 'utc'. You may also supply an instance of a {@link Zone} class.\n * @param {Object} opts - options\n * @param {boolean} [opts.keepLocalTime=false] - If true, adjust the underlying time so that the local time stays the same, but in the target zone. You should rarely need this.\n * @return {DateTime}\n */\n setZone(zone, { keepLocalTime = false, keepCalendarTime = false } = {}) {\n zone = normalizeZone(zone, Settings.defaultZone);\n if (zone.equals(this.zone)) {\n return this;\n } else if (!zone.isValid) {\n return DateTime.invalid(unsupportedZone(zone));\n } else {\n let newTS = this.ts;\n if (keepLocalTime || keepCalendarTime) {\n const offsetGuess = zone.offset(this.ts);\n const asObj = this.toObject();\n [newTS] = objToTS(asObj, offsetGuess, zone);\n }\n return clone(this, { ts: newTS, zone });\n }\n }\n\n /**\n * \"Set\" the locale, numberingSystem, or outputCalendar. Returns a newly-constructed DateTime.\n * @param {Object} properties - the properties to set\n * @example DateTime.local(2017, 5, 25).reconfigure({ locale: 'en-GB' })\n * @return {DateTime}\n */\n reconfigure({ locale, numberingSystem, outputCalendar } = {}) {\n const loc = this.loc.clone({ locale, numberingSystem, outputCalendar });\n return clone(this, { loc });\n }\n\n /**\n * \"Set\" the locale. Returns a newly-constructed DateTime.\n * Just a convenient alias for reconfigure({ locale })\n * @example DateTime.local(2017, 5, 25).setLocale('en-GB')\n * @return {DateTime}\n */\n setLocale(locale) {\n return this.reconfigure({ locale });\n }\n\n /**\n * \"Set\" the values of specified units. Returns a newly-constructed DateTime.\n * You can only set units with this method; for \"setting\" metadata, see {@link reconfigure} and {@link setZone}.\n * @param {Object} values - a mapping of units to numbers\n * @example dt.set({ year: 2017 })\n * @example dt.set({ hour: 8, minute: 30 })\n * @example dt.set({ weekday: 5 })\n * @example dt.set({ year: 2005, ordinal: 234 })\n * @return {DateTime}\n */\n set(values) {\n if (!this.isValid) return this;\n\n const normalized = normalizeObject(values, normalizeUnit, []),\n settingWeekStuff =\n !isUndefined(normalized.weekYear) ||\n !isUndefined(normalized.weekNumber) ||\n !isUndefined(normalized.weekday),\n containsOrdinal = !isUndefined(normalized.ordinal),\n containsGregorYear = !isUndefined(normalized.year),\n containsGregorMD = !isUndefined(normalized.month) || !isUndefined(normalized.day),\n containsGregor = containsGregorYear || containsGregorMD,\n definiteWeekDef = normalized.weekYear || normalized.weekNumber;\n\n if ((containsGregor || containsOrdinal) && definiteWeekDef) {\n throw new ConflictingSpecificationError(\n \"Can't mix weekYear/weekNumber units with year/month/day or ordinals\"\n );\n }\n\n if (containsGregorMD && containsOrdinal) {\n throw new ConflictingSpecificationError(\"Can't mix ordinal dates with month/day\");\n }\n\n let mixed;\n if (settingWeekStuff) {\n mixed = weekToGregorian(Object.assign(gregorianToWeek(this.c), normalized));\n } else if (!isUndefined(normalized.ordinal)) {\n mixed = ordinalToGregorian(Object.assign(gregorianToOrdinal(this.c), normalized));\n } else {\n mixed = Object.assign(this.toObject(), normalized);\n\n // if we didn't set the day but we ended up on an overflow date,\n // use the last day of the right month\n if (isUndefined(normalized.day)) {\n mixed.day = Math.min(daysInMonth(mixed.year, mixed.month), mixed.day);\n }\n }\n\n const [ts, o] = objToTS(mixed, this.o, this.zone);\n return clone(this, { ts, o });\n }\n\n /**\n * Add a period of time to this DateTime and return the resulting DateTime\n *\n * Adding hours, minutes, seconds, or milliseconds increases the timestamp by the right number of milliseconds. Adding days, months, or years shifts the calendar, accounting for DSTs and leap years along the way. Thus, `dt.plus({ hours: 24 })` may result in a different time than `dt.plus({ days: 1 })` if there's a DST shift in between.\n * @param {Duration|Object|number} duration - The amount to add. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()\n * @example DateTime.now().plus(123) //~> in 123 milliseconds\n * @example DateTime.now().plus({ minutes: 15 }) //~> in 15 minutes\n * @example DateTime.now().plus({ days: 1 }) //~> this time tomorrow\n * @example DateTime.now().plus({ days: -1 }) //~> this time yesterday\n * @example DateTime.now().plus({ hours: 3, minutes: 13 }) //~> in 3 hr, 13 min\n * @example DateTime.now().plus(Duration.fromObject({ hours: 3, minutes: 13 })) //~> in 3 hr, 13 min\n * @return {DateTime}\n */\n plus(duration) {\n if (!this.isValid) return this;\n const dur = friendlyDuration(duration);\n return clone(this, adjustTime(this, dur));\n }\n\n /**\n * Subtract a period of time to this DateTime and return the resulting DateTime\n * See {@link plus}\n * @param {Duration|Object|number} duration - The amount to subtract. Either a Luxon Duration, a number of milliseconds, the object argument to Duration.fromObject()\n @return {DateTime}\n */\n minus(duration) {\n if (!this.isValid) return this;\n const dur = friendlyDuration(duration).negate();\n return clone(this, adjustTime(this, dur));\n }\n\n /**\n * \"Set\" this DateTime to the beginning of a unit of time.\n * @param {string} unit - The unit to go to the beginning of. Can be 'year', 'quarter', 'month', 'week', 'day', 'hour', 'minute', 'second', or 'millisecond'.\n * @example DateTime.local(2014, 3, 3).startOf('month').toISODate(); //=> '2014-03-01'\n * @example DateTime.local(2014, 3, 3).startOf('year').toISODate(); //=> '2014-01-01'\n * @example DateTime.local(2014, 3, 3).startOf('week').toISODate(); //=> '2014-03-03', weeks always start on Mondays\n * @example DateTime.local(2014, 3, 3, 5, 30).startOf('day').toISOTime(); //=> '00:00.000-05:00'\n * @example DateTime.local(2014, 3, 3, 5, 30).startOf('hour').toISOTime(); //=> '05:00:00.000-05:00'\n * @return {DateTime}\n */\n startOf(unit) {\n if (!this.isValid) return this;\n const o = {},\n normalizedUnit = Duration.normalizeUnit(unit);\n switch (normalizedUnit) {\n case \"years\":\n o.month = 1;\n // falls through\n case \"quarters\":\n case \"months\":\n o.day = 1;\n // falls through\n case \"weeks\":\n case \"days\":\n o.hour = 0;\n // falls through\n case \"hours\":\n o.minute = 0;\n // falls through\n case \"minutes\":\n o.second = 0;\n // falls through\n case \"seconds\":\n o.millisecond = 0;\n break;\n case \"milliseconds\":\n break;\n // no default, invalid units throw in normalizeUnit()\n }\n\n if (normalizedUnit === \"weeks\") {\n o.weekday = 1;\n }\n\n if (normalizedUnit === \"quarters\") {\n const q = Math.ceil(this.month / 3);\n o.month = (q - 1) * 3 + 1;\n }\n\n return this.set(o);\n }\n\n /**\n * \"Set\" this DateTime to the end (meaning the last millisecond) of a unit of time\n * @param {string} unit - The unit to go to the end of. Can be 'year', 'quarter', 'month', 'week', 'day', 'hour', 'minute', 'second', or 'millisecond'.\n * @example DateTime.local(2014, 3, 3).endOf('month').toISO(); //=> '2014-03-31T23:59:59.999-05:00'\n * @example DateTime.local(2014, 3, 3).endOf('year').toISO(); //=> '2014-12-31T23:59:59.999-05:00'\n * @example DateTime.local(2014, 3, 3).endOf('week').toISO(); // => '2014-03-09T23:59:59.999-05:00', weeks start on Mondays\n * @example DateTime.local(2014, 3, 3, 5, 30).endOf('day').toISO(); //=> '2014-03-03T23:59:59.999-05:00'\n * @example DateTime.local(2014, 3, 3, 5, 30).endOf('hour').toISO(); //=> '2014-03-03T05:59:59.999-05:00'\n * @return {DateTime}\n */\n endOf(unit) {\n return this.isValid\n ? this.plus({ [unit]: 1 })\n .startOf(unit)\n .minus(1)\n : this;\n }\n\n // OUTPUT\n\n /**\n * Returns a string representation of this DateTime formatted according to the specified format string.\n * **You may not want this.** See {@link toLocaleString} for a more flexible formatting tool. For a table of tokens and their interpretations, see [here](https://moment.github.io/luxon/docs/manual/formatting.html#table-of-tokens).\n * Defaults to en-US if no locale has been specified, regardless of the system's locale.\n * @see https://moment.github.io/luxon/docs/manual/formatting.html#table-of-tokens\n * @param {string} fmt - the format string\n * @param {Object} opts - opts to override the configuration options\n * @example DateTime.now().toFormat('yyyy LLL dd') //=> '2017 Apr 22'\n * @example DateTime.now().setLocale('fr').toFormat('yyyy LLL dd') //=> '2017 avr. 22'\n * @example DateTime.now().toFormat('yyyy LLL dd', { locale: \"fr\" }) //=> '2017 avr. 22'\n * @example DateTime.now().toFormat(\"HH 'hours and' mm 'minutes'\") //=> '20 hours and 55 minutes'\n * @return {string}\n */\n toFormat(fmt, opts = {}) {\n return this.isValid\n ? Formatter.create(this.loc.redefaultToEN(opts)).formatDateTimeFromString(this, fmt)\n : INVALID;\n }\n\n /**\n * Returns a localized string representing this date. Accepts the same options as the Intl.DateTimeFormat constructor and any presets defined by Luxon, such as `DateTime.DATE_FULL` or `DateTime.TIME_SIMPLE`.\n * The exact behavior of this method is browser-specific, but in general it will return an appropriate representation\n * of the DateTime in the assigned locale.\n * Defaults to the system's locale if no locale has been specified\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat\n * @param opts {Object} - Intl.DateTimeFormat constructor options and configuration options\n * @example DateTime.now().toLocaleString(); //=> 4/20/2017\n * @example DateTime.now().setLocale('en-gb').toLocaleString(); //=> '20/04/2017'\n * @example DateTime.now().toLocaleString({ locale: 'en-gb' }); //=> '20/04/2017'\n * @example DateTime.now().toLocaleString(DateTime.DATE_FULL); //=> 'April 20, 2017'\n * @example DateTime.now().toLocaleString(DateTime.TIME_SIMPLE); //=> '11:32 AM'\n * @example DateTime.now().toLocaleString(DateTime.DATETIME_SHORT); //=> '4/20/2017, 11:32 AM'\n * @example DateTime.now().toLocaleString({ weekday: 'long', month: 'long', day: '2-digit' }); //=> 'Thursday, April 20'\n * @example DateTime.now().toLocaleString({ weekday: 'short', month: 'short', day: '2-digit', hour: '2-digit', minute: '2-digit' }); //=> 'Thu, Apr 20, 11:27 AM'\n * @example DateTime.now().toLocaleString({ hour: '2-digit', minute: '2-digit', hour12: false }); //=> '11:32'\n * @return {string}\n */\n toLocaleString(opts = Formats.DATE_SHORT) {\n return this.isValid\n ? Formatter.create(this.loc.clone(opts), opts).formatDateTime(this)\n : INVALID;\n }\n\n /**\n * Returns an array of format \"parts\", meaning individual tokens along with metadata. This is allows callers to post-process individual sections of the formatted output.\n * Defaults to the system's locale if no locale has been specified\n * @see https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/DateTimeFormat/formatToParts\n * @param opts {Object} - Intl.DateTimeFormat constructor options, same as `toLocaleString`.\n * @example DateTime.now().toLocaleParts(); //=> [\n * //=> { type: 'day', value: '25' },\n * //=> { type: 'literal', value: '/' },\n * //=> { type: 'month', value: '05' },\n * //=> { type: 'literal', value: '/' },\n * //=> { type: 'year', value: '1982' }\n * //=> ]\n */\n toLocaleParts(opts = {}) {\n return this.isValid\n ? Formatter.create(this.loc.clone(opts), opts).formatDateTimeParts(this)\n : [];\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this DateTime\n * @param {Object} opts - options\n * @param {boolean} [opts.suppressMilliseconds=false] - exclude milliseconds from the format if they're 0\n * @param {boolean} [opts.suppressSeconds=false] - exclude seconds from the format if they're 0\n * @param {boolean} [opts.includeOffset=true] - include the offset, such as 'Z' or '-04:00'\n * @param {string} [opts.format='extended'] - choose between the basic and extended format\n * @example DateTime.utc(1982, 5, 25).toISO() //=> '1982-05-25T00:00:00.000Z'\n * @example DateTime.now().toISO() //=> '2017-04-22T20:47:05.335-04:00'\n * @example DateTime.now().toISO({ includeOffset: false }) //=> '2017-04-22T20:47:05.335'\n * @example DateTime.now().toISO({ format: 'basic' }) //=> '20170422T204705.335-0400'\n * @return {string}\n */\n toISO(opts = {}) {\n if (!this.isValid) {\n return null;\n }\n\n return `${this.toISODate(opts)}T${this.toISOTime(opts)}`;\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this DateTime's date component\n * @param {Object} opts - options\n * @param {string} [opts.format='extended'] - choose between the basic and extended format\n * @example DateTime.utc(1982, 5, 25).toISODate() //=> '1982-05-25'\n * @example DateTime.utc(1982, 5, 25).toISODate({ format: 'basic' }) //=> '19820525'\n * @return {string}\n */\n toISODate({ format = \"extended\" } = {}) {\n let fmt = format === \"basic\" ? \"yyyyMMdd\" : \"yyyy-MM-dd\";\n if (this.year > 9999) {\n fmt = \"+\" + fmt;\n }\n\n return toTechFormat(this, fmt);\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this DateTime's week date\n * @example DateTime.utc(1982, 5, 25).toISOWeekDate() //=> '1982-W21-2'\n * @return {string}\n */\n toISOWeekDate() {\n return toTechFormat(this, \"kkkk-'W'WW-c\");\n }\n\n /**\n * Returns an ISO 8601-compliant string representation of this DateTime's time component\n * @param {Object} opts - options\n * @param {boolean} [opts.suppressMilliseconds=false] - exclude milliseconds from the format if they're 0\n * @param {boolean} [opts.suppressSeconds=false] - exclude seconds from the format if they're 0\n * @param {boolean} [opts.includeOffset=true] - include the offset, such as 'Z' or '-04:00'\n * @param {boolean} [opts.includePrefix=false] - include the `T` prefix\n * @param {string} [opts.format='extended'] - choose between the basic and extended format\n * @example DateTime.utc().set({ hour: 7, minute: 34 }).toISOTime() //=> '07:34:19.361Z'\n * @example DateTime.utc().set({ hour: 7, minute: 34, seconds: 0, milliseconds: 0 }).toISOTime({ suppressSeconds: true }) //=> '07:34Z'\n * @example DateTime.utc().set({ hour: 7, minute: 34 }).toISOTime({ format: 'basic' }) //=> '073419.361Z'\n * @example DateTime.utc().set({ hour: 7, minute: 34 }).toISOTime({ includePrefix: true }) //=> 'T07:34:19.361Z'\n * @return {string}\n */\n toISOTime({\n suppressMilliseconds = false,\n suppressSeconds = false,\n includeOffset = true,\n includePrefix = false,\n format = \"extended\"\n } = {}) {\n return toTechTimeFormat(this, {\n suppressSeconds,\n suppressMilliseconds,\n includeOffset,\n includePrefix,\n format\n });\n }\n\n /**\n * Returns an RFC 2822-compatible string representation of this DateTime, always in UTC\n * @example DateTime.utc(2014, 7, 13).toRFC2822() //=> 'Sun, 13 Jul 2014 00:00:00 +0000'\n * @example DateTime.local(2014, 7, 13).toRFC2822() //=> 'Sun, 13 Jul 2014 00:00:00 -0400'\n * @return {string}\n */\n toRFC2822() {\n return toTechFormat(this, \"EEE, dd LLL yyyy HH:mm:ss ZZZ\", false);\n }\n\n /**\n * Returns a string representation of this DateTime appropriate for use in HTTP headers.\n * Specifically, the string conforms to RFC 1123.\n * @see https://www.w3.org/Protocols/rfc2616/rfc2616-sec3.html#sec3.3.1\n * @example DateTime.utc(2014, 7, 13).toHTTP() //=> 'Sun, 13 Jul 2014 00:00:00 GMT'\n * @example DateTime.utc(2014, 7, 13, 19).toHTTP() //=> 'Sun, 13 Jul 2014 19:00:00 GMT'\n * @return {string}\n */\n toHTTP() {\n return toTechFormat(this.toUTC(), \"EEE, dd LLL yyyy HH:mm:ss 'GMT'\");\n }\n\n /**\n * Returns a string representation of this DateTime appropriate for use in SQL Date\n * @example DateTime.utc(2014, 7, 13).toSQLDate() //=> '2014-07-13'\n * @return {string}\n */\n toSQLDate() {\n return toTechFormat(this, \"yyyy-MM-dd\");\n }\n\n /**\n * Returns a string representation of this DateTime appropriate for use in SQL Time\n * @param {Object} opts - options\n * @param {boolean} [opts.includeZone=false] - include the zone, such as 'America/New_York'. Overrides includeOffset.\n * @param {boolean} [opts.includeOffset=true] - include the offset, such as 'Z' or '-04:00'\n * @example DateTime.utc().toSQL() //=> '05:15:16.345'\n * @example DateTime.now().toSQL() //=> '05:15:16.345 -04:00'\n * @example DateTime.now().toSQL({ includeOffset: false }) //=> '05:15:16.345'\n * @example DateTime.now().toSQL({ includeZone: false }) //=> '05:15:16.345 America/New_York'\n * @return {string}\n */\n toSQLTime({ includeOffset = true, includeZone = false } = {}) {\n return toTechTimeFormat(this, {\n includeOffset,\n includeZone,\n spaceZone: true\n });\n }\n\n /**\n * Returns a string representation of this DateTime appropriate for use in SQL DateTime\n * @param {Object} opts - options\n * @param {boolean} [opts.includeZone=false] - include the zone, such as 'America/New_York'. Overrides includeOffset.\n * @param {boolean} [opts.includeOffset=true] - include the offset, such as 'Z' or '-04:00'\n * @example DateTime.utc(2014, 7, 13).toSQL() //=> '2014-07-13 00:00:00.000 Z'\n * @example DateTime.local(2014, 7, 13).toSQL() //=> '2014-07-13 00:00:00.000 -04:00'\n * @example DateTime.local(2014, 7, 13).toSQL({ includeOffset: false }) //=> '2014-07-13 00:00:00.000'\n * @example DateTime.local(2014, 7, 13).toSQL({ includeZone: true }) //=> '2014-07-13 00:00:00.000 America/New_York'\n * @return {string}\n */\n toSQL(opts = {}) {\n if (!this.isValid) {\n return null;\n }\n\n return `${this.toSQLDate()} ${this.toSQLTime(opts)}`;\n }\n\n /**\n * Returns a string representation of this DateTime appropriate for debugging\n * @return {string}\n */\n toString() {\n return this.isValid ? this.toISO() : INVALID;\n }\n\n /**\n * Returns the epoch milliseconds of this DateTime. Alias of {@link toMillis}\n * @return {number}\n */\n valueOf() {\n return this.toMillis();\n }\n\n /**\n * Returns the epoch milliseconds of this DateTime.\n * @return {number}\n */\n toMillis() {\n return this.isValid ? this.ts : NaN;\n }\n\n /**\n * Returns the epoch seconds of this DateTime.\n * @return {number}\n */\n toSeconds() {\n return this.isValid ? this.ts / 1000 : NaN;\n }\n\n /**\n * Returns an ISO 8601 representation of this DateTime appropriate for use in JSON.\n * @return {string}\n */\n toJSON() {\n return this.toISO();\n }\n\n /**\n * Returns a BSON serializable equivalent to this DateTime.\n * @return {Date}\n */\n toBSON() {\n return this.toJSDate();\n }\n\n /**\n * Returns a JavaScript object with this DateTime's year, month, day, and so on.\n * @param opts - options for generating the object\n * @param {boolean} [opts.includeConfig=false] - include configuration attributes in the output\n * @example DateTime.now().toObject() //=> { year: 2017, month: 4, day: 22, hour: 20, minute: 49, second: 42, millisecond: 268 }\n * @return {Object}\n */\n toObject(opts = {}) {\n if (!this.isValid) return {};\n\n const base = Object.assign({}, this.c);\n\n if (opts.includeConfig) {\n base.outputCalendar = this.outputCalendar;\n base.numberingSystem = this.loc.numberingSystem;\n base.locale = this.loc.locale;\n }\n return base;\n }\n\n /**\n * Returns a JavaScript Date equivalent to this DateTime.\n * @return {Date}\n */\n toJSDate() {\n return new Date(this.isValid ? this.ts : NaN);\n }\n\n // COMPARE\n\n /**\n * Return the difference between two DateTimes as a Duration.\n * @param {DateTime} otherDateTime - the DateTime to compare this one to\n * @param {string|string[]} [unit=['milliseconds']] - the unit or array of units (such as 'hours' or 'days') to include in the duration.\n * @param {Object} opts - options that affect the creation of the Duration\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @example\n * var i1 = DateTime.fromISO('1982-05-25T09:45'),\n * i2 = DateTime.fromISO('1983-10-14T10:30');\n * i2.diff(i1).toObject() //=> { milliseconds: 43807500000 }\n * i2.diff(i1, 'hours').toObject() //=> { hours: 12168.75 }\n * i2.diff(i1, ['months', 'days']).toObject() //=> { months: 16, days: 19.03125 }\n * i2.diff(i1, ['months', 'days', 'hours']).toObject() //=> { months: 16, days: 19, hours: 0.75 }\n * @return {Duration}\n */\n diff(otherDateTime, unit = \"milliseconds\", opts = {}) {\n if (!this.isValid || !otherDateTime.isValid) {\n return Duration.invalid(\n this.invalid || otherDateTime.invalid,\n \"created by diffing an invalid DateTime\"\n );\n }\n\n const durOpts = Object.assign(\n { locale: this.locale, numberingSystem: this.numberingSystem },\n opts\n );\n\n const units = maybeArray(unit).map(Duration.normalizeUnit),\n otherIsLater = otherDateTime.valueOf() > this.valueOf(),\n earlier = otherIsLater ? this : otherDateTime,\n later = otherIsLater ? otherDateTime : this,\n diffed = diff(earlier, later, units, durOpts);\n\n return otherIsLater ? diffed.negate() : diffed;\n }\n\n /**\n * Return the difference between this DateTime and right now.\n * See {@link diff}\n * @param {string|string[]} [unit=['milliseconds']] - the unit or units units (such as 'hours' or 'days') to include in the duration\n * @param {Object} opts - options that affect the creation of the Duration\n * @param {string} [opts.conversionAccuracy='casual'] - the conversion system to use\n * @return {Duration}\n */\n diffNow(unit = \"milliseconds\", opts = {}) {\n return this.diff(DateTime.now(), unit, opts);\n }\n\n /**\n * Return an Interval spanning between this DateTime and another DateTime\n * @param {DateTime} otherDateTime - the other end point of the Interval\n * @return {Interval}\n */\n until(otherDateTime) {\n return this.isValid ? Interval.fromDateTimes(this, otherDateTime) : this;\n }\n\n /**\n * Return whether this DateTime is in the same unit of time as another DateTime.\n * Higher-order units must also be identical for this function to return `true`.\n * Note that time zones are **ignored** in this comparison, which compares the **local** calendar time. Use {@link setZone} to convert one of the dates if needed.\n * @param {DateTime} otherDateTime - the other DateTime\n * @param {string} unit - the unit of time to check sameness on\n * @example DateTime.now().hasSame(otherDT, 'day'); //~> true if otherDT is in the same current calendar day\n * @return {boolean}\n */\n hasSame(otherDateTime, unit) {\n if (!this.isValid) return false;\n\n const inputMs = otherDateTime.valueOf();\n const otherZoneDateTime = this.setZone(otherDateTime.zone, { keepLocalTime: true });\n return otherZoneDateTime.startOf(unit) <= inputMs && inputMs <= otherZoneDateTime.endOf(unit);\n }\n\n /**\n * Equality check\n * Two DateTimes are equal iff they represent the same millisecond, have the same zone and location, and are both valid.\n * To compare just the millisecond values, use `+dt1 === +dt2`.\n * @param {DateTime} other - the other DateTime\n * @return {boolean}\n */\n equals(other) {\n return (\n this.isValid &&\n other.isValid &&\n this.valueOf() === other.valueOf() &&\n this.zone.equals(other.zone) &&\n this.loc.equals(other.loc)\n );\n }\n\n /**\n * Returns a string representation of a this time relative to now, such as \"in two days\". Can only internationalize if your\n * platform supports Intl.RelativeTimeFormat. Rounds down by default.\n * @param {Object} options - options that affect the output\n * @param {DateTime} [options.base=DateTime.now()] - the DateTime to use as the basis to which this time is compared. Defaults to now.\n * @param {string} [options.style=\"long\"] - the style of units, must be \"long\", \"short\", or \"narrow\"\n * @param {string|string[]} options.unit - use a specific unit or array of units; if omitted, or an array, the method will pick the best unit. Use an array or one of \"years\", \"quarters\", \"months\", \"weeks\", \"days\", \"hours\", \"minutes\", or \"seconds\"\n * @param {boolean} [options.round=true] - whether to round the numbers in the output.\n * @param {number} [options.padding=0] - padding in milliseconds. This allows you to round up the result if it fits inside the threshold. Don't use in combination with {round: false} because the decimal output will include the padding.\n * @param {string} options.locale - override the locale of this DateTime\n * @param {string} options.numberingSystem - override the numberingSystem of this DateTime. The Intl system may choose not to honor this\n * @example DateTime.now().plus({ days: 1 }).toRelative() //=> \"in 1 day\"\n * @example DateTime.now().setLocale(\"es\").toRelative({ days: 1 }) //=> \"dentro de 1 día\"\n * @example DateTime.now().plus({ days: 1 }).toRelative({ locale: \"fr\" }) //=> \"dans 23 heures\"\n * @example DateTime.now().minus({ days: 2 }).toRelative() //=> \"2 days ago\"\n * @example DateTime.now().minus({ days: 2 }).toRelative({ unit: \"hours\" }) //=> \"48 hours ago\"\n * @example DateTime.now().minus({ hours: 36 }).toRelative({ round: false }) //=> \"1.5 days ago\"\n */\n toRelative(options = {}) {\n if (!this.isValid) return null;\n const base = options.base || DateTime.fromObject({ zone: this.zone }),\n padding = options.padding ? (this < base ? -options.padding : options.padding) : 0;\n let units = [\"years\", \"months\", \"days\", \"hours\", \"minutes\", \"seconds\"];\n let unit = options.unit;\n if (Array.isArray(options.unit)) {\n units = options.unit;\n unit = undefined;\n }\n return diffRelative(\n base,\n this.plus(padding),\n Object.assign(options, {\n numeric: \"always\",\n units,\n unit\n })\n );\n }\n\n /**\n * Returns a string representation of this date relative to today, such as \"yesterday\" or \"next month\".\n * Only internationalizes on platforms that supports Intl.RelativeTimeFormat.\n * @param {Object} options - options that affect the output\n * @param {DateTime} [options.base=DateTime.now()] - the DateTime to use as the basis to which this time is compared. Defaults to now.\n * @param {string} options.locale - override the locale of this DateTime\n * @param {string} options.unit - use a specific unit; if omitted, the method will pick the unit. Use one of \"years\", \"quarters\", \"months\", \"weeks\", or \"days\"\n * @param {string} options.numberingSystem - override the numberingSystem of this DateTime. The Intl system may choose not to honor this\n * @example DateTime.now().plus({ days: 1 }).toRelativeCalendar() //=> \"tomorrow\"\n * @example DateTime.now().setLocale(\"es\").plus({ days: 1 }).toRelative() //=> \"\"mañana\"\n * @example DateTime.now().plus({ days: 1 }).toRelativeCalendar({ locale: \"fr\" }) //=> \"demain\"\n * @example DateTime.now().minus({ days: 2 }).toRelativeCalendar() //=> \"2 days ago\"\n */\n toRelativeCalendar(options = {}) {\n if (!this.isValid) return null;\n\n return diffRelative(\n options.base || DateTime.fromObject({ zone: this.zone }),\n this,\n Object.assign(options, {\n numeric: \"auto\",\n units: [\"years\", \"months\", \"days\"],\n calendary: true\n })\n );\n }\n\n /**\n * Return the min of several date times\n * @param {...DateTime} dateTimes - the DateTimes from which to choose the minimum\n * @return {DateTime} the min DateTime, or undefined if called with no argument\n */\n static min(...dateTimes) {\n if (!dateTimes.every(DateTime.isDateTime)) {\n throw new InvalidArgumentError(\"min requires all arguments be DateTimes\");\n }\n return bestBy(dateTimes, i => i.valueOf(), Math.min);\n }\n\n /**\n * Return the max of several date times\n * @param {...DateTime} dateTimes - the DateTimes from which to choose the maximum\n * @return {DateTime} the max DateTime, or undefined if called with no argument\n */\n static max(...dateTimes) {\n if (!dateTimes.every(DateTime.isDateTime)) {\n throw new InvalidArgumentError(\"max requires all arguments be DateTimes\");\n }\n return bestBy(dateTimes, i => i.valueOf(), Math.max);\n }\n\n // MISC\n\n /**\n * Explain how a string would be parsed by fromFormat()\n * @param {string} text - the string to parse\n * @param {string} fmt - the format the string is expected to be in (see description)\n * @param {Object} options - options taken by fromFormat()\n * @return {Object}\n */\n static fromFormatExplain(text, fmt, options = {}) {\n const { locale = null, numberingSystem = null } = options,\n localeToUse = Locale.fromOpts({\n locale,\n numberingSystem,\n defaultToEN: true\n });\n return explainFromTokens(localeToUse, text, fmt);\n }\n\n /**\n * @deprecated use fromFormatExplain instead\n */\n static fromStringExplain(text, fmt, options = {}) {\n return DateTime.fromFormatExplain(text, fmt, options);\n }\n\n // FORMAT PRESETS\n\n /**\n * {@link toLocaleString} format like 10/14/1983\n * @type {Object}\n */\n static get DATE_SHORT() {\n return Formats.DATE_SHORT;\n }\n\n /**\n * {@link toLocaleString} format like 'Oct 14, 1983'\n * @type {Object}\n */\n static get DATE_MED() {\n return Formats.DATE_MED;\n }\n\n /**\n * {@link toLocaleString} format like 'Fri, Oct 14, 1983'\n * @type {Object}\n */\n static get DATE_MED_WITH_WEEKDAY() {\n return Formats.DATE_MED_WITH_WEEKDAY;\n }\n\n /**\n * {@link toLocaleString} format like 'October 14, 1983'\n * @type {Object}\n */\n static get DATE_FULL() {\n return Formats.DATE_FULL;\n }\n\n /**\n * {@link toLocaleString} format like 'Tuesday, October 14, 1983'\n * @type {Object}\n */\n static get DATE_HUGE() {\n return Formats.DATE_HUGE;\n }\n\n /**\n * {@link toLocaleString} format like '09:30 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get TIME_SIMPLE() {\n return Formats.TIME_SIMPLE;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get TIME_WITH_SECONDS() {\n return Formats.TIME_WITH_SECONDS;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23 AM EDT'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get TIME_WITH_SHORT_OFFSET() {\n return Formats.TIME_WITH_SHORT_OFFSET;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23 AM Eastern Daylight Time'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get TIME_WITH_LONG_OFFSET() {\n return Formats.TIME_WITH_LONG_OFFSET;\n }\n\n /**\n * {@link toLocaleString} format like '09:30', always 24-hour.\n * @type {Object}\n */\n static get TIME_24_SIMPLE() {\n return Formats.TIME_24_SIMPLE;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23', always 24-hour.\n * @type {Object}\n */\n static get TIME_24_WITH_SECONDS() {\n return Formats.TIME_24_WITH_SECONDS;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23 EDT', always 24-hour.\n * @type {Object}\n */\n static get TIME_24_WITH_SHORT_OFFSET() {\n return Formats.TIME_24_WITH_SHORT_OFFSET;\n }\n\n /**\n * {@link toLocaleString} format like '09:30:23 Eastern Daylight Time', always 24-hour.\n * @type {Object}\n */\n static get TIME_24_WITH_LONG_OFFSET() {\n return Formats.TIME_24_WITH_LONG_OFFSET;\n }\n\n /**\n * {@link toLocaleString} format like '10/14/1983, 9:30 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_SHORT() {\n return Formats.DATETIME_SHORT;\n }\n\n /**\n * {@link toLocaleString} format like '10/14/1983, 9:30:33 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_SHORT_WITH_SECONDS() {\n return Formats.DATETIME_SHORT_WITH_SECONDS;\n }\n\n /**\n * {@link toLocaleString} format like 'Oct 14, 1983, 9:30 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_MED() {\n return Formats.DATETIME_MED;\n }\n\n /**\n * {@link toLocaleString} format like 'Oct 14, 1983, 9:30:33 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_MED_WITH_SECONDS() {\n return Formats.DATETIME_MED_WITH_SECONDS;\n }\n\n /**\n * {@link toLocaleString} format like 'Fri, 14 Oct 1983, 9:30 AM'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_MED_WITH_WEEKDAY() {\n return Formats.DATETIME_MED_WITH_WEEKDAY;\n }\n\n /**\n * {@link toLocaleString} format like 'October 14, 1983, 9:30 AM EDT'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_FULL() {\n return Formats.DATETIME_FULL;\n }\n\n /**\n * {@link toLocaleString} format like 'October 14, 1983, 9:30:33 AM EDT'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_FULL_WITH_SECONDS() {\n return Formats.DATETIME_FULL_WITH_SECONDS;\n }\n\n /**\n * {@link toLocaleString} format like 'Friday, October 14, 1983, 9:30 AM Eastern Daylight Time'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_HUGE() {\n return Formats.DATETIME_HUGE;\n }\n\n /**\n * {@link toLocaleString} format like 'Friday, October 14, 1983, 9:30:33 AM Eastern Daylight Time'. Only 12-hour if the locale is.\n * @type {Object}\n */\n static get DATETIME_HUGE_WITH_SECONDS() {\n return Formats.DATETIME_HUGE_WITH_SECONDS;\n }\n}\n\n/**\n * @private\n */\nexport function friendlyDateTime(dateTimeish) {\n if (DateTime.isDateTime(dateTimeish)) {\n return dateTimeish;\n } else if (dateTimeish && dateTimeish.valueOf && isNumber(dateTimeish.valueOf())) {\n return DateTime.fromJSDate(dateTimeish);\n } else if (dateTimeish && typeof dateTimeish === \"object\") {\n return DateTime.fromObject(dateTimeish);\n } else {\n throw new InvalidArgumentError(\n `Unknown datetime argument: ${dateTimeish}, of type ${typeof dateTimeish}`\n );\n }\n}\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DateTime } from \"luxon\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CountdownOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n datetimeValue: string;\n width: number;\n height: number;\n pastAllowed: boolean;\n countsUp: boolean;\n paused: boolean;\n pausedDateTime: string;\n time_textColor: string;\n time_fontFamily: string;\n time_fontSize: number;\n time_fontWeight: number;\n time_textTransform: string;\n time_fontStyle: string;\n time_letterSpacing: number;\n label_textColor: string;\n label_fontFamily: string;\n label_fontSize: number;\n label_fontWeight: number;\n label_textTransform: string;\n label_fontStyle: string;\n label_letterSpacing: number;\n syncFonts: boolean;\n syncTextColors: boolean;\n marginX: number;\n marginY: number;\n timeUnits: string;\n showDivider: boolean;\n showYears: boolean;\n showMonths: boolean;\n showWeeks: boolean;\n showDays: boolean;\n showHours: boolean;\n showMinutes: boolean;\n showSeconds: boolean;\n}\n\nconst date = new Date();\nconst year = date.getFullYear();\nconst month = date.getMonth();\nconst day = date.getDate();\nconst future = new Date(year, month + 1, day + 12).toISOString().split(\"T\")[0];\nconst dateTime = DateTime.now().toISO();\n\nexport const defaultCountdownOptions = {\n datetimeValue: future,\n countsUp: false,\n pastAllowed: false,\n paused: false,\n pausedDateTime: dateTime,\n time_textColor: \"#1d283a\",\n time_fontFamily: \"Montserrat\",\n time_fontSize: 108,\n time_fontWeight: 700,\n time_textTransform: \"capitalize\",\n time_fontStyle: \"normal\",\n time_letterSpacing: 0,\n label_textColor: \"#16a249\",\n label_fontFamily: \"Montserrat\",\n label_fontSize: 24,\n label_fontWeight: 400,\n label_textTransform: \"uppercase\",\n label_fontStyle: \"normal\",\n label_letterSpacing: 0,\n syncFonts: true,\n syncTextColors: false,\n marginX: 5,\n marginY: 0,\n timeUnits: \"days,hours,minutes,seconds\",\n showDivider: true,\n showYears: false,\n showMonths: false,\n showWeeks: false,\n showDays: true,\n showHours: true,\n showMinutes: true,\n showSeconds: true,\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultCountdownOptions,\n showDivider: true,\n};\n\nconst CreateCountdown = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"Countdown\",\n wid: makeId(6),\n name: DEFAULT_LAYER_NAMES.Countdown,\n });\n};\n\nexport default CreateCountdown;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CalendarWeekOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n data: any[];\n previewLiveData: boolean;\n\n use24Hour: boolean;\n weekOffset: number;\n syncFonts: boolean;\n syncTextColors: boolean;\n\n // TODO: Do we want this? Never exposed..\n dayFormat: string;\n\n verticalLines_color: string;\n event_backgroundColor: string;\n\n // can be exposed together\n today_backgroundColor: string;\n today_textColor: string; // TODO: Might be better to store as todayTextColor, so does not get synced... but worried about backwards compat..\n\n eventTitle_fontFamily: string | null;\n eventTitle_textColor: string | null;\n eventTitle_fontSize: number;\n eventTitle_fontWeight: number;\n eventTitle_letterSpacing: number;\n eventTitle_lineHeight: number;\n eventTitle_textAlign: string;\n eventTitle_fontStyle: string;\n eventTitle_textTransform: string;\n\n eventTime_fontFamily: string | null;\n eventTime_textColor: string | null;\n eventTime_fontSize: number;\n eventTime_fontWeight: number;\n eventTime_letterSpacing: number;\n eventTime_lineHeight: number;\n eventTime_textAlign: string;\n eventTime_fontStyle: string;\n eventTime_textTransform: string;\n\n days_fontFamily: string | null;\n days_textColor: string | null;\n days_fontSize: number;\n days_fontWeight: number;\n days_letterSpacing: number;\n days_lineHeight: number;\n days_textAlign: string;\n days_fontStyle: string;\n days_textTransform: string;\n\n multidayEventTitle_fontFamily: string | null;\n multidayEventTitle_textColor: string | null;\n multidayEventTitle_fontSize: number;\n multidayEventTitle_fontWeight: number;\n multidayEventTitle_letterSpacing: number;\n multidayEventTitle_lineHeight: number;\n multidayEventTitle_textAlign: string;\n multidayEventTitle_fontStyle: string;\n multidayEventTitle_textTransform: string;\n}\nconst mainColor = \"rgba(52, 66, 86, 100)\";\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n borderRadius: 0,\n previewLiveData: false,\n use24Hour: false,\n weekOffset: 0,\n dayFormat: \"date-short\",\n verticalLines_color: \"rgba(0,0,0,0.2)\",\n event_backgroundColor: mainColor,\n today_backgroundColor: mainColor,\n today_textColor: \"white\",\n syncFonts: false,\n syncTextColors: false,\n};\n\nexport const defaultTextOptions: any = {\n days_fontSize: 32,\n days_fontWeight: 700,\n days_textAlign: \"center\",\n days_textColor: mainColor,\n days_textTransform: \"uppercase\",\n days_lineHeight: 1.2,\n eventTitle_fontSize: 28,\n eventTitle_fontWeight: 400,\n eventTitle_textAlign: \"left\",\n eventTitle_textColor: mainColor,\n eventTitle_textTransform: \"none\",\n eventTitle_lineHeight: 1.2,\n eventTime_fontSize: 32,\n eventTime_fontWeight: 700,\n eventTime_textAlign: \"left\",\n eventTime_textColor: mainColor,\n eventTime_textTransform: \"none\",\n eventTime_lineHeight: 1.2,\n multidayEventTitle_fontSize: 28,\n multidayEventTitle_fontWeight: 400,\n multidayEventTitle_textAlign: \"left\",\n multidayEventTitle_textColor: \"#ffffff\",\n multidayEventTitle_textTransform: \"none\",\n multidayEventTitle_lineHeight: 2.0,\n};\n\n// Add uniform defaults to all text fields:\n[\"eventTitle\", \"eventTime\", \"days\", \"multidayEventTitle\"].forEach((prop) => {\n defaultTextOptions[`${prop}_fontFamily`] = \"Montserrat\";\n defaultTextOptions[`${prop}_textDecoration`] = \"none\";\n defaultTextOptions[`${prop}_fontStyle`] = \"normal\";\n defaultTextOptions[`${prop}_letterSpacing`] = 0;\n});\n\ndefaultTextOptions.days_textAlign = \"center\";\n\nconst CreateCalendarWeek = (options: Partial) => {\n return Object.assign({}, defaultOptions, defaultTextOptions, options, {\n wid: makeId(),\n type: \"CalendarWeek\",\n name: DEFAULT_LAYER_NAMES.CalendarWeek,\n });\n};\n\nexport default CreateCalendarWeek;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CalendarDayOptions\n extends ComponentOptions,\n TransformOptions,\n BackgroundOptions {\n data?: any[];\n previewLiveData: boolean;\n\n dayOffset: number;\n dayFormat: string;\n startHour: number;\n endHour: number;\n use24Hour: boolean;\n useDefaultStartTime: boolean;\n showEventLocation: boolean;\n showEventTime: boolean;\n syncFonts: boolean;\n syncTextColors: boolean;\n\n // Hour Lines\n showVerticalLines: boolean;\n verticalLinesColor: string;\n\n // Now Line\n nowLineColor: string;\n showNowLine: boolean;\n\n // Background\n showBackground: boolean;\n calendarBackgroundColor: string;\n eventBackgroundColor: string;\n\n // Text Styles ---------------------------------------\n\n day_fontFamily: string | null;\n day_textColor: string | null;\n day_fontSize: number;\n day_fontWeight: number;\n day_letterSpacing: number;\n day_lineHeight: number;\n day_textAlign: string;\n day_fontStyle: string;\n day_textTransform: string;\n day_textDecoration: string;\n\n // title_fontFamily: string | null;\n // title_textColor: string;\n // title_fontSize: number;\n // title_fontWeight: number;\n // title_letterSpacing: number;\n // title_lineHeight: number;\n // title_textAlign: string;\n // title_fontStyle: string;\n // title_textTransform: string;\n // title_textDecoration: string;\n\n hours_fontFamily: string | null;\n hours_textColor: string | null;\n hours_fontSize: number;\n hours_fontWeight: number;\n hours_letterSpacing: number;\n hours_lineHeight: number;\n hours_textAlign: string;\n hours_fontStyle: string;\n hours_textTransform: string;\n hours_textDecoration: string;\n\n eventTitle_fontFamily: string | null;\n eventTitle_textColor: string | null;\n eventTitle_fontSize: number;\n eventTitle_fontWeight: number;\n eventTitle_letterSpacing: number;\n eventTitle_lineHeight: number;\n eventTitle_textAlign: string;\n eventTitle_fontStyle: string;\n eventTitle_textTransform: string;\n eventTitle_textDecoration: string;\n\n eventLocation_fontFamily: string | null;\n eventLocation_textColor: string | null;\n eventLocation_fontSize: number;\n eventLocation_fontWeight: number;\n eventLocation_letterSpacing: number;\n eventLocation_lineHeight: number;\n eventLocation_textAlign: string;\n eventLocation_fontStyle: string;\n eventLocation_textTransform: string;\n eventLocation_textDecoration: string;\n\n eventTime_fontFamily: string | null;\n eventTime_textColor: string | null;\n eventTime_fontSize: number;\n eventTime_fontWeight: number;\n eventTime_letterSpacing: number;\n eventTime_lineHeight: number;\n eventTime_textAlign: string;\n eventTime_fontStyle: string;\n eventTime_textTransform: string;\n eventTime_textDecoration: string;\n}\n\nexport const defaultTextOptions: any = {\n day_fontSize: 72,\n day_fontWeight: 300,\n day_textColor: \"rgba(22,162,73,100)\",\n day_textTransform: \"uppercase\",\n day_textAlign: \"center\",\n hours_fontSize: 28,\n hours_fontWeight: 700,\n hours_textColor: \"rgba(52, 66, 86, 100)\",\n hours_textTransform: \"none\",\n eventTitle_fontSize: 32,\n eventTitle_fontWeight: 700,\n eventTitle_textColor: \"rgba(255,255,255,100)\",\n eventTitle_textTransform: \"capitalize\",\n eventLocation_fontSize: 36,\n eventLocation_fontWeight: 300,\n eventLocation_textColor: \"rgba(255,255,255,100)\",\n eventLocation_textTransform: \"none\",\n eventTime_fontSize: 26,\n eventTime_fontWeight: 300,\n eventTime_textColor: \"rgba(255,255,255,100)\",\n eventTime_textTransform: \"none\",\n};\n\n[\"day\", \"hours\", \"eventTitle\", \"eventLocation\", \"eventTime\"].forEach((prop) => {\n defaultTextOptions[`${prop}_fontFamily`] = \"Montserrat\";\n defaultTextOptions[`${prop}_textDecoration`] = \"none\";\n defaultTextOptions[`${prop}_fontStyle`] = \"normal\";\n defaultTextOptions[`${prop}_lineHeight`] = 1.2;\n defaultTextOptions[`${prop}_letterSpacing`] = 0;\n});\n\n[\"hours\", \"eventTitle\", \"eventLocation\", \"eventTime\"].forEach((prop) => {\n defaultTextOptions[`${prop}_textAlign`] = \"left\";\n});\n\nconst defaultOptions = {\n ...defaultTextOptions,\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBackgroundOptions,\n dayOffset: 0,\n dayFormat: \"today\",\n startHour: 8,\n endHour: 17,\n previewLiveData: false,\n // title: \"Calendar Title\",\n // showTitle: true,\n showEventLocation: true,\n showEventTime: true,\n use24Hour: false,\n useDefaultStartTime: true,\n showVerticalLines: true,\n showNowLine: true,\n showBackground: true,\n verticalLinesColor: \"rgba(203,213,225,100)\",\n nowLineColor: \"rgba(33,196,93,100)\",\n eventBackgroundColor: \"rgba(52,66,86,100)\",\n calendarBackgroundColor: \"rgba(241,245,249,100)\",\n syncFonts: false,\n syncTextColors: false,\n};\n\nconst CreateCalendarDay = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"CalendarDay\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.CalendarDay,\n });\n};\n\nexport default CreateCalendarDay;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface CalendarEventOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n dataIndex: number;\n data?: any[];\n // previewLiveData: boolean;\n\n showEventLocation: boolean;\n showEventCreator: boolean;\n showEventDescription: boolean;\n showDivider: boolean;\n use24Hour: boolean;\n dividerColor: string;\n syncFonts: boolean;\n syncTextColors: boolean;\n\n eventTitle_fontFamily: string | null;\n eventTitle_textColor: string | null;\n eventTitle_fontSize: number;\n eventTitle_fontWeight: number;\n eventTitle_letterSpacing: number;\n eventTitle_lineHeight: number;\n eventTitle_textAlign: string;\n eventTitle_fontStyle: string; // \"normal\" | \"italic\"\n eventTitle_textTransform: string;\n eventTitle_textDecoration: string; // \"none\" | \"strikethrough\" | \"underline\"\n // Technically one could do both underline and strike.....do we care about that case?\n\n eventTime_fontFamily: string | null;\n eventTime_textColor: string | null;\n eventTime_fontSize: number;\n eventTime_fontWeight: number;\n eventTime_letterSpacing: number;\n eventTime_lineHeight: number;\n eventTime_textAlign: string;\n eventTime_fontStyle: string;\n eventTime_textTransform: string;\n eventTime_textDecoration: string;\n\n eventLocation_fontFamily: string | null;\n eventLocation_textColor: string | null;\n eventLocation_fontSize: number;\n eventLocation_fontWeight: number;\n eventLocation_letterSpacing: number;\n eventLocation_lineHeight: number;\n eventLocation_textAlign: string;\n eventLocation_fontStyle: string;\n eventLocation_textTransform: string;\n eventLocation_textDecoration: string;\n\n eventCreator_fontFamily: string | null;\n eventCreator_textColor: string | null;\n eventCreator_fontSize: number;\n eventCreator_fontWeight: number;\n eventCreator_letterSpacing: number;\n eventCreator_lineHeight: number;\n eventCreator_textAlign: string;\n eventCreator_fontStyle: string;\n eventCreator_textTransform: string;\n eventCreator_textDecoration: string;\n\n eventDescription_fontFamily: string | null;\n eventDescription_textColor: string | null;\n eventDescription_fontSize: number;\n eventDescription_fontWeight: number;\n eventDescription_letterSpacing: number;\n eventDescription_lineHeight: number;\n eventDescription_textAlign: string;\n eventDescription_fontStyle: string;\n eventDescription_textTransform: string;\n eventDescription_textDecoration: string;\n}\n\nconst mainColor = \"rgba(52,66,86,100)\";\nconst accentColor = \"rgba(244,62,92,100)\";\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n borderRadius: 0,\n dataIndex: 0,\n // previewLiveData: false,\n showEventLocation: true,\n showEventCreator: true,\n showDivider: true,\n showEventDescription: true,\n use24Hour: false,\n dividerColor: accentColor,\n syncFonts: false,\n syncTextColors: false,\n};\n\nexport const defaultTextOptions: any = {\n eventTitle_fontSize: 180,\n eventTitle_fontWeight: 700,\n eventTitle_textColor: mainColor,\n eventTime_fontSize: 120,\n eventTime_fontWeight: 400,\n eventTime_textColor: accentColor,\n eventLocation_fontSize: 84,\n eventLocation_fontWeight: 700,\n eventLocation_textColor: mainColor,\n eventDescription_fontSize: 60,\n eventDescription_fontWeight: 400,\n eventDescription_textColor: mainColor,\n eventCreator_fontSize: 60,\n eventCreator_fontWeight: 700,\n eventCreator_textColor: mainColor,\n};\n\n// Add uniform defaults to all text fields:\n[\"Title\", \"Location\", \"Time\", \"Description\", \"Creator\"].forEach((prop) => {\n defaultTextOptions[`event${prop}_fontFamily`] = \"Montserrat\";\n defaultTextOptions[`event${prop}_textTransform`] = \"none\";\n defaultTextOptions[`event${prop}_textDecoration`] = \"none\";\n defaultTextOptions[`event${prop}_fontStyle`] = \"normal\";\n defaultTextOptions[`event${prop}_lineHeight`] = 1.2;\n defaultTextOptions[`event${prop}_letterSpacing`] = 0;\n defaultTextOptions[`event${prop}_textAlign`] = \"left\";\n});\n\nconst CreateCalendarEvent = (options: Partial) => {\n return Object.assign({}, defaultOptions, defaultTextOptions, options, {\n wid: makeId(),\n type: \"CalendarEvent\",\n name: DEFAULT_LAYER_NAMES.CalendarEvent,\n });\n};\n\nexport default CreateCalendarEvent;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { DateTime, DateTimeFormatOptions } from \"luxon\";\n// Luxon's preset formats (most of them):\n\nexport const LUXON_TIME_FORMATS = [\n \"TIME_SIMPLE\",\n \"TIME_WITH_SECONDS\",\n \"TIME_24_SIMPLE\",\n \"TIME_24_WITH_SECONDS\",\n];\n\nexport const LUXON_DATE_FORMATS = [\n \"DATE_SHORT\",\n \"DATE_MED\",\n \"DATE_MED_WITH_WEEKDAY\",\n \"DATE_FULL\",\n \"DATE_HUGE\",\n];\n\nexport const LUXON_DATETIME_FORMATS = [\n ...LUXON_DATE_FORMATS,\n ...LUXON_TIME_FORMATS,\n \"DATETIME_SHORT\",\n \"DATETIME_SHORT_WITH_SECONDS\",\n \"DATETIME_MED\",\n \"DATETIME_MED_WITH_SECONDS\",\n];\n\n/**\n * Instead of relying on Luxon presets, we copy in all those presets here, and then we can add to this list\n */\n\nexport const CUSTOM_DATETIME_FORMATS: DateTimeFormatOptions[] = [\n { year: \"numeric\", month: \"numeric\", day: \"numeric\" }, // DATE_SHORT: 10/14/1983\n { year: \"numeric\", month: \"short\", day: \"numeric\" }, // DATE_MED: Oct 14, 1983\n { year: \"numeric\", month: \"short\", day: \"numeric\", weekday: \"short\" }, // DATE_MED_WITH_WEEKDAY: Fri, Oct 14, 1983\n { year: \"numeric\", month: \"long\", day: \"numeric\" }, // DATE_FULL: October 14, 1983\n { year: \"numeric\", month: \"long\", day: \"numeric\", weekday: \"long\" }, // DATE_HUGE: Friday, October 14, 1983\n { hour: \"numeric\", minute: \"numeric\" }, // TIME_SIMPLE: \t1:30 PM\n { hour: \"numeric\", minute: \"numeric\", second: \"numeric\" }, // TIME_WITH_SECONDS: \t1:30:23 PM\n { hour: \"numeric\", minute: \"numeric\", hour12: false }, // TIME_24_SIMPLE: 13:30\n { hour: \"numeric\", minute: \"numeric\", second: \"numeric\", hour12: false }, // TIME_24_WITH_SECONDS: 13:30:23\n {\n year: \"numeric\",\n month: \"numeric\",\n day: \"numeric\",\n hour: \"numeric\",\n minute: \"numeric\",\n }, // DATETIME_SHORT: 10/14/1983, 1:30 PM\n {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n hour: \"numeric\",\n minute: \"numeric\",\n }, // DATETIME_MED: Oct 14, 1983, 1:30 PM\n {\n year: \"numeric\",\n month: \"numeric\",\n day: \"numeric\",\n hour: \"numeric\",\n minute: \"numeric\",\n second: \"numeric\",\n }, // DATETIME_SHORT_WITH_SECONDS: 10/14/1983, 1:30:23 PM\n {\n year: \"numeric\",\n month: \"short\",\n day: \"numeric\",\n hour: \"numeric\",\n minute: \"numeric\",\n second: \"numeric\",\n }, // DATETIME_MED_WITH_SECONDS: Oct 14, 1983, 1:30:23 PM\n\n // CUSTOM, NON-LUXON PRESETS\n { weekday: \"short\" },\n { weekday: \"long\" },\n { month: \"short\" },\n { month: \"long\" },\n { month: \"long\", day: \"numeric\" },\n { month: \"numeric\", day: \"numeric\" },\n { hour: \"numeric\" },\n];\n\n// https://stackoverflow.com/questions/3191664/list-of-all-locales-and-their-short-codes\nexport const LOCALE_OPTIONS = [\n {\n label: \"English (U.S.)\",\n value: \"en-US\",\n },\n {\n label: \"English (U.K.)\",\n value: \"en-UK\",\n },\n {\n label: \"French\",\n value: \"fr-FR\",\n },\n {\n label: \"French (Can.)\",\n value: \"fr-CA\",\n },\n {\n label: \"German\",\n value: \"de-DE\",\n },\n {\n label: \"Spanish\",\n value: \"es-ES\",\n },\n {\n label: \"Swedish\",\n value: \"sv-SE\",\n },\n {\n label: \"Danish\",\n value: \"da\",\n },\n {\n label: \"Dutch (Neth.)\",\n value: \"nl-NL\",\n },\n {\n label: \"Dutch (Bel.)\",\n value: \"nl-BE\",\n },\n];\n\nexport interface DatetimeOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n datetimeFormat: string;\n datetimeValue: string;\n userSpecifiedLocale: string;\n useBrowserLocale: boolean;\n usePlayerTimezone: boolean;\n timezone: string | null;\n textColor: string;\n fontFamily: string;\n fontSize: number;\n fontWeight: number;\n textTransform: string;\n fontStyle: string;\n letterSpacing: number;\n lineHeight: number;\n alignHorizontal: string;\n // isDate: boolean;\n // isTime: boolean;\n showSeconds: boolean; // NOTE: Not used currently\n use24: boolean; // NOTE: Not used currently\n useCurrentTime: boolean;\n}\n\n// export const defaultTextOptions = {\n// textColor: \"#000000\",\n// fontFamily: \"Cabin\",\n// fontSize: 84,\n// fontWeight: 400,\n// textTransform: \"capitalize\",\n// fontStyle: \"normal\",\n// letterSpacing: 0,\n// alignHorizontal: \"Left\",\n// lineHeight: 1,\n// };\n\nexport const defaultTextOptions = {\n textColor: \"#1d283a\",\n fontFamily: \"Montserrat\",\n fontSize: 84,\n fontWeight: 700,\n textTransform: \"uppercase\",\n fontStyle: \"normal\",\n letterSpacing: 0,\n textAlign: \"center\",\n lineHeight: 1.2,\n alignHorizontal: \"center\",\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultTextOptions,\n datetimeValue: DateTime.now(),\n datetimeFormat: JSON.stringify(CUSTOM_DATETIME_FORMATS[0]),\n userSpecifiedLocale: \"en-US\",\n useBrowserLocale: true,\n usePlayerTimezone: true,\n timezone: null,\n showSeconds: true,\n use24: false,\n useCurrentTime: false,\n\n // isDate: false,\n // isTime: false,\n};\n\nconst CreateDatetime = (options: Partial) => {\n const wid = makeId();\n\n return Object.assign({ wid }, defaultOptions, options, {\n type: \"Datetime\",\n canScaleY: false,\n });\n};\n\nexport default CreateDatetime;\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport {\n ShadowOptions,\n DefaultShadowOptions,\n} from \"@/components/widgets/ShadowOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface SvgOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions,\n AnimationOptions,\n ShadowOptions {\n path: string;\n numSides: number;\n innerRadius: number;\n starEnabled: boolean;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...DefaultShadowOptions,\n ...getDefaultAnimationOptions(),\n};\n\nconst CreateSvg = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n wid: makeId(),\n type: \"Svg\",\n name: DEFAULT_LAYER_NAMES.Polygon,\n });\n};\n\nexport default CreateSvg;\n","import {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport {\n BackgroundOptions,\n DefaultBackgroundOptions,\n} from \"@/components/widgets/BackgroundOptions\";\nimport { makeId } from \"@/utils\";\nimport { DEFAULT_LAYER_NAMES } from \"@/constants\";\n\nexport interface StackedGraphOptions\n extends ComponentOptions,\n TransformOptions,\n BorderOptions,\n BackgroundOptions {\n h: number;\n w: number;\n topMargin: number;\n rightMargin: number;\n bottomMargin: number;\n leftMargin: number;\n minValue: number;\n maxValue: number;\n barPadding: number;\n\n value_fontFamily: string;\n value_fontSize: number;\n value_fontWeight: number;\n value_fontStyle: string;\n value_textColor: string;\n legend_fontFamily: string;\n legend_fontSize: number;\n legend_fontWeight: number;\n legend_fontStyle: string;\n legend_textColor: string;\n legend_textTransform: string;\n showLegend: boolean;\n legendMargin: number;\n legendPosition: string;\n colors: string[];\n showValues: boolean;\n valuesAsPercent: boolean;\n xAxisColumnId: string;\n yAxisColumnId: string;\n}\n\nexport const defaultStackedGraphOptions = {\n value_fontFamily: \"Montserrat\",\n value_fontSize: 24,\n value_fontWeight: 600,\n value_fontStyle: \"normal\",\n value_textColor: \"#000000\",\n legend_fontFamily: \"Montserrat\",\n legend_fontSize: 24,\n legend_fontWeight: 300,\n legend_fontStyle: \"normal\",\n legend_textColor: \"#000000\",\n legend_textTransform: \"capitalize\",\n showValues: false,\n showLegend: true,\n legendMargin: 30,\n legendPosition: \"Right\",\n valuesAsPercent: false,\n /**\n * Stores the UUID of the data column representing the x axis\n */\n xAxisColumnId: \"a\",\n /**\n * Stores the UUID of the data column representing the y axis\n */\n yAxisColumnId: \"b\",\n colors: [\n \"#0D3B5A\",\n \"#185783\",\n \"#2984C5\",\n \"#3ABFF8\",\n \"#BAE5FD\",\n \"#E1F3FE\",\n \"#BAE5FD\",\n \"#3ABFF8\",\n \"#2984C5\",\n \"#185783\",\n ],\n};\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...DefaultBackgroundOptions,\n ...defaultStackedGraphOptions,\n};\n\nconst CreateStackedGraph = (options: Partial) => {\n return Object.assign({}, defaultOptions, options, {\n type: \"StackedGraph\",\n wid: makeId(),\n name: DEFAULT_LAYER_NAMES.StackedGraph,\n });\n};\n\nexport default CreateStackedGraph;\n","import CreateEllipse from \"./Ellipse/EllipseOptions\";\nimport CreateImage from \"./Image/ImageOptions\";\nimport CreateRectangle from \"./Rectangle/RectangleDefinition\";\nimport CreateText from \"./Text/TextOptions\";\nimport CreateMultiframe from \"./Multiframe/MultiframeOptions\";\nimport CreateTriangle from \"./Triangle/TriangleOptions\";\nimport CreateLine from \"./Line/LineOptions\";\nimport CreateAnalogClock from \"./AnalogClock/AnalogClockOptions\";\nimport CreatePieGraph from \"./PieGraph/PieGraphOptions\";\nimport CreateLineGraph from \"./LineGraph/LineGraphOptions\";\nimport CreateBarGraph from \"./BarGraph/BarGraphOptions\";\nimport CreateColumnGraph from \"./ColumnGraph/ColumnGraphOptions\";\nimport CreateProgressDonut from \"./ProgressDonut/ProgressDonutOptions\";\nimport CreateProgressBar from \"./ProgressBar/ProgressBarOptions\";\nimport CreateCalendarRoomSchedule from \"./CalendarRoomSchedule/CalendarRoomScheduleOptions\";\nimport CreateCalendarAgenda from \"./CalendarAgenda/CalendarAgendaOptions\";\nimport CreateCountdown from \"./Countdown/CountdownOptions\";\nimport CreateRepeater from \"./Repeater/RepeaterOptions\";\nimport CreateCalendarWeek from \"./CalendarWeek/CalendarWeekOptions\";\nimport CreateCalendarDay from \"./CalendarDay/CalendarDayOptions\";\nimport CreateCalendarEvent from \"./CalendarEvent/CalendarEventOptions\";\nimport CreateDatetime from \"./Datetime/DatetimeOptions\";\nimport CreateSvg from \"./Svg/SvgOptions\";\nimport CreateStackedGraph from \"./StackedGraph/StackedGraphOptions\";\nimport { WidgetType } from \"@/types\";\n\nexport const getWidgetPropertyDefault = (\n widgetType: WidgetType,\n property: string\n) => {\n let defaults: any = null;\n switch (widgetType) {\n case \"Rectangle\":\n defaults = CreateRectangle({});\n break;\n case \"Text\":\n defaults = CreateText({});\n break;\n case \"Ellipse\":\n defaults = CreateEllipse({});\n break;\n case \"Image\":\n defaults = CreateImage({});\n break;\n case \"Group\":\n // Not needed.\n break;\n case \"Multiframe\":\n defaults = CreateMultiframe({});\n break;\n case \"Svg\":\n defaults = CreateSvg({});\n break;\n case \"Datetime\":\n defaults = CreateDatetime({});\n break;\n case \"CalendarEvent\":\n defaults = CreateCalendarEvent({});\n break;\n case \"CalendarDay\":\n defaults = CreateCalendarDay({});\n break;\n case \"CalendarWeek\":\n defaults = CreateCalendarWeek({});\n break;\n case \"Repeater\":\n defaults = CreateRepeater({});\n break;\n case \"Countdown\":\n defaults = CreateCountdown({});\n break;\n case \"CalendarAgenda\":\n defaults = CreateCalendarAgenda({});\n break;\n case \"CalendarRoomSchedule\":\n defaults = CreateCalendarRoomSchedule({});\n break;\n case \"ProgressBar\":\n defaults = CreateProgressBar({});\n break;\n case \"ProgressDonut\":\n defaults = CreateProgressDonut({});\n break;\n case \"ColumnGraph\":\n defaults = CreateColumnGraph({});\n break;\n case \"BarGraph\":\n defaults = CreateBarGraph({});\n break;\n case \"LineGraph\":\n defaults = CreateLineGraph({});\n break;\n case \"PieGraph\":\n defaults = CreatePieGraph({});\n break;\n case \"StackedGraph\":\n defaults = CreateStackedGraph({});\n break;\n case \"AnalogClock\":\n defaults = CreateAnalogClock({});\n break;\n case \"Triangle\":\n defaults = CreateTriangle({});\n break;\n case \"Line\":\n defaults = CreateLine({});\n break;\n }\n\n if (property in defaults) {\n return defaults[property];\n }\n};\n","import { NodeKind } from \"@/types/data\";\n\nexport const BarGraphPlaceholderData = [\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Nile\",\n formattedValue: \"Nile\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Length (mi.)\",\n uuid: \"b\",\n value: 4130,\n formattedValue: \"4130\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 0,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Amazon\",\n formattedValue: \"Amazon\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Length (mi.)\",\n uuid: \"b\",\n value: 3976,\n formattedValue: \"3976\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 1,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Yangtze\",\n formattedValue: \"Yangtze\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Length (mi.)\",\n uuid: \"b\",\n value: 3917,\n formattedValue: \"3917\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 2,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Mississippi\",\n formattedValue: \"Mississippi\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Length (mi.)\",\n uuid: \"b\",\n value: 3902,\n formattedValue: \"3902\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 3,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Yenisei\",\n formattedValue: \"Yenisei\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Length (mi.)\",\n uuid: \"b\",\n value: 3445,\n formattedValue: \"3445\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 4,\n query: \"__sf_index__\",\n },\n ],\n].map((row) => {\n return row.map((node) => {\n return {\n ...node,\n kind: NodeKind.node,\n isArtificial: node.displayName === \"Index\",\n };\n });\n});\n","import { NodeKind } from \"@/types/data\";\n\nexport const ColumnGraphPlaceholderData = [\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Burj Khalifa\",\n formattedValue: \"Burj Khalifa\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 830,\n formattedValue: \"830\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 0,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Shanghai Tower\",\n formattedValue: \"Shanghai Tower\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 632,\n formattedValue: \"632\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 1,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Abraj Al-Bait\",\n formattedValue: \"Abraj Al-Bait\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 601,\n formattedValue: \"601\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 2,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Taipei 101\",\n formattedValue: \"Taipei 101\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 508,\n formattedValue: \"508\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 3,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Empire State Building\",\n formattedValue: \"Empire State Building\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 381,\n formattedValue: \"381\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 4,\n query: \"__sf_index__\",\n },\n ],\n].map((row) => {\n return row.map((node) => {\n return {\n ...node,\n kind: NodeKind.node,\n isArtificial: node.displayName === \"Index\",\n };\n });\n});\n","import { NodeKind } from \"@/types/data\";\n\nexport const LineGraphPlaceholderData = [\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Burj Khalifa\",\n formattedValue: \"Burj Khalifa\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 830,\n formattedValue: \"830\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 0,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Shanghai Tower\",\n formattedValue: \"Shanghai Tower\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 632,\n formattedValue: \"632\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 1,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Abraj Al-Bait\",\n formattedValue: \"Abraj Al-Bait\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 601,\n formattedValue: \"601\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 2,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Taipei 101\",\n formattedValue: \"Taipei 101\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 508,\n formattedValue: \"508\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 3,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Empire State Building\",\n formattedValue: \"Empire State Building\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Height (m)\",\n uuid: \"b\",\n value: 381,\n formattedValue: \"381\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 4,\n query: \"__sf_index__\",\n },\n ],\n].map((row) => {\n return row.map((node) => {\n return {\n ...node,\n kind: NodeKind.node,\n isArtificial: node.displayName === \"Index\",\n };\n });\n});\n","import { NodeKind } from \"@/types/data\";\n\nexport const PieGraphPlaceholderData = [\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Cinnamon\",\n formattedValue: \"Cinnamon\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 0,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Fennel Seeds\",\n formattedValue: \"Fennel Seeds\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 6,\n formattedValue: \"6\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 1,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Sichuan Peppercorns\",\n formattedValue: \"Sichuan Peppercorns\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 2,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Cloves\",\n formattedValue: \"Cloves\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 1,\n formattedValue: \"1\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 3,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Sand Ginger\",\n formattedValue: \"Sand Ginger\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 4,\n query: \"__sf_index__\",\n },\n ],\n].map((row) => {\n return row.map((node) => {\n return {\n ...node,\n kind: NodeKind.node,\n isArtificial: node.displayName === \"Index\",\n };\n });\n});\n","import { NodeKind } from \"@/types/data\";\n\nexport const StackedGraphPlaceholderData = [\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Cinnamon\",\n formattedValue: \"Cinnamon\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 0,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Fennel Seeds\",\n formattedValue: \"Fennel Seeds\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 6,\n formattedValue: \"6\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 1,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Sichuan Peppercorns\",\n formattedValue: \"Sichuan Peppercorns\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 2,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Cloves\",\n formattedValue: \"Cloves\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 1,\n formattedValue: \"1\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 3,\n query: \"_sf_index__\",\n },\n ],\n [\n {\n dataType: \"String\",\n displayName: \"Name\",\n uuid: \"a\",\n value: \"Sand Ginger\",\n formattedValue: \"Sand Ginger\",\n query: \"A\",\n },\n {\n dataType: \"Number\",\n displayName: \"Amount\",\n uuid: \"b\",\n value: 3,\n formattedValue: \"3\",\n query: \"B\",\n },\n {\n dataType: \"Index\",\n displayName: \"Index\",\n uuid: \"i\",\n value: 4,\n query: \"__sf_index__\",\n },\n ],\n].map((row) => {\n return row.map((node) => {\n return {\n ...node,\n kind: NodeKind.node,\n isArtificial: node.displayName === \"Index\",\n };\n });\n});\n","import { DateTime, Interval } from \"luxon\";\nimport { NodeData, NodeSetData } from \"./data\";\n//\nexport class CalendarEvent {\n start: DateTime;\n end: DateTime;\n summary: string;\n location?: string;\n creatorEmail?: string;\n description?: string;\n\n get interval() {\n return Interval.fromDateTimes(this.start, this.end);\n }\n\n get isAllDay() {\n return this.start.hour === 0 && this.end.hour === 0;\n }\n\n get isSingleDay() {\n return this.isAllDay && this.end.day - this.start.day === 1;\n }\n\n get durationExceedsOneDay() {\n // return this.end - this.start\n return this.end.diff(this.start).milliseconds > SECONDS_IN_DAY;\n }\n\n get durationIsOrExceedsOneDay() {\n // return this.end - this.start\n return this.end.diff(this.start).milliseconds >= SECONDS_IN_DAY;\n }\n\n constructor(data: (NodeData | NodeSetData)[]) {\n data.forEach((d) => {\n // Strip the index number from the query if present,\n // converting a calendar connection to an editable table\n // connection, BE adds an index to the query for scalar bindings\n const query = d.query.replace(/[0-9]/g, \"\");\n const prop = query.toLocaleLowerCase();\n const value = (d as NodeData).value;\n switch (query) {\n case \"Start\":\n case \"End\":\n (this as any)[prop] = DateTime.fromJSDate(new Date(value as number));\n break;\n case \"CreatorEmail\":\n this.creatorEmail = value as string;\n case \"Attendees\":\n // ignore\n break;\n\n // This next check shouldn't be necessary, but currently there\n // are multiple elements in the dataset with query=\"Summary\"\n case \"Summary\":\n // console.log(\n // \"summary...\",\n // value,\n // this.summary,\n // d.kind === NodeKind.node,\n // d.kind\n // );\n // this.summary = d.kind === NodeKind.node ? value : this.summary;\n this.summary = value as string;\n break;\n default:\n (this as any)[prop] = value;\n }\n });\n }\n}\n\nexport const eventIsOver = (ev: CalendarEvent) => {\n return DateTime.now() > ev.end;\n};\n\nexport const eventIsUpcoming = (ev: CalendarEvent) => {\n return DateTime.now() < ev.start;\n};\n\nexport const SECONDS_IN_DAY = 1000 * 60 * 60 * 24;\n\nexport const eventLiesWithinRange = (\n ev: CalendarEvent,\n interval: DateTime[]\n) => {\n return intervalsOverlap([ev.start, ev.end], interval);\n};\n\nconst intervalsOverlap = (i1: DateTime[], i2: DateTime[]) => {\n return i1[0] < i2[1] && i1[1] > i2[0];\n};\n\nexport const createTracks = (events: CalendarEvent[]): CalendarEvent[][] => {\n const tracks: CalendarEvent[][] = [];\n\n for (let i = 0; i < events.length; i++) {\n const ev = events[i];\n\n // Loop through tracks, see if interval can be put onto any of them\n let eventAddedToTrack = false;\n for (let j = 0; j < tracks.length; j++) {\n // Does the current event overlap any other events in this track?\n if (!tracks[j].some((te) => te.interval.overlaps(ev.interval))) {\n eventAddedToTrack = true;\n tracks[j].push(ev);\n break;\n }\n }\n\n // create new track with this event on it\n if (!eventAddedToTrack) tracks.push([ev]);\n }\n\n return tracks;\n};\n\nexport const weekday = () => DateTime.now().weekday;\n\nexport const makeHourTodayDate = (hour: number, daysOffset = 0, minute = 0) => {\n const { month, day, year } = DateTime.now();\n\n const dt = DateTime.fromObject({\n month,\n day,\n year,\n hour: hour,\n minute,\n }).plus({ days: daysOffset });\n // console.log(\"make\", dt);\n return dt;\n};\n","import { NodeKind } from \"@/types/data\";\nimport { makeHourTodayDate, weekday } from \"@/types/calendar\";\nimport { DateTime } from \"luxon\";\n\n// NOTE: Should this actually populate the full week in the same way every time?\n// TODO: No...that's an issue..can't always just go 3 days into future, for week, e.g. if today is Saturday\n\nconst multiOffset = () => (weekday() < 5 ? 2 : -3);\n\ninterface EventPlaceholderNode {\n query: string;\n value: string | Date;\n dataType?: string;\n}\n\nexport const CalendarEventPlaceholderData = () => [\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(15, 0, 30).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(17, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Event Title Here\",\n formattedValue: \"Event Title Here\",\n },\n {\n query: \"Location\",\n value: \"Conference Room\",\n formattedValue: \"Conference Room\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n].map(prepareCalendarNode);\n\nexport const CalendarAgendaPlaceholderData = () => [\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(15, 0, 30).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(17, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Weekly Leadership Meeting\",\n formattedValue: \"Weekly Leadership Meeting\",\n },\n {\n query: \"Location\",\n value: \"Conference Room\",\n formattedValue: \"Conference Room\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n // Event 2\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(10, 0, 15).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(12, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Product Roadmap Discussion\",\n formattedValue: \"Product Roadmap Discussion\",\n },\n {\n query: \"Location\",\n value: \"Conference Room\",\n formattedValue: \"Conference Room\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n // Event 3 (all day)\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(0, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(0, 1, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Jeremy Click's Birthday (All day)\",\n formattedValue: \"Jeremy Click's Birthday (All day)\",\n },\n {\n query: \"Location\",\n value: \"Conference Room\",\n formattedValue: \"Conference Room\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n // Event 4 (multi day)\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(0, multiOffset(), 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(0, multiOffset() + 2, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Conference (Multi day)\",\n formattedValue: \"Conference (Multi day)\",\n },\n {\n query: \"Location\",\n value: \"Conference Room\",\n formattedValue: \"Conference Room\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n]\n // Add dummy data 3 weeks into future (needed for CalendarWeek):\n .flatMap((event: EventPlaceholderNode[]) => {\n const futureEvents = [event];\n for (let i = 1; i <= 3; i++) {\n const newEvent = event.map((node) => {\n if ([\"Start\", \"End\"].includes(node.query)) {\n return {\n ...node,\n value: DateTime.fromJSDate(node.value as Date)\n .plus({ days: 6 * i }) // Using 6 instead of 7 offsets all events by one day, each week into future, showing off the widget better\n .toJSDate(),\n };\n } else {\n return node;\n }\n });\n futureEvents.push(newEvent);\n }\n\n return futureEvents;\n })\n .map(prepareCalendarNode);\n\nexport const CalendarPlaceholderData = () => [\n // Event 1\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(15, 0, 30).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(17, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Homecoming Planning Meeting\",\n formattedValue: \"Homecoming Planning Meeting\",\n },\n {\n query: \"Location\",\n value: \"Room 123\",\n formattedValue: \"Room 123\",\n },\n {\n query: \"CreatorEmail\",\n value: \"user@email.com\",\n formattedValue: \"user@email.com\",\n },\n ],\n // Event 2\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(10, 0, 15).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(12, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Guidance Counselor Huddle\",\n formattedValue: \"Guidance Counselor Huddle\",\n },\n {\n query: \"Location\",\n value: \"Room 123\",\n formattedValue: \"Room 123\",\n },\n ],\n // Event 3 (all day)\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(0, 0, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(0, 1, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"New Student Orientation (All day)\",\n formattedValue: \"New Student Orientation (All day)\",\n },\n {\n query: \"Location\",\n value: \"Auditorium\",\n formattedValue: \"Auditorium\",\n },\n ],\n // Event 4 (multi day)\n [\n {\n query: \"Description\",\n value: \"This is an event description.\",\n formattedValue: \"This is an event description.\",\n },\n {\n query: \"Start\",\n value: makeHourTodayDate(0, multiOffset(), 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"End\",\n value: makeHourTodayDate(0, multiOffset() + 2, 0).toJSDate(),\n dataType: \"DateTime\",\n },\n {\n query: \"Summary\",\n value: \"Test Event 4 (Multi day)\",\n formattedValue: \"Test Event 4 (Multi day)\",\n },\n {\n query: \"Location\",\n value: \"Room 100\",\n formattedValue: \"Room 100\",\n },\n ],\n].map(prepareCalendarNode);\n\nfunction prepareCalendarNode(row: any[]) {\n return row.map((node) => {\n return {\n ...node,\n uuid: node.query,\n displayName: node.query,\n kind: NodeKind.node,\n dataType: node.dataType || \"String\",\n };\n });\n}\n","import { BarGraphPlaceholderData } from \"@/components/widgets/BarGraph/BarGraphPlaceholderData\";\nimport { ColumnGraphPlaceholderData } from \"@/components/widgets/ColumnGraph/ColumnGraphPlaceholderData\";\nimport { LineGraphPlaceholderData } from \"@/components/widgets/LineGraph/LineGraphPlaceholderData\";\nimport { PieGraphPlaceholderData } from \"@/components/widgets/PieGraph/PieGraphPlaceholderData\";\nimport { StackedGraphPlaceholderData } from \"@/components/widgets/StackedGraph/StackedGraphPlaceholderData\";\nimport {\n CalendarPlaceholderData,\n CalendarEventPlaceholderData,\n CalendarAgendaPlaceholderData,\n} from \"@/components/widgets/CalendarPlaceholderData\";\n\n/**\n * Returns placeholder data for supplied widget type (if available).\n * @param widgetType Type of widget\n * @returns Placeholder data for supplied type, or undefined if there is no fake data for type.\n */\nexport const getPlaceholderData = (widgetType: string): any => {\n switch (widgetType) {\n case \"BarGraph\":\n return BarGraphPlaceholderData;\n case \"ColumnGraph\":\n return ColumnGraphPlaceholderData;\n case \"LineGraph\":\n return LineGraphPlaceholderData;\n case \"PieGraph\":\n return PieGraphPlaceholderData;\n case \"StackedGraph\":\n return StackedGraphPlaceholderData;\n case \"CalendarEvent\":\n return CalendarEventPlaceholderData();\n case \"CalendarAgenda\":\n case \"CalendarRoomSchedule\":\n case \"CalendarWeek\":\n return CalendarAgendaPlaceholderData();\n case \"CalendarDay\":\n return CalendarPlaceholderData();\n default:\n return undefined;\n }\n};\n","import { DataBinding, NodeData, NodeSetData } from \"@/types/data\";\nimport { BASE_PARENT_ID } from \"@/constants\";\nimport { getActiveConditionId, getActiveWidget } from \"@/util/conditionsUtils\";\nimport { getPlaceholderData } from \"@/placeholderData\";\nimport { TextContent } from \"@/text\";\nimport { populateDataToken } from \"@/text/binding\";\nimport { TextOptions } from \"@/components/widgets/Text/TextOptions\";\nimport { RepeaterOptions } from \"@/components/widgets/Repeater/RepeaterOptions\";\nimport { WidgetDataCache } from \"./appData\";\nimport { cleanColorValue, removeReactivity } from \"@/utils\";\nimport { ConditionsData } from \"@/types/rendererData\";\nimport { AppMode, ResolvedWidgetData, SavedAsset } from \"@/types\";\nimport { WidgetWithConditions } from \"@/components/widgets/Widget\";\n\nexport const widgetData = (\n appMode: AppMode,\n widgets: Record,\n assets: SavedAsset[],\n dataBindings: DataBinding[],\n data: WidgetDataCache,\n conditions: ConditionsData | Record | undefined\n): ResolvedWidgetData => {\n /**\n * The `result` variable will contain dynamic properties for widgets.\n *\n * It's essentially a lookup table. They key is the widgetId, the value\n * is an array of objects which will be merged into the widget data before\n * it's rendered.\n *\n * We use an array for the value so that the same widget can have different\n * values in a repeater. Each element in the array corresponds to a 'row' in\n * the repeater's dataset.\n *\n * Widgets not in a repeater will only have a single object in the array.\n *\n * ```\n * {\n * // Below is a widget in a repeater with dynamic backgroundColor\n *\n * 'abc123': [\n * { backgroundColor: 'red' },\n * { backgroundColor: 'green' },\n * { backgroundColor: 'blue' }\n * ],\n *\n * // Below is a widget in a repeater with dynamic bg img and border colors\n *\n * 'def456': [\n * { backgroundImage: 'https://...', borderColor: \"hsl(0,0,0)\" },\n * { backgroundImage: 'https://...', borderColor: \"hsl(255,0,80)\" },\n * { backgroundImage: 'https://...', borderColor: \"hsl(0,0,0)\" },\n * ]\n *\n * // Below is a widget NOT in a repeater with dynamic background.\n * // Notice it only has a single value in the array.\n *\n * 'ghi789': [\n * { backgroundColor: 'red' }\n * ]\n * }\n * ```\n */\n const result: ResolvedWidgetData = {};\n for (const widgetId in widgets) {\n result[widgetId] = [];\n }\n\n /**\n * We'll need to loop through databindings multiple times\n * because, when we apply scalar bindings and asset bindings\n * we'll need to know ahead of time how \"big\" each repeater\n * is, so we can apply the correct data to each widget.\n */\n const repeaterDataSets: Map = new Map();\n\n /**\n * Now we are going to loop through all the databindings and populate the\n * result object with the data for each widget.\n */\n (dataBindings || []).forEach((db) => {\n /**\n * -------------------------------------------------------------\n * STEP 1: Skip if binding points to a widget that doesn't exist.\n * -------------------------------------------------------------\n * It's possible we have data bindings that refer to widgets that\n * no longer exist in state. We already know that at this point\n * `result` has a key for every widget in state.\n */\n if (!result[db.widgetId]) {\n // console.log(\n // `${db.bindingType} binding references non-existent widget ${db.widgetId}`,\n // JSON.stringify(db, null, 2)\n // );\n return;\n }\n\n // // * Skip data (or asset) binding if not associated with active condition for widget.\n // // * Only Asset, Scalar, DataSetNode bindings are given conditionIds.\n // // * Other bindings (DataSet, DataSetParent), at the moment, should NOT be scoped to a condition.\n // if (db.conditionUuid) {\n // const conditionUuid =\n // state.appMode === \"render\"\n // ? getRenderedConditionId(db.widgetId, conditions)\n // : useConditionGroupsStore().getActiveConditionId(db.widgetId);\n // if (db.conditionUuid !== conditionUuid) {\n // return;\n // }\n // }\n\n /**\n * -------------------------------------------------------------\n * STEP 2: Gather dynamic props for all repeater children\n * -------------------------------------------------------------\n */\n if (db.bindingType === \"DataSetNode\") {\n // If DataState is undefined, skip this binding.\n if (!data) {\n return;\n }\n\n const key = db.parentWidgetId || \"\";\n const prop = \"data\";\n if (!data[key] || !data[key][prop]) {\n return;\n }\n const nodeSet = data[key][prop] as NodeSetData;\n const rows = (nodeSet.children ?? []) as NodeData[][];\n const propsArray: Record[] = result[db.widgetId];\n\n rows.forEach((row, index) => {\n let value: any;\n const groupUuid = row[0].groupUuid;\n\n const childWidget = getActiveWidget(\n widgets[db.widgetId],\n conditions,\n groupUuid\n );\n\n // Ignore conditional DataSetNode bindings that are not active\n const isBindingActive =\n db.conditionUuid ===\n getActiveConditionId(db.widgetId, conditions, groupUuid);\n if (!isBindingActive && db.conditionUuid) return;\n\n const isTextContentBinding =\n childWidget.type === \"Text\" && db.property === \"content\";\n\n if (isTextContentBinding) {\n let content = (childWidget as TextOptions).content;\n\n // Check if we've already stored replaced content into this\n // local widget data, if so we want to use that instaed.\n if (propsArray[index] && propsArray[index][\"content\"]) {\n content = propsArray[index][\"content\"] as TextContent;\n }\n\n value = populateDataToken(content, db, row);\n } else {\n value = row.find((c) => c.uuid === db.dataUuid)?.value;\n }\n\n // We're going to store _groupUuid so that we can use it later\n // When applying conditional Scalar and Asset bindings.\n\n const handleNormally = () => {\n if (propsArray.length > 0 && index < propsArray.length) {\n propsArray[index][db.property] = value;\n propsArray[index][\"_groupUuid\"] = groupUuid;\n } else {\n propsArray.push({\n [db.property]: value,\n _groupUuid: groupUuid,\n });\n }\n };\n\n const handleSpecially = () => {\n // What if we get index 2 first?\n // Then we need to fill up 0 and 1 with nothing, and insert at index 2.\n if (index < propsArray.length) {\n propsArray[index][db.property] = value;\n propsArray[index][\"_groupUuid\"] = groupUuid;\n } else {\n for (let i = 0; i < index - propsArray.length; i++) {\n propsArray.push({});\n }\n propsArray.push({\n [db.property]: value,\n _groupUuid: groupUuid,\n });\n }\n };\n\n if (appMode === \"render\") {\n if (isTextContentBinding) {\n handleNormally();\n } else {\n handleSpecially();\n }\n } else {\n handleNormally();\n }\n });\n }\n\n /**\n * -------------------------------------------------------------\n * STEP 3: Bind datasets to Repeaters, Calendars, Graphs, etc\n * -------------------------------------------------------------\n */\n if (data && [\"DataSetParent\", \"DataSet\"].includes(db.bindingType)) {\n const key = db.widgetId;\n const prop = db.property || \"data\";\n\n // console.log(\"datasetparent...\", data);\n\n if (!data[key] || !data[key][prop]) {\n // console.log(\n // `${db.bindingType} binding references non-existent dataset`,\n // JSON.stringify(db, null, 2)\n // );\n return;\n }\n\n const d = data[key][prop] as NodeSetData;\n const dataSet = d.children ?? [];\n\n // Store the data set so we can use it for Scalar and Asset bindings later.\n repeaterDataSets.set(db.widgetId, dataSet as NodeData[][]);\n\n if (result[db.widgetId].length === 0) {\n // console.log(\n // \"Repeater didn't have any items in dynamicProps array, adding it\"\n // );\n result[db.widgetId].push({\n [db.property]: dataSet,\n importedRecordCount: d.importedRecordCount,\n });\n } else {\n // console.log(\"Repeater had items in dynamicProps array, merging it\");\n // Because Calendars, Graphs, and Repeaters are never repeated\n // the dynamicProps array will only ever have a single object\n // So let's merge the `data` property into that object.\n result[db.widgetId][0] = Object.assign({}, result[db.widgetId][0], {\n [db.property]: dataSet,\n importedRecordCount: d.importedRecordCount,\n });\n }\n }\n });\n\n /**\n * -------------------------------------------------------------\n * STEP 4: Bind Scalar and Asset bindings\n * -------------------------------------------------------------\n * Now that we already know the sizes of every databound repeater\n * in the app, we can loop through the databindings again and\n * populate Asset and Scalar bindings.\n */\n (dataBindings || []).forEach((db) => {\n const widget = widgets[db.widgetId];\n\n if (!widget) {\n // console.log(\n // `${db.bindingType} binding references non-existent widget ${db.widgetId}`,\n // JSON.stringify(db, null, 2)\n // );\n return;\n }\n\n const dynamicProps = result[db.widgetId];\n\n /**\n * We need to determine how many times the widget associated with\n * this databinding is going to be repeated, so we can push\n * ensure Scalar and Asset bindings are placed into the\n * dynamicProps array the correct number of times.\n *\n * If the widget is inside a data bound repeater, it will be repeated\n * based on the number of rows in the repeaters dataset.\n *\n * If it is inside a repeater that is NOT data bound, it will be\n * repeated based on the size (rows * columns).\n *\n * If it is not inside a repeater at all, it will not be repeated,\n * just rendered once.\n */\n\n let repeatCount = 1; // Default to 1, meaning no repeater.\n let repeaterData: NodeData[][] | undefined = undefined;\n\n if (widget.parentId !== BASE_PARENT_ID) {\n // We should check parent and grandparent because\n // the target widget could be a child of a group\n // inside a repeater.\n let ancestor = widgets[widget.parentId];\n if (ancestor && ancestor.parentId !== BASE_PARENT_ID) {\n ancestor = widgets[ancestor.parentId];\n }\n\n // Ensure we have the active condition for the ancestor.\n // because number of rows and columns could be different\n // under each condition.\n const ancestorWidget = getActiveWidget(ancestor, conditions);\n\n if (ancestorWidget.type === \"Repeater\") {\n const repeater = ancestorWidget as unknown as RepeaterOptions;\n repeaterData = repeaterDataSets.get(repeater.wid);\n repeatCount = repeaterData?.length ?? repeater.rows * repeater.columns;\n }\n }\n\n if (db.bindingType === \"Scalar\") {\n for (let i = 0; i < repeatCount; i++) {\n // Represents the dynamic props for this widget at this repeater index.\n const propsObject: Record = dynamicProps[i] ?? {};\n\n let isBindingActive = true;\n\n /**\n * A binding is only active if the conditionUuid matches the\n * active condition for the widget.\n */\n if (db.conditionUuid) {\n const groupUuid = repeaterData?.[i]?.[0]?.groupUuid;\n isBindingActive =\n db.conditionUuid ===\n getActiveConditionId(db.widgetId, conditions, groupUuid);\n }\n\n if (isBindingActive) {\n let value = null;\n // Use property names \"content_{id}\" for these properties\n if (widget.type === \"Text\" && db.property.startsWith(\"content\")) {\n const content = (getActiveWidget(widget, conditions) as TextOptions)\n .content;\n const contentData = data?.[db.widgetId]?.[db.property] as NodeData;\n value = populateDataToken(content, db, contentData);\n } else {\n value = (data?.[db.widgetId]?.[db.property] as NodeData)?.value;\n }\n // Label the populated text content as \"content\" so text component can pick it up:\n const property = db.property.startsWith(\"content\")\n ? \"content\"\n : db.property;\n propsObject[property] = value;\n }\n\n if (i < dynamicProps.length - 1) {\n dynamicProps[i] = propsObject;\n } else {\n dynamicProps.push(propsObject);\n }\n }\n }\n\n if (db.bindingType === \"Asset\") {\n for (let i = 0; i < repeatCount; i++) {\n // Represents the dynamic props for this widget at this repeater index.\n const propsObject: Record = dynamicProps[i] ?? {};\n\n let isBindingActive = true;\n\n /**\n * A binding is only active if the conditionUuid matches the\n * active condition for the widget.\n */\n if (db.conditionUuid) {\n const groupUuid = repeaterData?.[i]?.[0]?.groupUuid;\n isBindingActive =\n db.conditionUuid ===\n getActiveConditionId(db.widgetId, conditions, groupUuid);\n }\n\n if (isBindingActive) {\n const value = assets.find((a) => a.uuid === db.dataUuid)?.url;\n propsObject[db.property] = value;\n }\n\n if (i < dynamicProps.length - 1) {\n dynamicProps[i] = propsObject;\n } else {\n dynamicProps.push(propsObject);\n }\n }\n }\n });\n\n /**\n * -------------------------------------------------------------\n * STEP 5: Bind hover background images and placeholder data\n * -------------------------------------------------------------\n * Now that all data-bound properties for widgets have been\n * populated, we can loop through the `results` object and\n * - Populate any hover background images\n * - Populate placeholder data for Graphs and Calendars\n * - Clean up any color values\n */\n for (const widgetId in result) {\n const widget = widgets[widgetId];\n const dynamicProps = result[widgetId];\n\n if (!widget.type) {\n delete widgets[widgetId];\n continue;\n }\n\n if (widget.type?.includes(\"Graph\") || widget.type?.includes(\"Calendar\")) {\n const placeholderData = removeReactivity(getPlaceholderData(widget.type));\n if (\n dynamicProps.length > 0 &&\n typeof dynamicProps[0][\"data\"] === \"undefined\"\n ) {\n (dynamicProps[0] as any)[\"data\"] = placeholderData;\n (dynamicProps[0] as any)[\"_placeholderData\"] = true;\n } else if (dynamicProps.length === 0) {\n dynamicProps.push({\n data: placeholderData,\n _placeholderData: true,\n });\n }\n }\n\n dynamicProps.forEach((dp) => {\n Object.keys(dp).forEach((key) => {\n if (key.toLowerCase().includes(\"color\")) {\n dp[key] = cleanColorValue(dp[key]).value;\n }\n });\n });\n }\n\n return result;\n};\n","import { RepeaterOptions } from \"@/components/widgets/Repeater/RepeaterOptions\";\nimport { Widget } from \"@/components/widgets/Widget\";\nimport {\n ImageAssetInfo,\n FontAssetInfo,\n VideoAssetInfo,\n} from \"@/types/bundleTypes\";\nimport { AppState } from \"@/stores/AppState\";\n\nimport isEqual from \"lodash.isequal\";\nimport uniqWith from \"lodash.uniqwith\";\nimport pick from \"lodash.pick\";\nimport { GroupOptions } from \"@/components/widgets/Group/GroupOptions\";\nimport { NodeData } from \"@/types/data\";\nimport { getActiveWidget } from \"@/util/conditionsUtils\";\nimport { ConditionsData } from \"@/types/rendererData\";\n\nexport const gatherAssets = (\n state: AppState,\n boundWidgets: Widget[],\n widgetData: Record,\n conditions: ConditionsData | Record | undefined\n) => {\n type WidgetWithRenderInfo = Widget & { shouldPreload: boolean };\n\n const fonts: FontAssetInfo[] = [];\n const images: ImageAssetInfo[] = [];\n const videos: VideoAssetInfo[] = [];\n\n // Gather Font Properties\n const tryAddFontValue = (\n fonts: Map,\n key: string,\n value: string\n ) => {\n let prop = key;\n let group = \"default\";\n const parts = key.split(\"_\");\n if (parts.length === 2) {\n // This is a prop like calendarTitle_fontWeight\n group = parts[0];\n prop = parts[1];\n }\n if ([\"fontFamily\", \"fontWeight\", \"fontStyle\"].includes(prop)) {\n const info = (fonts.get(group) || {\n source: \"Google\",\n }) as FontAssetInfo;\n\n // Convert from `fontFamily` to `family`\n const newPropName = prop.toLowerCase().substring(4);\n (info as any)[newPropName] = value;\n fonts.set(group, info);\n }\n };\n\n const getAssetsForWidget = (wg: Widget, shouldPreload = true) => {\n const fontGroups: Map = new Map();\n for (const prop in wg) {\n const value = (wg as any)[prop];\n if (typeof value !== \"undefined\") {\n // Add font properties\n tryAddFontValue(fontGroups, prop, value);\n\n // Gather Image Properties\n if (\n (wg.type === \"Image\" && prop === \"url\") ||\n (prop === \"backgroundImageUrl\" && value !== null)\n ) {\n images.push({\n url: value,\n height: Math.ceil(wg.h * wg.scaleY),\n width: Math.ceil(wg.w * wg.scaleX),\n mode: (wg as any).backgroundSize === \"contain\" ? \"max\" : \"crop\",\n shouldPreload: shouldPreload,\n });\n }\n\n // Gather Video Properties\n if (wg.type === \"Video\" && prop === \"url\") {\n videos.push({\n url: value,\n height: Math.ceil(wg.h * wg.scaleY),\n width: Math.ceil(wg.w * wg.scaleX),\n shouldPreload: shouldPreload,\n });\n }\n }\n }\n\n Array.from(fontGroups.values()).forEach((info) => {\n fonts.push(info);\n });\n };\n\n const getDataBoundChildren = (\n repeater: RepeaterOptions\n ): WidgetWithRenderInfo[] => {\n const pageSize = repeater.rows * repeater.columns;\n\n const widgetIds: string[] = [];\n\n // Loop through all direct children of repeater and add id\n (state.parents[repeater.wid] || []).forEach((repeaterChildWidgetId) => {\n widgetIds.push(repeaterChildWidgetId);\n\n // If child is a \"Group\", loop through all group children and add id\n if (state.widgets[repeaterChildWidgetId]?.type === \"Group\") {\n (state.parents[repeaterChildWidgetId] || []).forEach(\n (groupChildWidgetId) => {\n widgetIds.push(groupChildWidgetId);\n }\n );\n }\n });\n\n return widgetIds.flatMap((wid) => {\n /**\n * At this point in the code, `widgetData[wid]` may contain\n * and array of dynamic props for each widget.\n *\n * Conditions is already applied to this array so it will only\n * have the props for the active condition.\n *\n * The size of the array will be equal to\n * - The number of cells of the repeater (if not bound to data)\n * - The number of rows in the data set (if bound to data)\n */\n const dynamicProps = widgetData[wid];\n if (dynamicProps.length === 0) {\n dynamicProps.push({});\n }\n\n return dynamicProps.map((props: any, index: number) => {\n /**\n * Here is where we merge the dynamic props with the props the user\n * defined manually in app builder.\n *\n * Because we are grabbing them manually here, we'll need to apply\n * conditions.\n *\n * But when we apply conditions to a repeater child, we'll also need\n * to pass along the `groupUuid` associated with the current row of\n * the data set because it may be relevent for getting the correct\n * condition see `recordCondRefToRecordCondition` for more info.\n */\n const repeaterData = (repeater as any).data as NodeData[][];\n\n /**\n * This `groupUuid` is basically the unique row id for the current row\n * of the repeaters data set (if it has one).\n *\n * We are using the `index` of the dynamic props array\n * because it corresponds to the data row index.\n */\n const groupUuid: string | undefined =\n repeaterData?.[index]?.[0]?.groupUuid;\n\n const nonDynamicProps = getActiveWidget(\n state.widgets[wid],\n conditions,\n groupUuid\n );\n\n return Object.assign({}, nonDynamicProps, props, {\n shouldPreload: index < pageSize,\n });\n });\n });\n };\n\n /**\n * Gather grouped widgets that are not inside a repeater\n */\n const getGroupChildren = (group: GroupOptions): WidgetWithRenderInfo[] => {\n return (state.parents[group.wid] || []).flatMap((wid) => {\n const dynamicProps = widgetData[wid];\n if (dynamicProps.length === 0) {\n dynamicProps.push({});\n }\n\n return dynamicProps.map((props: any) => {\n return Object.assign({}, state.widgets[wid], props, {\n shouldPreload: true,\n });\n });\n });\n };\n\n boundWidgets.forEach((wg: Widget) => {\n if (wg.type === \"Repeater\") {\n getDataBoundChildren(wg as any as RepeaterOptions).forEach((child) =>\n getAssetsForWidget(child, child.shouldPreload)\n );\n } else if (wg.type === \"Group\") {\n getGroupChildren(wg as GroupOptions).forEach((child) =>\n getAssetsForWidget(child, true)\n );\n } else {\n getAssetsForWidget(wg);\n }\n });\n\n // Add intro and fallback images to assets.images so that backend can add them to bundle:\n const assets = state.assets || [];\n const introImage = assets.find((a) => a.functionType === \"IntroImage\");\n const fallbackImage = assets.find((a) => a.functionType === \"FallbackImage\");\n if (introImage) {\n images.push({\n ...introImage,\n shouldPreload: true,\n url: introImage?.url || \"\",\n });\n }\n if (fallbackImage) {\n images.push({\n ...fallbackImage,\n shouldPreload: true,\n url: fallbackImage?.url || \"\",\n });\n }\n\n return {\n fonts: uniqWith(fonts, isEqual),\n images: uniqWith(images, (a, b) => {\n return isEqual(\n pick(a, [\"url\", \"height\", \"width\", \"mode\"]),\n pick(b, [\"url\", \"height\", \"width\", \"mode\"])\n );\n }),\n videos: uniqWith(videos, (a, b) => {\n return isEqual(\n pick(a, [\"url\", \"height\", \"width\"]),\n pick(b, [\"url\", \"height\", \"width\"])\n );\n }),\n };\n};\n","import { Point, TempBackgroundInfo, WidgetType } from \"@/types\";\nimport { DataBinding, DataType, NodeData } from \"@/types/data\";\nimport { defineStore } from \"pinia\";\nimport { EventBus } from \"@/eventbus\";\nimport { makeId } from \"@/utils\";\nimport { BASE_PARENT_ID } from \"@/constants\";\nimport { useConditionGroupsStore } from \"./conditionGroups\";\nimport { useAppEditorStore } from \"./appEditor\";\nimport { useConnectionEditorStore } from \"./connectionEditor\";\nimport { useConnectionDataStore } from \"./connectionData\";\n\nexport interface HoverTarget {\n widgetId: string;\n widgetType: WidgetType;\n // Allow setting context to determine this, rather than computing from list of components here\n isPhotoDropContainer?: boolean;\n}\n\nexport interface DataWidgetInfo extends NodeData {\n connectionUuid: string;\n nodeSetUuid?: string;\n children?: NodeData[];\n}\n\nexport interface DraggingInfo {\n type: \"Asset\" | \"Node\";\n parentWidgetId?: string;\n dropPoint?: Point;\n\n // For \"Nodes\"\n dataUuid?: string;\n connectionUuid?: string;\n dataParentUuid?: string;\n query?: string;\n dataType?: DataType;\n formattedValue?: string;\n children?: NodeData[];\n isArtificial?: boolean;\n\n // For \"Assets\"\n url?: string;\n width?: number;\n height?: number;\n mimeType?: string;\n}\n\ninterface DragGhostStyle {\n opacity: number;\n width: number;\n height: number;\n top: number;\n left: number;\n}\n\nexport interface DragDropState {\n hoverTarget: HoverTarget | null;\n draggingInfo: DraggingInfo | null;\n usagePromptIsOpen: boolean;\n isReplacingData: boolean; // Can probably be computed...\n isHandlingDrop: boolean;\n ghostStyle: DragGhostStyle;\n}\n\nexport const useDragDropStore = defineStore(\"dragDrop\", {\n state: (): DragDropState => {\n return {\n hoverTarget: null,\n draggingInfo: null,\n usagePromptIsOpen: false,\n isReplacingData: false,\n isHandlingDrop: false,\n ghostStyle: {\n opacity: 0,\n width: 400,\n height: 400,\n top: 0,\n left: 0,\n },\n };\n },\n getters: {\n isDraggingItem(): boolean {\n return this.draggingInfo !== null;\n },\n\n isDraggingNode(): boolean {\n return this.draggingInfo?.type === \"Node\";\n },\n\n isDraggingImage(): boolean {\n return (\n this.draggingInfo?.type === \"Asset\" ||\n [\"ImageUrl\", \"ImageUpload\"].includes(this.draggingInfo?.dataType || \"\")\n );\n },\n\n hoveredTextWidgetId(): string | null {\n if (this.hoverTarget === null) return null;\n if (this.hoverTarget.widgetType !== \"Text\") return null;\n return this.hoverTarget.widgetId;\n },\n\n hoveredPhotoDropWidgetId(): string | null {\n if (!this.hoverTarget?.isPhotoDropContainer) return null;\n return this.hoverTarget.widgetId;\n },\n\n draggedPhotoNodeUuid(): string | null {\n if (!this.isDraggingImage || !this.isDraggingNode) return null;\n return this.draggingInfo?.dataUuid || null;\n },\n\n tempBackgroundInfo(): TempBackgroundInfo | null {\n if (!this.isDraggingImage) return null;\n if (!this.hoverTarget?.isPhotoDropContainer) return null;\n\n const appEditor = useAppEditorStore();\n return {\n url: this.draggingInfo?.url,\n wid: this.hoverTarget?.widgetId || \"\",\n w: this.draggingInfo?.width,\n h: this.draggingInfo?.height,\n isDynamic:\n !!this.draggedPhotoNodeUuid && !appEditor.isBaseEditingContext,\n };\n },\n\n /**\n * There are 3 cases under which we need to hide the \"use on its own\" option:\n * - The node is from a calendar dataset\n * - The node represents a \"data index\"\n * - The node is from a row that has been moderated out of the dataset\n */\n\n preventScalarUsage(): boolean {\n const connectionDataStore = useConnectionDataStore();\n const connectionEditorStore = useConnectionEditorStore();\n\n const isFromCalendar =\n connectionEditorStore.connection?.schemaType === \"Calendar\";\n if (isFromCalendar) return true;\n\n const isArtificialNode = this.draggingInfo?.isArtificial === true;\n if (isArtificialNode) return true;\n\n if (connectionEditorStore.moderationMode === null) return false;\n\n const isFromModeratedOutRow =\n connectionEditorStore.moderationMode === \"Approval\"\n ? !connectionDataStore.moderationRow?.isSelected\n : connectionDataStore.moderationRow?.isSelected;\n\n return isFromModeratedOutRow;\n },\n },\n\n actions: {\n async handleNodeDrop() {\n const appEditor = useAppEditorStore();\n const editingContext = appEditor.editingContext;\n const dataBindings = appEditor.dataBindings;\n\n this.isHandlingDrop = true;\n\n const hoveredRepeaterWid =\n editingContext.parentId === BASE_PARENT_ID\n ? null\n : editingContext.parentId;\n const hoveredRepeaterBinding = dataBindings.find(\n (db: DataBinding) =>\n db.bindingType === \"DataSetParent\" &&\n db.widgetId === hoveredRepeaterWid\n );\n\n this.isReplacingData = false;\n\n /**\n * Show usagePrompt if:\n * (1) node is dropped onto canvas from a collection, OR\n * (2) node is dropped into repeater from a foreign collection.\n */\n\n const connectionDataStore = useConnectionDataStore();\n\n let shouldOpenUsagePrompt = !connectionDataStore.connectionIsTree;\n if (hoveredRepeaterWid && !connectionDataStore.connectionIsTree) {\n shouldOpenUsagePrompt =\n typeof hoveredRepeaterBinding?.dataConnectionUuid !== \"undefined\" &&\n hoveredRepeaterBinding?.dataConnectionUuid !==\n this.draggingInfo?.connectionUuid;\n if (shouldOpenUsagePrompt) {\n this.isReplacingData = true;\n }\n }\n if (this.hoveredTextWidgetId) {\n shouldOpenUsagePrompt = false;\n }\n\n if (shouldOpenUsagePrompt) {\n this.usagePromptIsOpen = true;\n return;\n }\n\n this.createDynamicWidgetAction(\n hoveredRepeaterWid,\n hoveredRepeaterBinding\n );\n },\n\n async createDynamicWidgetAction(\n hoveredRepeaterWid: string | null,\n hoveredRepeaterBinding: DataBinding | undefined\n ) {\n EventBus.emit(\"AWAITING_SERVER\", true);\n\n const creatingToken = !!this.hoveredTextWidgetId;\n\n const appEditor = useAppEditorStore();\n const conditionsStore = useConditionGroupsStore();\n\n if (creatingToken) {\n const hoveredTextWidgetConditionId =\n conditionsStore.getActiveConditionId(this.hoveredTextWidgetId || \"\");\n EventBus.emit(\"DATA_TOKEN_DROPPED\", {\n widgetId: this.hoveredTextWidgetId as string,\n conditionId: hoveredTextWidgetConditionId,\n draggingInfo: this.draggingInfo as DraggingInfo,\n parentWidgetId: hoveredRepeaterWid || BASE_PARENT_ID,\n });\n return;\n }\n\n try {\n const hasDataSetBinding = !!hoveredRepeaterBinding;\n // Binding should be scalar if node is from a foreign dataset;\n // that is, if the repeater already has a dataset bound, and the node is from a different one.\n let isScalar =\n hasDataSetBinding &&\n hoveredRepeaterBinding?.dataConnectionUuid !==\n this.draggingInfo?.connectionUuid;\n\n if (!creatingToken) {\n isScalar = useConnectionDataStore().connectionIsTree;\n }\n\n const hoveredPhotoDropWidgetConditionId =\n conditionsStore.getActiveConditionId(\n this.hoveredPhotoDropWidgetId || \"\"\n );\n\n await appEditor.createDynamicWidget(\n {\n isNewRepeater: false,\n isScalar: !hoveredRepeaterWid ? true : isScalar,\n hoveredPhotoDropWidgetId: this.hoveredPhotoDropWidgetId as string,\n hoveredPhotoDropWidgetConditionId:\n hoveredPhotoDropWidgetConditionId,\n draggingInfo: this.draggingInfo,\n widgetId: makeId(),\n parentWidgetId: hoveredRepeaterWid,\n },\n null,\n hoveredRepeaterBinding ? null : hoveredRepeaterWid\n );\n\n // Trigger data refresh\n await appEditor.updateApp();\n this.draggingInfo = null;\n this.hoverTarget = null;\n } finally {\n EventBus.emit(\"AWAITING_SERVER\", false);\n this.isHandlingDrop = false;\n }\n },\n\n updateGhostStyle(style: Partial) {\n this.ghostStyle = {\n ...this.ghostStyle,\n ...style,\n };\n },\n },\n});\n","import { useDragDropStore } from \"./dragDrop\";\nimport { DataBinding, NodeData, NodeSetData } from \"@/types/data\";\nimport { useConnectionEditorStore } from \"./connectionEditor\";\nimport { useConnectionDataStore } from \"./connectionData\";\nimport { WidgetDataCache } from \"./appData\";\nimport { ResolvedWidgetData } from \"@/types\";\nimport { WidgetWithConditions } from \"@/components/widgets/Widget\";\nimport { getActiveWidget } from \"@/util/conditionsUtils\";\nimport { useConditionGroupsStore } from \"./conditionGroups\";\nimport { RepeaterOptions } from \"@/components/widgets/Repeater/RepeaterOptions\";\n\nexport const applyDynamicImages = (\n widgets: Record,\n dataBindings: DataBinding[],\n data: WidgetDataCache,\n result: ResolvedWidgetData\n): ResolvedWidgetData => {\n const dragDropStore = useDragDropStore();\n const connectionDataStore = useConnectionDataStore();\n const connectionEditorStore = useConnectionEditorStore();\n const conditionGroupsStore = useConditionGroupsStore();\n const conditions = conditionGroupsStore.activeWidgetConditionsMap;\n\n const tempBackgroundInfo = dragDropStore.tempBackgroundInfo;\n const draggedPhotoNodeUuid = dragDropStore.draggedPhotoNodeUuid;\n\n for (const widgetId in result) {\n const widget = widgets[widgetId];\n\n // If tempBackgroundInfo matches this widgetId, override backgroundImageUrl property\n if (tempBackgroundInfo?.wid === widgetId && tempBackgroundInfo?.url) {\n result[widgetId] = (result[widgetId] ?? []).filter(\n (x) => !(\"backgroundImageUrl\" in x) && !(\"url\" in x)\n );\n\n // Pipe in the dynamic images as temporary previews, before photo is dropped\n // We refer to the bound dataset, when hovering over a photo or shape inside of a Repeater.\n // But for dynamic NON-repeating images, we can treat them as if \"isDynamic\" is false.\n // Because we can just pipe in tempBackgroundInfo.url\n // console.log(\"tempbg..\", tempBackgroundInfo, widgetId, widget?.type);\n if (tempBackgroundInfo?.isDynamic) {\n const db = dataBindings.find(\n (db) =>\n db.widgetId === widget.parentId && db.bindingType === \"DataSet\"\n );\n\n const isDraggingForeignNode =\n dragDropStore.isDraggingImage &&\n dragDropStore.isDraggingNode &&\n connectionEditorStore.connection?.uuid !== db?.dataConnectionUuid;\n\n const prop = db?.property || \"\";\n\n let rows: NodeData[][] = [];\n\n if (\n db &&\n data &&\n data[db.widgetId] &&\n data[db.widgetId][prop] &&\n !isDraggingForeignNode\n ) {\n rows = ((data?.[db.widgetId]?.[prop] as NodeSetData).children ??\n []) as NodeData[][];\n } else {\n // If dataBinding doesn't exist yet, pull data off the dataset from selectedConnection\n const d = connectionDataStore.connectionData as NodeSetData;\n rows = (d?.children || []) as NodeData[][];\n }\n\n // Find the proper \"column\" (that contains the nodeUuid that initiated the tempBackgroundInfo)\n let dataSourceIndex = -1;\n rows.forEach((row) => {\n row.forEach((cell, i) => {\n if (cell.uuid === draggedPhotoNodeUuid) dataSourceIndex = i;\n });\n });\n\n // console.log(\"Yeup, it's dynamic\", rows, dataSourceIndex);\n if (dataSourceIndex > -1) {\n for (let i = 0; i < rows.length; i++) {\n const url = rows[i][dataSourceIndex]?.value;\n // console.log(\"Url...\", url);\n result[widgetId].push({ backgroundImageUrl: url, url });\n }\n }\n } else {\n // Just replace background image(s) with static url of dragged asset:\n // We need to add a copy of the url for each child of a repeater\n let gridSize = 1;\n if (widgets[widget.parentId]) {\n const { rows, columns, type } = getActiveWidget(\n widgets[widget.parentId],\n conditions\n ) as unknown as RepeaterOptions;\n if (type === \"Repeater\") {\n gridSize = rows * columns;\n }\n }\n\n for (let i = 0; i < gridSize; i++) {\n result[widgetId].push({\n backgroundImageUrl: tempBackgroundInfo?.url,\n url: tempBackgroundInfo?.url,\n });\n }\n }\n }\n }\n\n return result;\n};\n","import { makeId } from \"@/utils\";\n\nimport {\n ComponentOptions,\n DefaultComponentOptions,\n} from \"@/components/widgets/ComponentOptions\";\nimport {\n TransformOptions,\n DefaultTransformOptions,\n} from \"@/components/widgets/TransformOptions\";\nimport {\n BorderOptions,\n DefaultBorderOptions,\n} from \"@/components/widgets/BorderOptions\";\nimport { getDefaultAnimationOptions } from \"@/components/widgets/Animation\";\nimport { AnimationOptions } from \"@/types/animation\";\n\nexport interface VideoOptions\n extends ComponentOptions,\n TransformOptions,\n AnimationOptions,\n BorderOptions {\n url: string;\n loop: boolean;\n mimeType: string;\n}\n\nconst defaultOptions = {\n ...DefaultComponentOptions,\n ...DefaultTransformOptions,\n ...DefaultBorderOptions,\n ...getDefaultAnimationOptions(),\n loop: true,\n mimeType: \"video/unknown\",\n};\n\nconst CreateVideo = (options: Partial) => {\n const wid = makeId();\n return Object.assign({ wid }, defaultOptions, options, { type: \"Video\" });\n};\n\nexport default CreateVideo;\n","import { defineStore } from \"pinia\";\n\nexport interface MainAppState {\n recentlyUsedFonts: string[];\n}\n\n/**\n * The purpose of this store is to keep track of info that should be available when a user swaps between apps, and no matter what page they first load.\n */\n\nexport const useMainAppStore = defineStore(\"mainApp\", {\n state: (): MainAppState => {\n return {\n recentlyUsedFonts: [],\n };\n },\n\n actions: {\n addFont(font: string) {\n if (this.recentlyUsedFonts.includes(font)) {\n return;\n }\n\n if (this.recentlyUsedFonts.length === 3) {\n this.recentlyUsedFonts.shift();\n }\n this.recentlyUsedFonts.push(font);\n },\n },\n});\n","/* eslint-disable @typescript-eslint/no-unused-vars */\nimport { defineStore } from \"pinia\";\n\nimport { AppState, makeInitialAppState } from \"./AppState\";\nimport {\n Point,\n Rectangle,\n ResizeDimension,\n SavedAsset,\n Size,\n ResolvedWidgetData,\n NO_UNDO,\n ContentMenuName,\n} from \"@/types\";\nimport {\n Widget,\n WidgetProperties,\n WidgetWithConditions,\n} from \"@/components/widgets/Widget\";\nimport {\n AppInfo,\n AppPayload,\n DataBinding,\n DataBindingProperty,\n DataBindingType,\n DataConnection,\n Feed,\n FeedDeliveryMethod,\n PublishMetadata,\n SavedAppInfo,\n SchemaType,\n} from \"@/types/data\";\nimport { useConditionGroupsStore } from \"./conditionGroups\";\nimport {\n addConditionalVersions,\n getActiveConditionId,\n getActiveWidget,\n isConditionalBindingType,\n} from \"@/util/conditionsUtils\";\nimport { BASE_PARENT_ID, DEFAULT_CONDITION_ID } from \"@/constants\";\nimport {\n addVectors,\n dimsToUpdateApparentHeight,\n getAngledBoundingBox,\n getApparentDims,\n getApparentDimsInvert,\n getBoundingBox,\n makeId,\n purgeNullValues,\n removeDuplicateStrings,\n removeReactivity,\n rotateVector,\n scaleVector,\n scaledCoordsInvert,\n subtractVectors,\n} from \"@/utils\";\nimport {\n CreateGroup,\n GroupOptions,\n} from \"@/components/widgets/Group/GroupOptions\";\nimport { appSchemaVersion, applyModelMigrations } from \"@/migrations\";\nimport { api } from \"@/api/backend\";\nimport { useConnectionEditorStore } from \"@/stores/connectionEditor\";\nimport { EventBus } from \"@/eventbus\";\nimport CreateRepeater, {\n createRepeaterDimensions,\n RepeaterOptions,\n} from \"@/components/widgets/Repeater/RepeaterOptions\";\nimport Vue from \"vue\";\nimport { TransformOptions } from \"@/components/widgets/TransformOptions\";\nimport { makeDynamicTextContent, TextContent } from \"@/text\";\nimport CreateText, { TextOptions } from \"@/components/widgets/Text/TextOptions\";\nimport { measureText } from \"@/textfit\";\nimport { isTextContentEmpty, updateTokenBindings } from \"@/text/binding\";\nimport { getWidgetPropertyDefault } from \"@/components/widgets/getWidgetPropertyDefault\";\nimport CreateImage from \"@/components/widgets/Image/ImageOptions\";\nimport uniqWith from \"lodash.uniqwith\";\nimport { widgetData } from \"./widgetData\";\n\nimport { gatherAssets } from \"./gatherAssets\";\nimport { FontAssetInfo, ImageAssetInfo } from \"@/types/bundleTypes\";\nimport { useAppDataStore } from \"./appData\";\nimport { useConnectionsStore } from \"./connections\";\nimport { useConnectionDataStore } from \"./connectionData\";\nimport { applyDynamicImages } from \"./dynamicImages\";\nimport CreateDatetime from \"@/components/widgets/Datetime/DatetimeOptions\";\nimport { DraggingInfo } from \"./dragDrop\";\nimport CreateVideo from \"@/components/widgets/Video/VideoOptions\";\nimport { useMainAppStore } from \"./mainApp\";\n\n/**\n * Thanks to https://medium.com/@karenmarkosyan/how-to-manage-promises-into-dynamic-queue-with-vanilla-javascript-9d0d1f8d4df5\n *\n * This allows to us to manage calls to updateApp, which can come from 3 sources:\n * - user clicks \"Save button\"\n * - auto-save fires\n * - some \"app-triggered\" save event fires, such as pasteWidgetsClick, or remapCollection\n *\n * We want to ensure that only one call to updateApp is made at a time. So if a call is in progress, we should not initiate another.\n * We should wait until that first call is processed before firing the next one, to avoid triggering 409 errors on the back-end.\n * That is what this queue accomplishes! :)\n */\nclass Queue {\n static queue: any[] = [];\n static workingOnPromise = false;\n\n static enqueue(promise: () => Promise) {\n return new Promise((resolve, reject) => {\n this.queue.push({\n promise,\n resolve,\n reject,\n });\n this.dequeue();\n });\n }\n\n static dequeue() {\n if (this.workingOnPromise) {\n return false;\n }\n const item = this.queue.shift();\n if (!item) {\n return false;\n }\n try {\n this.workingOnPromise = true;\n item\n .promise()\n .then((value: any) => {\n this.workingOnPromise = false;\n item.resolve(value);\n setTimeout(() => this.dequeue(), 0);\n })\n .catch((err: any) => {\n this.workingOnPromise = false;\n item.reject(err);\n setTimeout(() => this.dequeue(), 0);\n });\n } catch (err) {\n this.workingOnPromise = false;\n item.reject(err);\n setTimeout(() => this.dequeue(), 0);\n }\n return true;\n }\n}\n\n// const updatePromiseQueue = new Queue();\n\n/**\n * Remove any data bindings that are no longer associated with a widget.\n */\nconst pruneUnusedDataBindings = (app: AppPayload) => {\n const bindings = app.dataBindings;\n const widgets = app.model.widgets;\n\n const filteredBindings = bindings.filter((b) => {\n const hasParentId =\n b.parentWidgetId && b.parentWidgetId !== BASE_PARENT_ID\n ? b.parentWidgetId in widgets\n : true;\n return hasParentId && b.widgetId in widgets;\n });\n\n app.dataBindings = filteredBindings;\n};\n\nconst rulerSize = 20;\nexport const initialArtboardPosition: Point = {\n x: rulerSize * 2,\n y: rulerSize * 2,\n};\n\nconst getMaxZ = (widgets: Widget[]) => {\n const values = widgets.map((wg) => wg.z).concat([0]);\n return Math.max(...values);\n};\n\nconst getMaxZByParentId = (state: EditorState, parentId?: string) => {\n parentId = parentId || state.editingContext.parentId;\n\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const values = Object.values(state.widgets)\n .map((wg) => getActiveWidget(wg, conditions))\n .filter((wg) => wg.parentId === parentId)\n .map((wg) => wg.z)\n .concat([0]);\n\n return Math.max(...values);\n};\n\n/**\n * Returns a mutable reference to a widget's 'active' props without\n * requiring you to specify the activeConditionId.\n *\n * DOES NOT RETURN top level props `wid`, `parentId`, `type` or `locked`.\n */\nconst getMutableWidgetProps = (\n widget: WidgetWithConditions\n): WidgetProperties => {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const cid = getActiveConditionId(widget?.wid, conditions);\n\n // Return a reference to this object, so it can be mutated\n return widget?.conditionalVersions?.[cid] as WidgetProperties;\n};\n\n// Used to position children on the canvas after ungrouped:\nconst getChildAbsolutePosition = (group: Widget, child: Widget): Point => {\n const { x, y, w, h, angle } = group as Widget;\n\n if (angle === 0) {\n // If not rotated, we simply add the parent x,y to the child x,y\n return { x: child.x + group.x, y: child.y + group.y };\n } else {\n // must account for group's rotation:\n const groupCenter = { x: x + w / 2, y: y + h / 2 };\n const groupPosRelCenter = { x: -w / 2, y: -h / 2 };\n const radians = (angle * Math.PI) / 180;\n const rotatedGroupPosRelCtr = rotateVector(groupPosRelCenter, radians);\n const rotatedGroupPosAbsolute = addVectors(\n groupCenter,\n rotatedGroupPosRelCtr\n );\n const wVec = rotateVector({ x: w, y: 0 }, radians);\n const hVec = rotateVector({ x: 0, y: h }, radians);\n\n let wgAbsolutePosition = addVectors(\n rotatedGroupPosAbsolute,\n scaleVector(wVec, child.x / w)\n );\n wgAbsolutePosition = addVectors(\n wgAbsolutePosition,\n scaleVector(hVec, child.y / h)\n );\n\n // now must find wVec and hVec for widget (child) itself, to find wg center.\n const wgWVec = rotateVector({ x: child.w, y: 0 }, radians);\n const wgHVec = rotateVector({ x: 0, y: child.h }, radians);\n\n let wgCtr = addVectors(wgAbsolutePosition, scaleVector(wgWVec, 0.5));\n wgCtr = addVectors(wgCtr, scaleVector(wgHVec, 0.5));\n\n const rotatedWgPosRelCtr = addVectors(\n scaleVector(wgWVec, -0.5),\n scaleVector(wgHVec, -0.5)\n );\n const ogWgPosRelCtr = rotateVector(rotatedWgPosRelCtr, -radians);\n const ogWgPosAbsolute = addVectors(wgCtr, ogWgPosRelCtr);\n return ogWgPosAbsolute;\n }\n};\n\nexport const recalculateGroupPosition = (\n group: TransformOptions,\n offset: Point\n): Point => {\n const { x, y, w, h, angle } = group;\n const radians = (Math.PI * angle) / 180;\n const ogCtr = { x: x + w / 2, y: y + h / 2 };\n const newGrpPreRotate = addVectors(group, offset);\n const newGrpRelOgCtr = subtractVectors(newGrpPreRotate, ogCtr);\n const newGrpRotatedRelCtr = rotateVector(newGrpRelOgCtr, radians);\n const newGrpRotated = addVectors(ogCtr, newGrpRotatedRelCtr);\n\n const wVec = rotateVector({ x: w - offset.x, y: 0 }, radians);\n const hVec = rotateVector({ x: 0, y: h - offset.y }, radians);\n\n let newGrpCtr = addVectors(newGrpRotated, scaleVector(wVec, 0.5));\n newGrpCtr = addVectors(newGrpCtr, scaleVector(hVec, 0.5));\n const newGrpRotatedPosRelCtr = addVectors(\n scaleVector(wVec, -0.5),\n scaleVector(hVec, -0.5)\n );\n const newGrpPosRelCtr = rotateVector(newGrpRotatedPosRelCtr, -radians);\n const newGrpPos = addVectors(newGrpCtr, newGrpPosRelCtr);\n\n return newGrpPos;\n};\n\n// TODO: Does this need to be a standalone method? Can it be an action?\nconst addWidget = (\n state: EditorState,\n widget: Widget | WidgetWithConditions\n) => {\n addConditionalVersions(widget as Widget);\n\n Vue.set(state.widgets, widget.wid, widget);\n\n state.hasUnsavedChanges = true;\n\n // Add parent reference\n if (widget.parentId !== BASE_PARENT_ID) {\n Vue.set(\n state.parents,\n widget.parentId,\n (state.parents[widget.parentId] || []).concat([widget.wid])\n );\n }\n};\n\nexport type CreateDataTokenOptions = {\n widgetId: string;\n conditionId: string;\n connectionId: string;\n parentWidgetId: string;\n isScalar: boolean;\n content: TextContent;\n dataUuid: string;\n\n query?: any;\n textTokenUuid?: string;\n\n shouldCreateRepeaterBinding?: boolean;\n dataParentUuid?: string;\n};\n\nexport declare type SelectedProps = {\n [property: string]: string[] | number[] | boolean[];\n};\n\nexport interface HoveredTextInfo {\n wid: string;\n}\n\nexport interface BindingReplacement {\n oldBinding: DataBinding;\n\n /**\n * If the new binding is undefined, the old binding will be removed rather than replaced.\n */\n newBinding: DataBinding | undefined;\n}\n\nexport interface RemapCollectionPayload {\n replacements: BindingReplacement[];\n newConnection: DataConnection;\n}\n\nexport interface BindingsSearchOptions {\n widgetId?: string;\n bindingType?: DataBindingType;\n schemaType?: SchemaType;\n dataConnectionUuid?: string;\n dataParentUuid?: string;\n property?: string;\n conditionUuid?: string;\n}\n\nexport interface EditingContext {\n parentId: string;\n widgetX: number;\n widgetY: number;\n offsetX: number;\n offsetY: number;\n width?: number;\n height?: number;\n repeaterIndex?: number;\n}\n\nconst defaultEditingContext = (): EditingContext => {\n return {\n parentId: BASE_PARENT_ID,\n widgetX: 0,\n widgetY: 0,\n offsetX: 0,\n offsetY: 0,\n };\n};\n\nexport interface EditorState extends AppState {\n selections: string[];\n hoveredId: string;\n rulerSize: number;\n savedAppInfo: SavedAppInfo;\n artboardPosition: Point;\n scale: number;\n scaleStep: number;\n clipboard: WidgetWithConditions[];\n canvasBox: Rectangle;\n editingContext: EditingContext;\n textEditingWidget: string | null;\n isLoaded: boolean;\n hasUnsavedChanges: boolean;\n modifiedAt: string | null;\n awaitingServer: boolean;\n\n // TODO: Move to wherever connections live\n cachedConnections: DataConnection[];\n\n activeContentMenu: ContentMenuName | null;\n animationPlaying: boolean;\n clickFromEditorPanel: boolean;\n isDraggingWidget: boolean;\n\n /**\n * If a widget is being resized using resize handles, this will\n * store the dimension (x, y or x & y) of the resize action.\n *\n * If a widget is not being resized, this will be null.\n */\n dragResizeDimension: ResizeDimension | null;\n\n cyclingIndexes: Record;\n}\n\nexport const makeInitialEditorState = (): EditorState => {\n const appState = makeInitialAppState();\n appState.appMode = \"edit\";\n return {\n ...appState,\n appSchemaVersion: 1,\n savedAppInfo: {} as SavedAppInfo,\n selections: [],\n hoveredId: \"\",\n clipboard: [],\n rulerSize: rulerSize,\n artboardPosition: { ...initialArtboardPosition },\n scale: 1,\n scaleStep: 0.1,\n canvasBox: { x: 0, y: 0, w: 0, h: 0 },\n editingContext: defaultEditingContext(),\n textEditingWidget: null,\n isLoaded: false,\n hasUnsavedChanges: false,\n modifiedAt: null,\n awaitingServer: false,\n cachedConnections: [],\n\n dataBindings: [],\n\n activeContentMenu: null,\n animationPlaying: false,\n clickFromEditorPanel: false,\n isDraggingWidget: false,\n dragResizeDimension: null,\n\n cyclingIndexes: {},\n };\n};\n\nexport const useAppEditorStore = defineStore(\"appEditor\", {\n state: (): EditorState => ({\n ...makeInitialEditorState(),\n }),\n\n getters: {\n widgetData(): ResolvedWidgetData {\n const appData = useAppDataStore();\n /**\n * Because renderer and editor have different ways\n * for determining the active condition for a widget,\n * we'll inject a function so the `widgetData` function\n * doesn't need to do so many conditionals based on appMode.\n */\n const result = widgetData(\n this.appMode,\n this.widgets,\n this.assets,\n this.dataBindings,\n appData.data,\n useConditionGroupsStore().activeWidgetConditionsMap\n );\n\n /**\n * This will scan through the widgetData and insert any\n * temporary, dynamic \"hover\" images that are\n * being dragged into the app.\n */\n return applyDynamicImages(\n this.widgets,\n this.dataBindings,\n appData.data,\n result\n );\n },\n\n renderableWidgets(): Widget[] {\n /**\n * Must spread out *all* dynamic options in getters.wigetData[wg.wid],\n * to account for case where a widget has multiple properties powered dynamically.\n */\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return (\n Object.values(this.widgets)\n .filter((wg) => wg.parentId === BASE_PARENT_ID)\n .map((wg) => getActiveWidget(wg, conditions))\n // .map((wg) => Object.assign({}, wg, getters.widgetData[wg.wid][0]));\n .map((wg) => {\n return {\n ...wg,\n ...this.widgetData[wg.wid].reduce((acc: object, val: object) => {\n return { ...acc, ...val };\n }, {}),\n };\n })\n );\n },\n editableWidgets(): Widget[] {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return Object.values(this.widgets)\n .filter((wg) => wg.parentId === this.editingContext.parentId)\n .map((wg) => getActiveWidget(wg, conditions))\n .map((wg) => Object.assign({}, wg, this.widgetData[wg.wid][0]));\n },\n\n artboard(): Rectangle {\n return {\n w: this.width,\n h: this.height,\n ...this.artboardPosition,\n };\n },\n\n customFonts(): string[] {\n return this.custom.fonts || [];\n },\n\n selectionCount(): number {\n return this.selections.length;\n },\n selectedProps(): SelectedProps {\n if (this.selections.length === 0) {\n return {};\n }\n\n return this.selections.reduce((result: any, wid: string) => {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const widget = getActiveWidget(this.widgets[wid], conditions);\n\n if (!widget) {\n return result;\n }\n\n for (const prop in widget) {\n const propValue = widget[prop as keyof Widget];\n if (typeof propValue !== \"undefined\") {\n if (prop in result) {\n const valuesArray = result[prop] as any[];\n if (!valuesArray.includes(propValue)) {\n valuesArray.push(propValue);\n }\n } else {\n result[prop] = [propValue];\n }\n }\n }\n\n return result;\n }, {});\n },\n selectedTypes(): string[] {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return this.selections.reduce((result: any, wid: string) => {\n const widget = getActiveWidget(this.widgets[wid], conditions);\n if (widget && !result.includes(widget.type)) {\n result.push(widget.type);\n }\n return result;\n }, []);\n },\n selectedWidget(): Widget | undefined {\n if (this.selections.length === 1) {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return getActiveWidget(this.widgets[this.selections[0]], conditions);\n }\n },\n\n selectedWidgetBinding(): DataBinding | undefined {\n return this.dataBindings.find(\n (b: DataBinding) => b.widgetId === this.selectedWidget?.wid\n );\n },\n\n /**\n * Returns the Point (x,y) of the active editing context relative to canvas origin\n */\n origin(): Point {\n let offsetX = this.editingContext.offsetX;\n let offsetY = this.editingContext.offsetY;\n if (!this.isBaseEditingContext) {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const wg = getActiveWidget(\n this.widgets[this.editingContext.parentId],\n conditions\n );\n if (wg) {\n const { x, y } = getApparentDims(wg);\n offsetX += x;\n offsetY += y;\n }\n }\n return {\n x: this.artboardPosition.x + offsetX,\n y: this.artboardPosition.y + offsetY,\n };\n },\n editContextOrigin(): Point {\n if (!this.isBaseEditingContext) {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const wg = getActiveWidget(\n this.widgets[this.editingContext.parentId],\n conditions\n );\n if (wg) {\n const { x, y } = getApparentDims(wg);\n const offsetX = this.editingContext.offsetX;\n const offsetY = this.editingContext.offsetY;\n return {\n x: this.artboardPosition.x + x + offsetX,\n y: this.artboardPosition.y + y + offsetY,\n };\n }\n }\n return this.artboardPosition;\n },\n isBaseEditingContext(): boolean {\n return this.editingContext.parentId === BASE_PARENT_ID;\n },\n assetsInUse(): {\n fonts: FontAssetInfo[];\n images: ImageAssetInfo[];\n } {\n const conditionGroupsStore = useConditionGroupsStore();\n const conditions = conditionGroupsStore.activeWidgetConditionsMap;\n return gatherAssets(\n this,\n this.renderableWidgets,\n this.widgetData,\n conditions\n );\n },\n\n // DATA BINDING GETTERS ----------------------------------\n\n uniquConnectionFromBindingsCount(): number {\n return this.dataBindings\n .map((b) => b.dataConnectionUuid)\n .filter((value, index, self) => self.indexOf(value) === index).length;\n },\n },\n\n actions: {\n // Adding a simple action for this, since we may want to make it undo-able\n setCyclingIndex(payload: { widgetId: string; value: number }): void {\n const { widgetId, value } = payload;\n Vue.set(this.cyclingIndexes, widgetId, value);\n },\n /**\n * Translates a rectangle from the scaled \"render\" context to the\n * unscaled \"editor\" context.\n *\n * It takes artboard position and canvas scale into account.\n *\n * This is useful for drawing boxes in the \"editor\" layer which will\n * correspond to objects in the \"render layer\" (which is scaled via CSS).\n *\n * This does not account for chilren of a rotated repeater, so we disallow rotated repeaters\n *\n */\n scaleForEditLayer(box: Rectangle): Rectangle {\n const { canvasBox, scale } = this;\n const canvasCenter = {\n x: canvasBox.x + canvasBox.w / 2,\n y: canvasBox.y + canvasBox.h / 2,\n };\n\n const editContext = this.editContextOrigin;\n const editContextRelCtr = subtractVectors(editContext, canvasCenter);\n const scaledVec = scaleVector(editContextRelCtr, scale);\n const editContextPos = addVectors(canvasCenter, scaledVec);\n\n // what?! why scale - 1??\n const boxX = box.x * scale + editContextPos.x + (scale - 1) * canvasBox.x;\n const boxY = box.y * scale + editContextPos.y + (scale - 1) * canvasBox.y;\n\n return {\n x: boxX,\n y: boxY,\n w: box.w * scale,\n h: box.h * scale,\n };\n },\n\n getChildren(parentWidgetId: string) {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return (this.parents[parentWidgetId] ?? []).map((childWidgetId) => {\n return Object.assign(\n {},\n getActiveWidget(this.widgets[childWidgetId], conditions),\n this.widgetData[childWidgetId]?.[0] ?? {}\n );\n });\n },\n\n widgetById(wid: string): Widget | undefined {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n return getActiveWidget(this.widgets[wid], conditions);\n },\n\n getArtboardCoordinates(client: Point, offset: Point): Point {\n const offsetX = this.editingContext?.offsetX ?? 0;\n const offsetY = this.editingContext?.offsetY ?? 0;\n const widgetX = this.editingContext?.widgetX ?? 0;\n const widgetY = this.editingContext?.widgetY ?? 0;\n\n const p2 = subtractVectors(client, this.canvasBox);\n const coords = scaledCoordsInvert(p2, this.canvasBox, this.scale);\n const coordsRelArtboard = subtractVectors(coords, this.artboard);\n const x = coordsRelArtboard.x - offset.x / this.scale - offsetX - widgetX;\n const y = coordsRelArtboard.y - offset.y / this.scale - offsetY - widgetY;\n return { x, y };\n },\n\n // Calculate the y value of each child in a vertically dynamic group (as it will be rendered):\n verticallyDynamicChild(payload: { parentWid: string; childWid: string }) {\n const { parentWid, childWid } = payload;\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const { verticalMargin, verticalDynamism, orderedChildIds } =\n getActiveWidget(this.widgets[parentWid], conditions) as GroupOptions;\n if (!verticalDynamism) return undefined;\n let y = 0;\n const children = this.parents[parentWid].map((wid) => this.widgets[wid]);\n const result: Widget[] = [];\n children\n .slice(0)\n .sort((a, b) => {\n if (!orderedChildIds) return 1;\n return (\n orderedChildIds.indexOf(a.wid) - orderedChildIds.indexOf(b.wid)\n );\n })\n .map((wg) => getActiveWidget(wg, conditions))\n .forEach((child) => {\n result.push({ ...child, y });\n y += child.h * child.scaleY;\n y += verticalMargin;\n });\n return result.find((wg) => wg.wid === childWid);\n },\n\n // TODO: Revisit these \"save\" and \"preserve\" lists...\n // This is used to clear the state (but preserve certain info, and reset the widgets to saved state) when undo is called:\n emptyState() {\n const clearThroughUndo = [\"widgets\", \"parents\", \"name\"];\n\n // https://github.com/vuejs/vuex/issues/1118\n const s = makeInitialEditorState();\n\n // const conditionsStore = useConditionGroupsStore();\n\n clearThroughUndo.forEach((key) => ((this as any)[key] = (s as any)[key]));\n this.dataBindings = [];\n\n // seems to fix weird bug:\n if (this.savedAppInfo === undefined) {\n this.savedAppInfo = {} as SavedAppInfo;\n }\n // Initialize the state correctly, given the saved state of the current app:\n // console.log(\"empty state\", state.savedAppInfo);\n this.widgets = removeReactivity(this.savedAppInfo.widgets);\n this.parents = removeReactivity(this.savedAppInfo.parents);\n this.name = removeReactivity(this.savedAppInfo.name);\n this.dataBindings = removeReactivity(this.savedAppInfo.dataBindings);\n\n // (this as any).data.data = removeReactivity(this.savedAppInfo.data);\n\n // console.log(\"emptied state...\", this.widgets, this.savedAppInfo);\n },\n\n resetEditorState() {\n // TODO: determine whether Pinia's built-in $reset method is the\n // same as using makeInitialEditorState();\n this.$reset();\n // eslint-disable-next-line @typescript-eslint/no-unused-vars\n // state = makeInitialEditorState();\n },\n\n initStateWith(payload: AppPayload) {\n this.hasUnsavedChanges = false;\n Object.keys(payload).forEach((key) => {\n if (key === \"dataBindings\") {\n // DataBindings are handled in loadApp()\n } else if (key === \"model\") {\n this.widgets = payload.model.widgets as Record<\n string,\n WidgetWithConditions\n >;\n this.parents = payload.model.parents;\n this.custom = payload.model.custom;\n this.appSchemaVersion = payload.model.appSchemaVersion;\n this.selections = [];\n } else {\n (this as any)[key] = (payload as any)[key];\n }\n });\n },\n\n setChecksum(payload: { checksum: number; modifiedAt: string }) {\n this.checksum = payload.checksum;\n this.modifiedAt = payload.modifiedAt;\n },\n\n setHasUnsavedChanges(hasChanges: boolean) {\n this.hasUnsavedChanges = hasChanges;\n },\n\n // This is used to totally clear state in app builder's beforeRouteEnter method:\n clearWidgets() {\n this.widgets = {};\n },\n\n // ARTBOARD STUFF -----------------------------------------------\n\n positionArtboard(point: Point) {\n this.artboardPosition.x = point.x;\n this.artboardPosition.y = point.y;\n },\n\n resetArtboard() {\n this.artboardPosition = { ...initialArtboardPosition };\n this.scale = 1;\n },\n\n // SELECTIONS STUFF -------------------------------------------\n\n addSelection(wid: string) {\n if (!this.selections.includes(wid)) {\n this.selections.push(wid);\n }\n },\n\n removeSelection(wid: string) {\n // console.log(\"remove sels State\", wid);\n\n const index = this.selections.indexOf(wid);\n if (index > -1) {\n this.selections.splice(index, 1);\n }\n },\n\n replaceSelections(wids: string[]) {\n // console.log(\"replace sels State\", wids);\n this.selections = wids;\n },\n\n setHoveredId(wid: string) {\n this.hoveredId = wid;\n },\n\n copyWidgets() {\n if (this.selections.length > 0) {\n this.clipboard = this.selections\n .map((wid) => {\n return { ...this.widgets[wid] };\n })\n .filter((wg) => !wg.locked);\n }\n },\n\n setAppName(name: string) {\n this.hasUnsavedChanges = true;\n // console.log(\"set appp name\", name);\n this.name = name;\n },\n\n setAppDimensions(payload: { width: number; height: number }) {\n this.hasUnsavedChanges = true;\n this.width = payload.width;\n this.height = payload.height;\n },\n\n setWidgets(widgets: Record) {\n this.widgets = widgets;\n },\n\n setTimeZone(timeZone: string) {\n this.hasUnsavedChanges = true;\n this.ianaTimeZone = timeZone;\n },\n\n resetUndoRedoState(appInfo?: SavedAppInfo) {\n EventBus.emit(\"CLEAR_UNDO_STACK\");\n\n if (typeof appInfo === \"undefined\") {\n /* Reconstruct appInfo based on current app state */\n appInfo = {\n name: this.name,\n appSchemaVersion: this.appSchemaVersion,\n parents: this.parents,\n widgets: this.widgets,\n data: (this as any).data,\n dataBindings: this.dataBindings,\n custom: {\n fonts: [],\n userUploadedAssets: [],\n },\n };\n }\n\n // console.log(\"reset undo redo...setting savedappinfo\", appInfo);\n\n this.savedAppInfo = removeReactivity(appInfo);\n },\n\n setCanvasBox(box: Rectangle) {\n // console.log(\"set canvasbox\", box);\n this.canvasBox.x = box.x;\n this.canvasBox.y = box.y;\n this.canvasBox.w = box.w;\n this.canvasBox.h = box.h;\n },\n\n // WIDGET STUFF -----------------------------------------------\n\n // NOTE: Think we must pass in group id to this, so exists in payload and can be undone/redone\n\n // I think we should just create fresh children...using scaleX,scaleY,w,h,x,y to find apparentDims, and give those to child as its real dims\n // I.e. whenever a widget is grouped, or ungrouped, it will start \"fresh\" with a new w and h (assuming it has been scaled).\n // Aha, and if we do this when we create a group, don't have to worry about ungrouping -- bc no way to resize children. Nice.\n groupWidgets(payload: {\n childWidgetIds: string[];\n groupWidgetId: string;\n groupAsRepeater?: boolean;\n parentId: string;\n }) {\n const { childWidgetIds, groupWidgetId, groupAsRepeater, parentId } =\n payload;\n\n EventBus.emit(\"IGNORE_TEXT_WIDGET_LISTENERS\", true);\n\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const widgets = removeReactivity(\n childWidgetIds.map((id) => this.widgets[id])\n ).map((wg) => getActiveWidget(wg, conditions));\n\n const z = getMaxZ(widgets);\n const groupBox = getBoundingBox(widgets);\n let group: GroupOptions | RepeaterOptions = Object.assign(\n {},\n CreateGroup({\n ...groupBox,\n z: z,\n parentId,\n }),\n { wid: groupWidgetId }\n );\n\n /*\n * Size and position the Repeater so that the selected widgets fill one cell, and they do not appear to move when grouped\n */\n\n const rowGap = 0;\n const columnGap = 0;\n\n if (groupAsRepeater) {\n // NOTE: would be nice to compute how many cells can fit based on size/pos of selection, and artboard size\n const rows = 1;\n const columns = 1;\n // rowGap = 20;\n // columnGap = 20;\n const repeaterWidth = columns * groupBox.w + (columns + 1) * columnGap;\n const repeaterHeight = rows * groupBox.h + (rows + 1) * rowGap;\n const repeaterBox = {\n x: groupBox.x - columnGap,\n y: groupBox.y - rowGap,\n w: repeaterWidth,\n h: repeaterHeight,\n };\n group = Object.assign(\n {},\n CreateRepeater({\n ...repeaterBox,\n z: z,\n parentId,\n rowGap,\n columnGap,\n rows,\n columns,\n }),\n { wid: groupWidgetId }\n );\n }\n\n const children = this.parents[parentId];\n if (Array.isArray(children)) {\n // Remove the new group child ids from the current editingContext child list.\n Vue.set(\n this.parents,\n parentId,\n children.filter((wid) => !childWidgetIds.includes(wid))\n );\n // Add new group to current editingContext child list.\n if (!children.includes(group.wid)) {\n this.parents[parentId].push(group.wid);\n }\n }\n\n // Update position & parent info for group children.\n childWidgetIds.forEach((wid) => {\n const { conditionalVersions } = this.widgets[wid];\n\n for (const cid in conditionalVersions) {\n const wg = this.widgets[wid].conditionalVersions[cid];\n const rect = getApparentDims(wg);\n wg.x = rect.x - group.x - rowGap;\n wg.y = rect.y - group.y - rowGap;\n wg.w = rect.w;\n wg.h = rect.h;\n wg.scaleX = 1;\n wg.scaleY = 1;\n this.widgets[wid].parentId = group.wid;\n }\n\n // wg.parentId = group.wid;\n });\n\n addConditionalVersions(group as Widget);\n\n // Add new group to widgets list.\n Vue.set(this.widgets, group.wid, group);\n\n // Add child widgets to new group child list\n Vue.set(this.parents, group.wid, childWidgetIds);\n\n // Select new group widget\n this.selections = [group.wid];\n\n // On the next tick, turn this off, so that we can ignore height-shifting watcher in TextWrapper\n setTimeout(() => {\n EventBus.emit(\"IGNORE_TEXT_WIDGET_LISTENERS\", false);\n }, 0);\n },\n\n addToNewRepeater(\n payload: { wid: string; parentWidgetId: string },\n NO_UNDO?: NO_UNDO\n ) {\n const { wid, parentWidgetId } = payload;\n const widget = this.widgets[wid];\n const mutableProps = getMutableWidgetProps(widget);\n Vue.set(widget, \"parentId\", parentWidgetId);\n Vue.set(mutableProps, \"x\", 20);\n Vue.set(mutableProps, \"y\", 20);\n\n Vue.set(this.parents, parentWidgetId, [wid]);\n },\n\n destroyGroup(wid: string) {\n this.verticalDynamismOff({ wid: wid }, \"NO_UNDO\");\n this.ungroupWidgets(wid);\n },\n\n ungroupWidgets(wid: string) {\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const groupWidget = getActiveWidget(this.widgets[wid], conditions);\n const childIds = this.parents[wid];\n\n // console.log(\"ungroup\", childIds.slice(0));\n\n if (Array.isArray(childIds)) {\n // Update position for children\n childIds.forEach((wid) => {\n for (const cid in this.widgets[wid].conditionalVersions) {\n const child = this.widgets[wid].conditionalVersions[cid];\n const newPosition = getChildAbsolutePosition(\n groupWidget,\n child as Widget\n );\n child.x = newPosition.x;\n child.y = newPosition.y;\n child.angle += groupWidget.angle;\n }\n });\n\n // Figure out how to adjust the z-index of all child widgets\n // so they fit in the same visual order after ungrouping.\n\n // Ahhh, cannot use state.widgets' keys here, because they include all children\n const groupSiblingIds =\n groupWidget.parentId === BASE_PARENT_ID\n ? Object.values(this.widgets)\n .filter((wg) => wg.parentId === BASE_PARENT_ID)\n .map((wg) => wg.wid)\n : this.parents[groupWidget.parentId];\n\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const sortByZ = (wids: string[]) => {\n const copy = wids.slice(0);\n copy.sort(\n (a, b) =>\n getActiveWidget(this.widgets[a], conditions).z -\n getActiveWidget(this.widgets[b], conditions).z\n );\n return copy;\n };\n\n const sortedWidgetIds = sortByZ(groupSiblingIds);\n const sortedChildIds = sortByZ(childIds);\n const indexOfGroupWidget = sortedWidgetIds.indexOf(wid);\n sortedWidgetIds.splice(indexOfGroupWidget, 1, ...sortedChildIds);\n\n // Update new z indexes for all widgets\n // TODO: Do for each version\n removeDuplicateStrings(sortedWidgetIds).forEach(\n (wid: string, i: number) => {\n const wg = getMutableWidgetProps(this.widgets[wid]);\n this.widgets[wid].parentId = groupWidget.parentId;\n wg.z = i + 1;\n\n // Add child to new parent array\n const children = this.parents[groupWidget.parentId];\n if (Array.isArray(children) && !children.includes(wid)) {\n children.push(wid);\n }\n }\n );\n\n // Remove old group widget from it's parent's child list\n const children = this.parents[groupWidget.parentId];\n if (Array.isArray(children)) {\n const groupWidgetChildIndex = children.indexOf(groupWidget.wid);\n if (groupWidgetChildIndex > -1) {\n children.splice(groupWidgetChildIndex, 1);\n }\n }\n\n // Remove Group Widget\n Vue.delete(this.widgets, groupWidget.wid);\n Vue.delete(this.parents, groupWidget.wid);\n this.selections = childIds;\n }\n },\n\n nudge(payload: {\n direction: \"x\" | \"y\";\n distance: number;\n selections: string[];\n }) {\n const { direction, distance, selections } = payload;\n selections.forEach((wid: string) => {\n const props = getMutableWidgetProps(this.widgets[wid]);\n if (props) {\n props[direction] += distance;\n }\n });\n },\n\n distribute(payload: { direction: \"vert\" | \"hor\"; selections: string[] }) {\n const { direction, selections } = payload;\n const props: TransformOptions[] = selections\n .map((wid: string) => this.widgets[wid])\n .map(getMutableWidgetProps);\n\n if (props.length <= 2) {\n return;\n }\n const axis = direction === \"hor\" ? \"x\" : \"y\";\n const span = direction === \"hor\" ? \"w\" : \"h\";\n const sorted = props.sort((a, b) => a[axis] - b[axis]);\n const [first, ...rest] = sorted;\n const last = rest.splice(rest.length - 1, 1)[0];\n const firstEnd = first[axis] + first[span];\n const lastStart = last[axis];\n let spaceToDistribute = lastStart - firstEnd;\n const restSpans = rest.reduce((acc: number, val) => acc + val[span], 0);\n spaceToDistribute -= restSpans;\n const numSpaces = sorted.length - 1;\n const eachSpanLength = spaceToDistribute / numSpaces;\n let current = firstEnd;\n rest.forEach((wg) => {\n wg[axis] = Math.round(current + eachSpanLength);\n current += wg[span] + eachSpanLength;\n });\n },\n\n toggleSyncTextColors(payload: {\n wid: string;\n syncTextColors: boolean;\n textElements: string[];\n }) {\n const { wid, syncTextColors, textElements } = payload;\n // NOTE: won't work for children\n const props = getMutableWidgetProps(this.widgets[wid]);\n Vue.set(props, \"syncTextColors\", syncTextColors);\n\n if (syncTextColors) {\n // Always assume \"master\" textcolor is first in textElements array\n const masterTextColorProp = `${textElements[0]}_textColor`;\n const masterColor = props[masterTextColorProp];\n // console.log(\"turnon stc\", masterColor);\n textElements.forEach((el) => {\n Vue.set(props, `${el}_textColor`, masterColor);\n });\n }\n },\n\n applyTextColor(\n widgetId: string,\n syncTextColors: boolean,\n element: string,\n colorValue: string,\n NO_UNDO?: NO_UNDO\n ) {\n const props = getMutableWidgetProps(this.widgets[widgetId]);\n\n if (syncTextColors) {\n const keysToSync = Object.keys(props).filter((key) =>\n key.includes(\"_textColor\")\n );\n keysToSync.forEach((key) => {\n Vue.set(props, key, colorValue);\n });\n } else {\n Vue.set(props, `${element}_textColor`, colorValue);\n }\n },\n\n toggleSyncFonts(payload: {\n wid: string;\n syncFonts: boolean;\n textElements: string[];\n }) {\n const { wid, syncFonts, textElements } = payload;\n\n // NOTE: won't work for children\n const props = getMutableWidgetProps(this.widgets[wid]);\n Vue.set(props, \"syncFonts\", syncFonts);\n\n // Activate sync fonts (do it here to work with undo/redo)\n if (syncFonts) {\n // Always assume \"master\" font is first in textElements array\n const masterFontProp = `${textElements[0]}_fontFamily`;\n const masterFont = props[masterFontProp];\n\n textElements.forEach((el: string) => {\n Vue.set(props, `${el}_fontFamily`, masterFont);\n });\n }\n },\n\n applyFontFamily(payload: {\n model: { wid: string; syncFonts: boolean };\n element: string;\n value: unknown;\n }) {\n const { model, element, value } = payload;\n const props = getMutableWidgetProps(this.widgets[model.wid]);\n\n if (model.syncFonts) {\n const keysToSync = Object.keys(props).filter((key) =>\n key.includes(\"_fontFamily\")\n );\n keysToSync.forEach((key) => {\n Vue.set(props, key, value);\n });\n } else {\n Vue.set(props, `${element}_fontFamily`, value);\n }\n },\n\n verticalDynamismOn(payload: { wid: string }) {\n const { wid } = payload;\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n // NOTE: won't work for children\n const props = getMutableWidgetProps(this.widgets[wid]);\n const children = this.parents[wid].map((id) =>\n getActiveWidget(this.widgets[id], conditions)\n );\n\n // Sort children based on current y-values:\n const orderedChildIds = children\n .slice(0)\n .sort((a, b) => {\n return a.y - b.y;\n })\n .map((wg) => wg.wid);\n\n Vue.set(props, \"orderedChildIds\", orderedChildIds);\n\n // Compute new height of group:\n const newH =\n children.slice(0).reduce((total: number, wg: Widget) => {\n return total + wg.h * wg.scaleY;\n }, 0) +\n props.verticalMargin * (children.length - 1);\n\n Vue.set(props, \"h\", newH);\n Vue.set(props, \"verticalDynamism\", true);\n },\n\n verticalDynamismOff(payload: { wid: string }, NO_UNDO?: NO_UNDO) {\n const { wid } = payload;\n const groupProps = getMutableWidgetProps(\n this.widgets[wid]\n ) as GroupOptions;\n\n if (!groupProps) return;\n\n // Give each child a new y-value to reflect its current (dynamic) position:\n let y = 0;\n (groupProps.orderedChildIds || []).forEach((wid: string) => {\n const childProps = getMutableWidgetProps(this.widgets[wid]);\n Vue.set(childProps, \"y\", y);\n\n y += childProps.h * childProps.scaleY;\n y += groupProps.verticalMargin;\n });\n\n Vue.set(groupProps, \"verticalDynamism\", false);\n },\n\n align(payload: { direction: string; selections: string[] }) {\n const { direction, selections } = payload;\n\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n const dimensions = selections.map((wid: string) =>\n getApparentDims(getActiveWidget(this.widgets[wid], conditions))\n );\n\n const box = getBoundingBox(dimensions);\n\n selections.forEach((wid: string) => {\n const wg = getMutableWidgetProps(this.widgets[wid]) as TransformOptions;\n\n const apparentWg = getApparentDims(wg);\n // const mergedWg = { ...apparentWg };\n\n // Account for possible widget rotation\n const mergedWg = getAngledBoundingBox({\n ...apparentWg,\n }) as TransformOptions;\n\n const originalMergedY = mergedWg.y;\n const originalMergedX = mergedWg.x;\n\n switch (direction) {\n case \"left\":\n // wg.x = box.x;\n mergedWg.x = box.x;\n break;\n case \"right\":\n mergedWg.x = box.x + box.w - mergedWg.w;\n break;\n case \"ctr-col\":\n mergedWg.x = box.x + box.w / 2 - mergedWg.w / 2;\n break;\n case \"top\":\n mergedWg.y = box.y;\n break;\n case \"bottom\":\n mergedWg.y = box.y + box.h - mergedWg.h;\n break;\n case \"ctr-row\":\n mergedWg.y = box.y + box.h / 2 - mergedWg.h / 2;\n break;\n }\n\n // Problem is....if real y is 0, and h is 100, and scaleY is 2, apparent Y is going to be -100.\n // Huh...don't see why this was needed here\n // OOOOH because this uses a DIFFERENT getapparentdims! Which rturns scalex ad scaley of 1\n // translate.js has its own...\n mergedWg.scaleX = wg.scaleX;\n mergedWg.scaleY = wg.scaleY;\n\n /**\n * If the widget is not rotated (has angle 0), we can simply compute the real coordinates that the widget must have in order to *appear* at its new dimensions/position.\n *\n * If the widget is rotated, we first compute the angled bounding box (the smallest non-rotated rectangle that encloses the rotated widget's bounding box).\n * Then, we compute the difference that this must be moved (along x or y axis) in order to appear at its target position.\n * Then we add this difference to the widget's current x or y position to get its target apparent position, and as before, compute the real coordinates based on that.\n */\n\n if (wg.angle == 0) {\n const dims = getApparentDimsInvert(mergedWg);\n\n wg.x = dims.x;\n wg.y = dims.y;\n wg.w = dims.w;\n wg.h = dims.h;\n } else {\n // Handle rotated widgets\n\n const dims = getApparentDimsInvert({\n ...wg,\n x: wg.x + mergedWg.x - originalMergedX,\n y: wg.y + mergedWg.y - originalMergedY,\n w: wg.w,\n h: wg.h,\n scaleX: 1,\n scaleY: 1,\n });\n wg.x = dims.x;\n wg.y = dims.y;\n wg.w = dims.w;\n wg.h = dims.h;\n }\n });\n },\n\n addWidget(widget: Widget, NO_UNDO?: NO_UNDO) {\n const z = getMaxZByParentId(this);\n widget.z = z + 1;\n addWidget(this, widget);\n },\n\n duplicateWidget(widget: any, isFromBaseLevel: boolean) {\n const dbs = widget.bindings;\n if (dbs.length > 0) {\n dbs.forEach((binding: DataBinding) => {\n const newBinding = Object.assign({}, binding, {\n widgetId: widget.wid,\n parentWidgetId: widget.parentId,\n });\n this.dataBindings.push(newBinding);\n });\n }\n\n const z = getMaxZByParentId(this);\n\n if (isFromBaseLevel) {\n const PASTE_OFFSET = { x: 20, y: 20 };\n\n for (const cid in widget.conditionalVersions) {\n const props = widget.conditionalVersions[cid];\n props.x += PASTE_OFFSET.x;\n props.y += PASTE_OFFSET.y;\n props.z = z + 1;\n }\n }\n\n addWidget(this, {\n ...widget,\n z: z + 1,\n });\n // }\n },\n\n /**\n * Must behave as follows:\n * - For all \"base level\" widgets (widgets from the level that we copied from),\n * - add data bindings that were passed on payload -- needed there, because they refer to original widget id\n * - and shift by paste_offset\n * - and go through all their children and Duplicate them\n * - Copy data bindings\n * - Copy widget\n */\n\n pasteWidgets(payload: {\n widgets: (WidgetWithConditions & { bindings: DataBinding[] })[];\n }) {\n const { widgets } = payload;\n\n widgets.forEach((wg, index) => {\n this.duplicateWidget(wg, wg.parentId === widgets[0].parentId);\n });\n },\n\n moveGroup(\n payload: {\n wids: string[];\n delta: Point;\n groupStartingPositions: Record;\n },\n NO_UNDO?: NO_UNDO\n ) {\n const widgets: WidgetWithConditions[] = payload.wids.map(\n (wid: string) => this.widgets[wid]\n );\n const { delta, groupStartingPositions } = payload;\n\n widgets.forEach((wg) => {\n const startingPos = groupStartingPositions[wg.wid];\n const props = getMutableWidgetProps(wg);\n props.x = startingPos.x + delta.x;\n props.y = startingPos.y + delta.y;\n });\n },\n\n removeDataBindingsForWidgets(payload: {\n widgetsInfo: { wid: string; conditionId: string }[];\n }) {\n const { widgetsInfo } = payload;\n this.hasUnsavedChanges = true;\n\n if (widgetsInfo && widgetsInfo.length > 0) {\n widgetsInfo.forEach(({ wid, conditionId }) => {\n const widget = this.widgets[wid];\n\n // Must clear out extra bindings here, because we are not yet deleting them when user updates bg image with new photo\n\n const bindings = this.dataBindings as DataBinding[];\n const newBindings = bindings.filter((db) => {\n return !(db.widgetId === wid && db.conditionUuid === conditionId);\n });\n Vue.set(this, \"dataBindings\", newBindings);\n Vue.set(widget, \"tempBackgroundInfo\", { url: \"\" }); // TODO: can we do this elsewhere?\n });\n }\n },\n\n removeWidgets(payload: { wids: string[] }, NO_UNDO?: NO_UNDO) {\n const widgetIds = payload.wids;\n\n EventBus.emit(\"IGNORE_TEXT_WIDGET_LISTENERS\", true);\n\n this.hasUnsavedChanges = true;\n\n if (widgetIds && widgetIds.length > 0) {\n widgetIds.forEach((wid) => {\n const widget = this.widgets[wid];\n if (!widget) return;\n const parentId = widget.parentId;\n const bindings = this.dataBindings as DataBinding[];\n\n // Remove parents reference and all children:\n if ([\"Repeater\", \"Group\"].includes(widget.type)) {\n // Delete children from state.widgets\n (this.parents[wid] || []).forEach((id) => {\n Vue.delete(this.widgets, id);\n\n // Also delete bindings for child:\n const newBindings = this.dataBindings.filter(\n (b: DataBinding) => b.widgetId !== id\n );\n Vue.set(this, \"dataBindings\", newBindings);\n });\n Vue.delete(this.parents, wid);\n }\n\n // Delete widget\n Vue.delete(this.widgets, wid);\n\n // Delete databindings associated with widget\n const newBindings = this.dataBindings.filter(\n (db: DataBinding) => db.widgetId !== wid\n );\n Vue.set(this, \"dataBindings\", newBindings);\n\n const children = this.parents[parentId];\n\n if (Array.isArray(children) && children.length > 0) {\n // Remove widget from parent array\n const childIndex = children.indexOf(wid);\n if (childIndex > -1) {\n children.splice(childIndex, 1);\n }\n }\n\n // If the widget has a parent, we need to do more work\n if (parentId !== BASE_PARENT_ID) {\n const parentType = this.widgets[parentId].type;\n const parentProps = getMutableWidgetProps(this.widgets[parentId]);\n if (!parentProps) return;\n\n if (parentType === \"Group\") {\n // OOOH. Only do this if was a group!\n // If the widget we deleted above was an \"only child\", delete it's parent too. So cruel...\n if (children.length === 0) {\n Vue.delete(this.widgets, parentId);\n Vue.delete(this.parents, parentId);\n\n const idx = bindings.findIndex(\n (db) => db.widgetId === parentId\n );\n if (idx > -1) {\n bindings.splice(idx, 1);\n }\n } else {\n // Resposition group widget and children\n const childWidgets = children\n .map((wid) => this.widgets[wid])\n .map((wg) => getMutableWidgetProps(wg));\n const childrenBounds = getBoundingBox(childWidgets);\n const newPosition = recalculateGroupPosition(\n parentProps,\n childrenBounds\n );\n\n parentProps.x = newPosition.x;\n parentProps.y = newPosition.y;\n parentProps.w = childrenBounds.w;\n parentProps.h = childrenBounds.h;\n\n childWidgets.forEach((wg) => {\n wg.x -= childrenBounds.x;\n wg.y -= childrenBounds.y;\n });\n }\n }\n }\n });\n }\n\n setTimeout(() => {\n EventBus.emit(\"IGNORE_TEXT_WIDGET_LISTENERS\", false);\n }, 0);\n },\n\n toggleLockAspect(payload: { lockAspect: boolean; wid: string }) {\n const { lockAspect, wid } = payload;\n const props = getMutableWidgetProps(this.widgets[wid]);\n\n if (typeof props === \"undefined\") {\n return;\n }\n\n props.lockAspect = lockAspect;\n },\n\n setWidgetStackOrder(widgetIds: string[]) {\n (widgetIds || []).forEach((wid: string, i: number) => {\n const props = getMutableWidgetProps(this.widgets[wid]);\n if (props) {\n props.z = i + 1;\n }\n });\n },\n\n addCustomFont(font: string) {\n if (!this.custom.fonts) {\n Vue.set(this.custom, \"fonts\", []);\n }\n this.custom.fonts.push(font);\n },\n\n setAwaitingServer(val: boolean) {\n this.awaitingServer = val;\n },\n\n setPublishMeta(val: PublishMetadata) {\n this.publishMeta = val;\n },\n\n setAppVersionUuid(val: string) {\n this.appVersionUuid = val;\n },\n\n // WIDGET SETTINGS STUFF -----------------------------------------------\n\n setWidgetProps(\n widgetIds: string[],\n props: Partial,\n NO_UNDO?: NO_UNDO\n ) {\n if (NO_UNDO === undefined) {\n this.hasUnsavedChanges = true;\n }\n\n const mainAppStore = useMainAppStore();\n\n widgetIds.forEach((widgetId: string) => {\n if (widgetId !== undefined) {\n const widgetProps = getMutableWidgetProps(this.widgets[widgetId]);\n if (!widgetProps) return;\n for (const prop in props) {\n Vue.set(widgetProps, prop, props[prop]);\n if (prop.includes(\"fontFamily\")) {\n mainAppStore.addFont(props[prop]);\n }\n }\n }\n });\n },\n\n updateTextContent(\n payload: {\n wid: string;\n content: TextContent;\n },\n NO_UNDO?: NO_UNDO\n ) {\n const { content, wid } = payload;\n if (!wid) return;\n const props = getMutableWidgetProps(this.widgets[wid]) as TextOptions;\n\n if (props) {\n props.content = content;\n const size = measureText(props);\n\n // wg.h = size.h; // OLD\n\n const { h, y } = dimsToUpdateApparentHeight(props, size.h);\n\n // const newH = size.h / wg.scaleY;\n // const newY = wg.y + (wg.h - newH) * (1 - wg.scaleY);\n\n props.h = h;\n props.y = y;\n }\n },\n\n /**\n * Replaces all datatoken bindings in text widgets\n */\n updateTextBindings(payload: { replacements: BindingReplacement[] }) {\n payload.replacements.forEach((replacement) => {\n const widgetId = replacement.oldBinding.widgetId;\n const props = getMutableWidgetProps(\n this.widgets[widgetId]\n ) as TextOptions;\n\n // Need to access `type` here because it doesn't\n // exist on the `props` object.\n const type = this.widgets[widgetId].type;\n\n if (typeof props !== \"undefined\" && type === \"Text\") {\n props.content = updateTokenBindings(\n props.content,\n replacement.oldBinding.dataUuid,\n replacement.newBinding?.dataUuid\n );\n }\n });\n },\n\n resetWidgetProperties(payload: {\n properties: { property: string; widgetId: string }[];\n }) {\n payload.properties.forEach((item) => {\n const widget = this.widgets[item.widgetId] as any;\n widget[item.property] = getWidgetPropertyDefault(\n widget.type,\n item.property\n );\n });\n },\n\n // SCALE STUFF -----------------------------------------------\n\n setScale(value: number) {\n if (Math.abs(1 - value) < 0.05) {\n value = Math.round(value);\n }\n this.scale = value;\n },\n\n leaveTextEditMode() {\n // Tricky..because we do NOT want to deselect if they trigger Blur by clicking a handle...\n // Also, want to be able to select another wg on same click\n // console.log(\"leave textedit\");\n this.selections = [];\n this.textEditingWidget = null;\n },\n\n enterTextEditMode(wid: string) {\n this.textEditingWidget = wid;\n },\n\n setEditingContext(context: EditingContext) {\n this.editingContext = context;\n this.selections = [];\n },\n\n resetEditingContext() {\n this.editingContext = defaultEditingContext();\n this.selections = [];\n },\n\n /**\n * We call this when an asset is added to the canvas\n * NOTE: When app data is loaded from the server, the\n * assets array from the server data overwrites the\n * local assets array. This is ok because they should\n * be completely synchronized.\n */\n addAsset(asset: SavedAsset) {\n const index = this.assets.findIndex((a) => a.uuid === asset.uuid);\n if (index > -1) {\n this.assets.splice(index, 1);\n }\n this.assets.push(asset);\n },\n\n // setShouldRefreshConnections(state, val: boolean) {\n // state.shouldRefreshConnections = val;\n // },\n\n setCachedConnections(conns: DataConnection[]) {\n this.cachedConnections = conns;\n },\n\n addWidgetCondition(payload: { conditionUuid: string; widgetId: string }) {\n const { conditionUuid, widgetId } = payload;\n const widget = this.widgets[widgetId];\n Vue.set(widget.conditionalVersions, conditionUuid, {\n ...widget.conditionalVersions[DEFAULT_CONDITION_ID],\n });\n },\n\n /**\n * Removes provided conditions from provided widgets\n */\n removeWidgetConditions(args: {\n widgetIds: string[];\n conditionUuids: string[];\n }) {\n args.widgetIds.forEach((widgetId) => {\n const widget = this.widgets[widgetId];\n if (typeof widget === \"undefined\") return;\n args.conditionUuids.forEach((id) => {\n Vue.delete(widget.conditionalVersions, id);\n });\n });\n },\n\n // ---------------\n // ACTIONS\n // ---------------\n\n // TODO: This doesn't get/set state.\n createApp(payload: {\n size: Size;\n name: string;\n ianaTimeZone: string;\n }): Promise {\n const { size, name, ianaTimeZone } = payload;\n\n const data = {\n name: name || \"DefaultAppName\",\n width: size.w,\n height: size.h,\n model: {\n appSchemaVersion: appSchemaVersion,\n widgets: {},\n custom: {},\n },\n feedDeliveryMethod: FeedDeliveryMethod.Html,\n ianaTimeZone: ianaTimeZone,\n };\n return api.post(\"apps\", data).then((response) => {\n return response.uuid;\n });\n },\n\n // TODO: This doesn't get/set state.\n async createFeed(payload: {\n appUuid: string;\n name: string;\n }): Promise {\n return api.post(`apps/${payload.appUuid}/feed`, {\n name: payload.name,\n deliveryMethod: FeedDeliveryMethod.Html,\n });\n },\n\n async updateAppName(payload: { id: string; name: string }) {\n return api.get(`apps/${payload.id}`).then((app) => {\n app.name = payload.name;\n return api.put(`apps/${payload.id}`, app).then((res) => {\n if (res) {\n this.name = res.name;\n this.checksum = res.checksum;\n this.modifiedAt = res.modifiedAt;\n }\n });\n });\n },\n\n // TODO: This references a 'data' module that doesn't exist in Pinia code yet.\n async updateApp() {\n if (!this.uuid) {\n return Promise.resolve();\n }\n\n // Use queue to ensure only one call to updateApp is made at a time.\n return Queue.enqueue(() => {\n // console.log(\"Enqueueing a promise...\");\n\n /**\n * Remove any duplicates from dataBindings array,\n * just in case any issue has occurred that caused one to be incorrectly added.\n * We want to remove any dupes even if query or dataUuid is different, as long as property, widgetId and conditionUuid match.\n * This avoids triggering a backend issue on the save action.\n */\n const dataBindings: DataBinding[] = uniqWith(\n removeReactivity(this.dataBindings),\n (db, otherDb) => {\n const sameWidgetId = db.widgetId === otherDb.widgetId;\n const sameConditionuuid =\n db.conditionUuid === otherDb.conditionUuid;\n // We want to allow for multiple bindings to same widget under same conditionUuid of the 'content' property\n const sameProperty =\n db.property !== \"content\" && db.property === otherDb.property;\n\n return sameWidgetId && sameConditionuuid && sameProperty;\n }\n );\n\n const payload: Partial = {\n appVersionUuid: this.appVersionUuid,\n width: this.width,\n height: this.height,\n dataBindings,\n model: {\n appSchemaVersion: this.appSchemaVersion,\n widgets: removeReactivity(this.widgets),\n parents: removeReactivity(this.parents),\n custom: removeReactivity(this.custom),\n },\n name: this.name,\n ianaTimeZone: this.ianaTimeZone,\n checksum: this.checksum,\n modifiedAt: this.modifiedAt,\n };\n\n if (typeof this.fallbackImageTimeoutSec === \"number\") {\n payload.fallbackImageTimeoutSec = this.fallbackImageTimeoutSec;\n }\n\n if (typeof this.introImageTimeoutSec === \"number\") {\n payload.introImageTimeoutSec = this.introImageTimeoutSec;\n }\n\n return api.put(`apps/${this.uuid}`, payload);\n })\n .then((response: any) => {\n if (response) {\n // Do this to handle case of pasting a repeater with a filter;\n // We need to use dataBinding from the server, that is updated with new filterUuid.\n\n this.setDataBindings(response.dataBindings);\n\n this.setChecksum({\n checksum: response.checksum as number,\n modifiedAt: response.modifiedAt as string,\n });\n }\n })\n .finally(() => {\n // console.log(\"Finally resolved.\");\n this.setHasUnsavedChanges(false);\n });\n },\n\n async loadPublishMeta(appId: string) {\n return api.get(`apps/${appId}`).then(async (app) => {\n this.publishMeta = app.publishMeta;\n this.appVersionUuid = app.appVersionUuid;\n // console.log(\"publishing app...\", app);\n if (app.dataBindings) {\n this.dataBindings = app.dataBindings;\n }\n });\n },\n\n async loadApp(appId: string) {\n return api.get(`apps/${appId}`).then(async (app) => {\n if (!app.model) {\n app.model = {\n appSchemaVersion: 1,\n widgets: {},\n parents: {},\n custom: { fonts: [], userUploadedAssets: [] },\n };\n }\n\n applyModelMigrations(app, \"edit\");\n\n pruneUnusedDataBindings(app);\n\n // This is formerly \"initStateWith\"\n this.hasUnsavedChanges = false;\n Object.keys(app).forEach((key) => {\n if (key === \"dataBindings\") {\n this.setDataBindings(app.dataBindings);\n } else if (key === \"model\") {\n this.widgets = app.model.widgets as Record<\n string,\n WidgetWithConditions\n >;\n this.parents = app.model.parents;\n this.custom = app.model.custom;\n this.appSchemaVersion = app.model.appSchemaVersion;\n this.selections = [];\n } else {\n (this as any)[key] = (app as any)[key];\n }\n });\n\n /**\n * Store initial widget data on savedAppInfo so that we can use it when user presses undo, clearing out state\n */\n const appInfo: SavedAppInfo = {\n appSchemaVersion: app.model.appSchemaVersion,\n name: app.name,\n widgets: purgeNullValues(app.model.widgets),\n parents: app.model.parents || {},\n dataBindings: app.dataBindings,\n data: {}, // rootState.data.data, TODO: UNCOMMENT\n custom: app.model.custom,\n };\n\n this.resetUndoRedoState(appInfo);\n\n // Load initial widget data\n await useAppDataStore().initializeWidgetData(app.uuid);\n\n // Reset the state of the connection editor since it is scoped to the currently edited app\n useConnectionEditorStore().$reset();\n\n // No need to await the following request. We can do it in the background.\n useConnectionsStore().initializeConnections(app.uuid);\n return app;\n });\n },\n\n removeWidgetsAction(payload: { widgetIds: string[] }) {\n const { widgetIds } = payload;\n const conditionsStore = useConditionGroupsStore();\n\n // Collect any children wids that will also be deleted, for the condition group deletion step\n const allWidsToRemove = [\n ...widgetIds,\n ...widgetIds\n .filter((wid) => wid in this.parents)\n .flatMap((wid) => this.parents[wid]),\n ];\n this.removeWidgets({ wids: widgetIds });\n\n // Remove condition groups, reset undo/redo\n const widgetsWithConditions = allWidsToRemove\n .map((wid) => {\n const cg = conditionsStore.conditionGroups.find((cg) =>\n cg.widgets.map((w) => w.widgetId).includes(wid)\n );\n if (!cg) return undefined;\n return { widgetId: wid, conditionGroupUuid: cg.uuid };\n })\n .filter((cg) => cg !== undefined);\n\n if (widgetsWithConditions.length === 0) return;\n\n Promise.all(\n widgetsWithConditions.map((data) =>\n conditionsStore.removeConditionGroup({\n appUuid: this.uuid,\n conditionGroupUuid: data?.conditionGroupUuid || \"\",\n widgetId: data?.widgetId || \"\",\n })\n )\n ).finally(() => {\n // Clear undo/redo state, because user will not be able to undo through a condition group deletion\n // this.storeInitialAppState();\n this.resetUndoRedoState();\n });\n },\n\n removeRepeaterBinding(\n payload: {\n widgetIds: string[];\n dataBinding: DataBinding;\n },\n NO_UNDO?: NO_UNDO\n ) {\n const { widgetIds, dataBinding } = payload;\n\n this.removeWidgets(\n {\n wids: widgetIds,\n },\n NO_UNDO\n );\n\n this.removeDataBinding(dataBinding, \"NO_UNDO\");\n },\n\n addWidgetAction(payload: {\n widget: Widget;\n preserveParentId?: boolean;\n track?: boolean;\n }) {\n const { widget, preserveParentId, track } = payload;\n // const z = getMaxZByParentId(state);\n // widget.z = z + 1;\n\n // Add parentId here so that it passes in as payload to addWidget mutation\n // But if payload already has it, avoid this\n if (!preserveParentId) {\n widget.parentId = this.editingContext.parentId || BASE_PARENT_ID;\n }\n\n this.addWidget(widget, track === false ? \"NO_UNDO\" : undefined);\n },\n\n updateBackgroundImage(payload: {\n asset: SavedAsset;\n wid: string;\n conditionUuid: string;\n parentWidgetId?: string;\n }) {\n const { asset, wid, conditionUuid, parentWidgetId } = payload;\n\n // Two issues when adding from Datasetexplorer: no uuid, and addAsset fails\n\n const binding: DataBinding = {\n dataUuid: asset.uuid, // asset's uuid\n bindingType: \"Asset\",\n widgetId: wid,\n property: \"backgroundImageUrl\",\n conditionUuid,\n };\n\n binding.parentWidgetId = parentWidgetId\n ? parentWidgetId\n : this.editingContext.parentId;\n\n // If there is already an asset binding associated with this Conditional Widget Version, must delete it\n const existingBinding = {\n widgetId: wid,\n property: \"backgroundImageUrl\",\n conditionUuid,\n };\n\n this.removeDataBinding(existingBinding, \"NO_UNDO\");\n\n const props = {\n backgroundImageW: asset.width,\n backgroundImageH: asset.height,\n };\n\n this.setWidgetProps([wid], props, \"NO_UNDO\");\n this.addAsset(asset); // Maybe just ignore this in undo/redo for now?\n\n this.addDataBinding(binding, \"NO_UNDO\");\n },\n\n updateImageSource(payload: {\n asset: SavedAsset;\n wid: string;\n conditionUuid: string;\n parentWidgetId?: string;\n }) {\n const { asset, wid, conditionUuid, parentWidgetId } = payload;\n\n const binding: DataBinding = {\n dataUuid: asset.uuid,\n bindingType: \"Asset\",\n widgetId: wid,\n property: \"url\",\n conditionUuid,\n };\n\n binding.parentWidgetId = parentWidgetId\n ? parentWidgetId\n : this.editingContext.parentId;\n // console.log(\"update image source\", conditionUuid);\n\n // If there is already an asset binding associated with this Conditional Widget Version, must delete it\n const existingBinding = { widgetId: wid, property: \"url\", conditionUuid };\n\n this.removeDataBinding(existingBinding, \"NO_UNDO\");\n\n this.addAsset(asset);\n\n this.addDataBinding(binding, \"NO_UNDO\");\n },\n\n async addImageComponent(payload: {\n asset: SavedAsset;\n boundingBox: Rectangle;\n parentWidgetId?: string;\n newWidgetId: string;\n }): Promise {\n const { asset, boundingBox, parentWidgetId, newWidgetId } = payload;\n\n const widget = CreateImage({\n name: asset.name,\n x: boundingBox.x,\n y: boundingBox.y,\n w: boundingBox.w,\n h: boundingBox.h,\n lockAspect: true,\n canScaleX: true,\n canScaleY: true,\n wid: newWidgetId,\n });\n\n const binding: DataBinding = {\n dataUuid: asset.uuid, // asset's uuid\n bindingType: \"Asset\",\n widgetId: widget.wid,\n property: \"url\",\n conditionUuid: DEFAULT_CONDITION_ID,\n };\n\n binding.parentWidgetId = parentWidgetId\n ? parentWidgetId\n : this.editingContext?.parentId;\n // NOTE: Perhaps we must update app with the bindings?\n\n const addWidgetPayload: any = {\n widget,\n track: false,\n };\n\n // If we are dropping node into a repeater...\n if (parentWidgetId) {\n addWidgetPayload.preserveParentId = true;\n addWidgetPayload.widget.parentId = parentWidgetId;\n }\n\n await this.addWidgetAction(addWidgetPayload);\n\n this.addAsset(asset);\n\n this.addDataBinding(binding, \"NO_UNDO\");\n this.replaceSelections([widget.wid]);\n },\n\n async addVideoComponent(payload: {\n asset: SavedAsset;\n boundingBox: Rectangle;\n parentWidgetId?: string;\n newWidgetId: string;\n }): Promise {\n const { asset, boundingBox, parentWidgetId, newWidgetId } = payload;\n\n const widget = CreateVideo({\n name: asset.name,\n x: boundingBox.x,\n y: boundingBox.y,\n w: boundingBox.w,\n h: boundingBox.h,\n lockAspect: true,\n canScaleX: true,\n canScaleY: true,\n wid: newWidgetId,\n mimeType: asset.type,\n });\n\n const binding: DataBinding = {\n dataUuid: asset.uuid, // asset's uuid\n bindingType: \"Asset\",\n widgetId: widget.wid,\n property: \"url\",\n conditionUuid: DEFAULT_CONDITION_ID,\n };\n\n binding.parentWidgetId = parentWidgetId\n ? parentWidgetId\n : this.editingContext?.parentId;\n // NOTE: Perhaps we must update app with the bindings?\n\n const addWidgetPayload: any = {\n widget,\n track: false,\n };\n\n // If we are dropping node into a repeater...\n if (parentWidgetId) {\n addWidgetPayload.preserveParentId = true;\n addWidgetPayload.widget.parentId = parentWidgetId;\n }\n\n await this.addWidgetAction(addWidgetPayload);\n\n this.addAsset(asset);\n\n this.addDataBinding(binding, \"NO_UNDO\");\n this.replaceSelections([widget.wid]);\n },\n\n // ----------------------------------------------------\n // DATA BINDINGS STUFF\n // ----------------------------------------------------\n\n setDataBindings(dataBindings: DataBinding[] = []) {\n Vue.set(this, \"dataBindings\", dataBindings);\n },\n\n addDataBinding(binding: DataBinding, NO_UNDO?: NO_UNDO) {\n if (!(\"conditionUuid\" in binding)) {\n binding.conditionUuid = DEFAULT_CONDITION_ID;\n }\n this.dataBindings.push(binding);\n this.hasUnsavedChanges = true;\n },\n\n // When user disconnects a data source, backend will send back list of new bindings that point at new manual data source\n // These must replace their existing counterparts in the state.dataBindings array\n replaceDataBindings(args: { bindings: DataBinding[] | null }) {\n const { bindings } = args;\n if (bindings) {\n bindings.forEach((binding) => {\n const idx = this.dataBindings.findIndex(\n (b) =>\n b.bindingType === binding.bindingType &&\n b.widgetId === binding.widgetId &&\n b.parentWidgetId === binding.parentWidgetId &&\n b.dataName === binding.dataName\n );\n this.dataBindings.splice(idx, 1, binding);\n });\n }\n },\n\n updateDataBinding(binding: DataBinding) {\n const indexes = this.dataBindings.map((db, index) => {\n const matchesConditionUuid = binding.conditionUuid\n ? db.conditionUuid === binding.conditionUuid\n : true;\n if (\n db.widgetId === binding.widgetId &&\n db.property === binding.property &&\n matchesConditionUuid\n ) {\n return index;\n }\n });\n this.hasUnsavedChanges = true;\n indexes.forEach((index) => {\n if (typeof index === \"number\") {\n this.dataBindings.splice(index, 1);\n }\n });\n this.dataBindings.push(binding);\n },\n\n async undoableRemapAction(payload: RemapCollectionPayload) {\n const { replacements } = payload;\n\n // Remap the collection bindings\n replacements.forEach((replacement) => {\n const oldBinding = replacement.oldBinding;\n const newBinding = replacement.newBinding;\n\n // Because bindings don't have a uuid (primary key) we search with multiple properties\n const existingBindingIndex = this.dataBindings.findIndex(\n (db) =>\n db.bindingType === oldBinding.bindingType &&\n db.dataUuid === oldBinding.dataUuid &&\n db.dataConnectionUuid === oldBinding.dataConnectionUuid &&\n db.property === oldBinding.property &&\n db.widgetId === oldBinding.widgetId\n );\n\n if (existingBindingIndex === -1) {\n console.log(JSON.stringify(oldBinding, null, 2));\n throw new Error(\"unable to find existing binding to replace\");\n }\n\n if (typeof newBinding === \"undefined\") {\n /**\n * If the new binding is 'undefined' that means the user did\n * not select a replacement column from their new dataset and\n * the existing binding should be deleted.\n */\n this.dataBindings.splice(existingBindingIndex, 1);\n } else {\n // Copy all properties from the new binding onto the existing one.\n const binding = this.dataBindings[existingBindingIndex];\n Object.keys(newBinding).forEach((key) => {\n (binding as any)[key] = (newBinding as any)[key];\n });\n }\n });\n\n // Update Text bindings\n this.updateTextBindings({ replacements });\n\n // Gather all databound widgets where the user did not provide a replacement\n const deletedBindings = payload.replacements.filter(\n (r) => typeof r.newBinding === \"undefined\"\n );\n\n //////////////////////////////////////////////////////////////////////////////////////////\n // DELETE EMPTY TEXT & IMAGE WIDGETS\n //////////////////////////////////////////////////////////////////////////////////////////\n\n const conditions = useConditionGroupsStore().activeWidgetConditionsMap;\n\n // Find any text widgets which are now \"empty\" and remove them.\n const emptyTextWidgetIds = deletedBindings\n .map((r) => this.widgets[r.oldBinding.widgetId])\n .map((w) => getActiveWidget(w, conditions))\n .filter(\n (w) =>\n w.type === \"Text\" && isTextContentEmpty((w as TextOptions).content)\n )\n .map((w) => w.wid);\n\n const emptyImageWidgets = deletedBindings\n .map((r) => this.widgets[r.oldBinding.widgetId])\n .map((w) => getActiveWidget(w, conditions))\n .filter((w) => {\n // Return the image widgets which now have zero dataBindings.\n return (\n w.type === \"Image\" &&\n this.dataBindings.filter((db) => db.widgetId === w.wid).length === 0\n );\n })\n .map((w) => w.wid);\n\n const widgetsToDelete: string[] =\n emptyTextWidgetIds.concat(emptyImageWidgets);\n\n if (widgetsToDelete.length > 0) {\n this.removeWidgets({ wids: widgetsToDelete }, \"NO_UNDO\");\n }\n\n //////////////////////////////////////////////////////////////////////////////////////////\n // RESET DEFAULT PROPERTIES\n //////////////////////////////////////////////////////////////////////////////////////////\n\n // Loop through each one and set the default binding value for that widget property\n const widgetProps = deletedBindings\n // Filter out any widgets that were _just_ deleted\n .filter((r) => !widgetsToDelete.includes(r.oldBinding.widgetId))\n // Generate a mapping of widget, type, property\n .map((r) => {\n const w = this.widgets[r.oldBinding.widgetId];\n return {\n property: r.oldBinding.property,\n widgetId: w.wid,\n type: w.type,\n };\n })\n // Remove any text content bindings (since they were already dealt with)\n .filter((x) => !(x.type === \"Text\" && x.property === \"content\"));\n\n if (widgetProps.length > 0) {\n this.resetWidgetProperties({ properties: widgetProps });\n }\n },\n\n /**\n * Called when a user replaces a connection that is bound to a Repeater or Slide.\n * This updates\n * - the `data` binding for the repeater/slide\n * - any bindings for child widgets bound to data in the replaced connection\n * including any text token bindings\n * @returns\n */\n async remapCollection(payload: RemapCollectionPayload) {\n const { replacements, newConnection } = payload;\n\n this.undoableRemapAction(payload);\n\n // Make sure we invalidate cache for both connections, to ensure widget fetches fresh data after remap:\n useAppDataStore().invalidateCacheForConnection(newConnection?.uuid);\n useAppDataStore().invalidateCacheForConnection(\n replacements[0]?.oldBinding?.dataConnectionUuid || \"\"\n );\n\n // Ensure info about new connection is reflected in left-hand panel, if it is open to \"data\" submenu:\n const connectionEditor = useConnectionEditorStore();\n connectionEditor.setConnection(newConnection);\n connectionEditor.previouslySelectedConnection = newConnection;\n useConnectionDataStore().fetchConnectionData({\n connectionId: newConnection?.uuid,\n });\n\n return this.updateApp();\n },\n\n removeDataBinding(\n payload: {\n widgetId?: string;\n property: string;\n conditionUuid?: string;\n dataUuid?: string;\n },\n NO_UNDO?: NO_UNDO\n ) {\n // const { widgetId, property, conditionId } = payload;\n\n const index = this.dataBindings.findIndex((db) => {\n let result = true;\n Object.keys(payload).forEach((key) => {\n if (key in db && (db as any)[key] !== (payload as any)[key]) {\n result = false;\n }\n });\n return result;\n });\n if (index > -1) {\n this.hasUnsavedChanges = true;\n this.dataBindings.splice(index, 1);\n }\n },\n\n cloneDataBindingsForNewCondition(args: {\n widgetId: string;\n conditionUuid: string;\n }) {\n const copies: DataBinding[] = [];\n\n this.dataBindings.forEach((db) => {\n /**\n * This method may have been called immediately after\n * adding the first condition for a widget. If this case\n * existing databindings will not have a conditionUuid.\n *\n * So we should first loop through and ensure that all\n * data bindings that should have a condition id, have one.\n */\n if (\n db.widgetId === args.widgetId &&\n isConditionalBindingType(db.bindingType)\n ) {\n // Set the default condition id for the binding\n if (typeof db.conditionUuid === \"undefined\") {\n db.conditionUuid = DEFAULT_CONDITION_ID;\n }\n\n // Make a copy\n copies.push(\n Object.assign({}, db, {\n conditionUuid: args.conditionUuid,\n })\n );\n }\n });\n\n copies.forEach((db) => {\n this.dataBindings.push(db);\n });\n },\n\n removeDataBindingsForConditions(args: {\n conditionUuids: string[];\n widgetId?: string;\n }) {\n if (\n typeof args.conditionUuids === \"undefined\" ||\n args.conditionUuids.length === 0\n ) {\n return;\n }\n\n // Replace entire array of dataBindings with a filtered list\n // https://v2.vuejs.org/v2/guide/list.html#Replacing-an-Array\n this.dataBindings = this.dataBindings.filter((db) => {\n if (typeof db.conditionUuid === \"undefined\") {\n return true;\n }\n let shouldRemoveBinding = args.conditionUuids.includes(\n db.conditionUuid\n );\n\n // If a widgetId is passed in, only filter out the binding if it is attached to that widget\n if (args.widgetId) {\n shouldRemoveBinding =\n shouldRemoveBinding && args.widgetId === db.widgetId;\n }\n return !shouldRemoveBinding;\n });\n },\n\n removeDataTokenBinding(payload: {\n widgetId: string;\n dataUuid: string;\n conditionUuid?: string;\n }) {\n const { widgetId, dataUuid, conditionUuid } = payload;\n const index = this.dataBindings.findIndex((db) => {\n let isTarget = db.widgetId === widgetId && db.dataUuid === dataUuid;\n if (conditionUuid) {\n isTarget = isTarget && db.conditionUuid === conditionUuid;\n }\n return isTarget;\n });\n\n if (index > -1) {\n this.dataBindings.splice(index, 1);\n }\n },\n\n bindingsForComponent(searchOptions: BindingsSearchOptions) {\n if (searchOptions.widgetId === undefined) {\n return [];\n }\n const bindings = this.dataBindings.filter((db) => {\n return (\n db.widgetId === searchOptions.widgetId &&\n (typeof searchOptions.dataConnectionUuid !== \"undefined\"\n ? db.dataConnectionUuid === searchOptions.dataConnectionUuid\n : true) &&\n (typeof searchOptions.dataParentUuid !== \"undefined\"\n ? db.dataParentUuid === searchOptions.dataParentUuid\n : true) &&\n (typeof searchOptions.bindingType !== \"undefined\"\n ? db.bindingType === searchOptions.bindingType\n : true) &&\n (typeof searchOptions.property !== \"undefined\"\n ? db.property === searchOptions.property\n : true) &&\n (typeof searchOptions.conditionUuid !== \"undefined\"\n ? db.conditionUuid === searchOptions.conditionUuid\n : true)\n );\n });\n\n // TODO: determine where we want connections to live\n const connectionSources = (\n useConnectionsStore().connections as DataConnection[]\n ).reduce(\n (obj: any, c: DataConnection) =>\n Object.assign(obj, { [c.uuid]: c.schemaType }),\n {}\n );\n\n return bindings.filter((b) => {\n return typeof searchOptions.schemaType !== \"undefined\" &&\n typeof b.dataConnectionUuid === \"string\"\n ? connectionSources[b.dataConnectionUuid] === searchOptions.schemaType\n : true;\n });\n },\n\n // These two are from connection.ts\n\n replaceDataBinding(args: { connection: DataConnection; widget: Widget }) {\n const appEditor = useAppEditorStore();\n const dataBindings = appEditor.dataBindings;\n const { connection, widget } = args;\n const binding = dataBindings.find(\n (db) => db.property === \"data\" && db.widgetId === widget.wid\n );\n\n if (binding === undefined) {\n return;\n }\n\n const newBinding: DataBinding = {\n ...binding,\n dataConnectionUuid: connection.uuid,\n dataUuid: connection.nodeSets[0].uuid as string,\n };\n appEditor.updateDataBinding(newBinding);\n\n return useConnectionDataStore().initializeConnection(\n connection as DataConnection\n );\n },\n\n // Used in Preview and SetupComplete (and now also in ScalarSelect)\n createDataBinding(args: {\n connection: DataConnection;\n widget: Widget;\n property: DataBindingProperty;\n isScalar?: boolean; // Technically redundant; presence of query indicates same thing\n query?: string;\n dataUuid?: string;\n conditionUuid?: string;\n }) {\n // Starting to feel like createScalarBinding could be own thing...\n const {\n connection,\n widget,\n property,\n isScalar,\n query,\n dataUuid,\n\n conditionUuid,\n } = args;\n\n const bindingType = isScalar\n ? \"Scalar\"\n : widget.type === \"Repeater\"\n ? \"DataSetParent\"\n : \"DataSet\";\n\n const uuid = isScalar\n ? (dataUuid as string)\n : (connection.nodeSets[0].uuid as string);\n\n // TODO: pass in real name somehow...\n const dataName = isScalar ? query : connection.nodeSets[0].name;\n\n const binding: DataBinding = {\n widgetId: widget.wid,\n property,\n dataUuid: uuid,\n dataName,\n dataConnectionUuid: connection.uuid,\n bindingType,\n parentWidgetId: BASE_PARENT_ID,\n };\n if (isScalar) {\n binding.query = query;\n binding.conditionUuid = conditionUuid;\n }\n\n // console.log(JSON.stringify(binding));\n\n const appEditor = useAppEditorStore();\n appEditor.addDataBinding(binding);\n\n // Calling initializeConnection with scalar bindings breaks things.. So we use refreshWidgetData instead\n if (isScalar) return;\n\n // This is what must be called to get datasetchooser and datasetexplorer to recognize data\n // (pushes it on to this.connections, AND loads data)\n const store = useConnectionDataStore();\n store.initializeConnection(connection as DataConnection);\n },\n\n createDataToken(options: CreateDataTokenOptions) {\n const {\n widgetId,\n conditionId,\n connectionId,\n parentWidgetId,\n textTokenUuid,\n isScalar,\n content,\n query,\n dataParentUuid,\n dataUuid,\n shouldCreateRepeaterBinding,\n } = options;\n\n const binding: DataBinding = {\n widgetId: widgetId,\n property: isScalar\n ? (`content_${textTokenUuid}` as DataBindingProperty)\n : \"content\",\n parentWidgetId: parentWidgetId ? parentWidgetId : BASE_PARENT_ID,\n dataUuid: dataUuid,\n query: query,\n bindingType: isScalar ? \"Scalar\" : \"DataSetNode\",\n dataConnectionUuid: connectionId,\n conditionUuid: conditionId,\n };\n if (dataParentUuid && !isScalar) {\n binding.dataParentUuid = dataParentUuid;\n delete binding.query;\n }\n\n if (shouldCreateRepeaterBinding) {\n const repeaterBinding: DataBinding = {\n dataConnectionUuid: connectionId,\n dataUuid: dataParentUuid as string,\n bindingType: \"DataSetParent\",\n widgetId: parentWidgetId,\n parentWidgetId: BASE_PARENT_ID,\n property: \"data\",\n };\n\n this.addDataBinding(repeaterBinding, \"NO_UNDO\");\n }\n\n this.addDataBinding(binding as DataBinding, \"NO_UNDO\");\n this.setWidgetProps([widgetId], { content }, \"NO_UNDO\");\n },\n\n /**\n * Point of entry for all undo-able \"dragDrop\"-initiated actions.\n * (Create widget from usage prompt, or from bare drop event.)\n * Handles creating a repeater if needed, to house the new widget, and a repeater binding if none exists.\n * All of these actions must be \"chained together\" here for undo/redo to work properly.\n *\n * The binding should be scalar when:\n * - user drops node from Tree connection into Repeater,\n * - user drops node from Tree connection onto canvas,\n * - user drops node from Collection onto canvas and chooses \"use on its own\"\n *\n * And it should be be non-scalar (node binding) when user drops node:\n * - into a non-data-bound Repeater,\n * - or from the same collection that is already bound to the Repeater.\n * - or from a foreign collection.\n */\n createDynamicWidget(\n args: {\n isNewRepeater: boolean;\n hoveredPhotoDropWidgetId: string;\n hoveredPhotoDropWidgetConditionId?: string;\n isScalar: boolean;\n draggingInfo: DraggingInfo | null;\n widgetId: string;\n parentWidgetId?: string | null;\n },\n newRepeaterInfo: {\n rows: number;\n columns: number;\n isSlide: boolean;\n shouldCycle: boolean;\n repeaterWidgetId: string;\n } | null,\n repeaterBindingWidgetId: string | null\n ) {\n const {\n hoveredPhotoDropWidgetId,\n hoveredPhotoDropWidgetConditionId,\n isScalar,\n isNewRepeater,\n draggingInfo,\n widgetId,\n parentWidgetId,\n } = args;\n\n const connectionEditor = useConnectionEditorStore();\n\n // console.log(\"create dynamic widget\", draggingInfo);\n\n let repeaterWidget: Widget | null = null;\n\n if (!!newRepeaterInfo) {\n const { isSlide, shouldCycle, rows, columns, repeaterWidgetId } =\n newRepeaterInfo;\n\n const options: Partial = createRepeaterDimensions(\n this.artboard\n );\n if (isSlide) {\n options.columns = 1;\n options.rows = 1;\n }\n if (rows && columns) {\n options.rows = rows;\n options.columns = columns;\n }\n if (shouldCycle === false) {\n options.cycleContent = false;\n }\n options.wid = repeaterWidgetId;\n repeaterWidget = CreateRepeater(options);\n\n this.addWidgetAction({\n widget: repeaterWidget as unknown as Widget,\n track: false,\n });\n }\n\n // ======\n\n /**\n * This is a bandaid fix for a bug where selectedConnection.nodeSets is cleared out to null when user opens Data manager view.\n * We can fallback to relying on connectionData.uuid based on the connection that is live in the left hand panel.\n */\n const dataUuid =\n draggingInfo?.dataParentUuid ||\n (draggingInfo as any)?.nodeSetUuid ||\n useConnectionEditorStore().connection?.nodeSets?.[0]?.uuid ||\n useConnectionDataStore().connectionData?.uuid;\n\n if (repeaterBindingWidgetId || repeaterWidget) {\n const repeaterBinding: DataBinding = {\n dataConnectionUuid: draggingInfo?.connectionUuid,\n dataUuid: dataUuid as string,\n bindingType: \"DataSetParent\",\n widgetId: repeaterBindingWidgetId\n ? repeaterBindingWidgetId\n : repeaterWidget?.wid ?? \"\",\n parentWidgetId: BASE_PARENT_ID,\n property: \"data\",\n };\n\n this.addDataBinding(repeaterBinding, \"NO_UNDO\");\n }\n\n // ======\n\n if (!draggingInfo) return;\n\n const { x, y } = draggingInfo?.dropPoint || { x: 0, y: 0 };\n\n const parentWid =\n repeaterWidget?.wid || parentWidgetId || this.editingContext?.parentId;\n const artboard = this.artboard;\n\n let widget: any = {\n wid: widgetId,\n x: x,\n y: y,\n };\n\n if (isNewRepeater) {\n // NOTE: Maybe center in repeater? Not sure\n widget.x = 20;\n widget.y = 20;\n }\n\n const binding: Partial = {\n dataConnectionUuid:\n draggingInfo?.connectionUuid ?? connectionEditor.connection?.uuid,\n dataUuid: draggingInfo?.dataUuid || \"\",\n bindingType: isScalar ? \"Scalar\" : \"DataSetNode\",\n widgetId: widget.wid || \"\",\n parentWidgetId: parentWid,\n conditionUuid: DEFAULT_CONDITION_ID,\n };\n\n if (isScalar) {\n binding.query = draggingInfo?.query;\n } else {\n binding.dataParentUuid =\n draggingInfo?.dataParentUuid ||\n connectionEditor.connection?.nodeSets?.[0]?.uuid;\n }\n\n if (typeof binding.parentWidgetId === \"undefined\") {\n delete binding.parentWidgetId;\n }\n\n const parentRepeater = this.widgetById(parentWid);\n\n let cellWidth = artboard.w * 0.3;\n\n // widgetById returns {} if none is found, so use presence of \"wid\" to test existence\n if (parentRepeater !== undefined && \"wid\" in parentRepeater) {\n const { w, scaleX, columns } =\n parentRepeater as unknown as RepeaterOptions;\n cellWidth = (w * scaleX) / columns;\n }\n\n let skipWidgetCreation = false;\n\n if (hoveredPhotoDropWidgetId) {\n const targetWidget = this.widgetById(hoveredPhotoDropWidgetId);\n const isBackgroundImage = targetWidget?.type !== \"Image\";\n\n binding.property = isBackgroundImage ? \"backgroundImageUrl\" : \"url\";\n binding.widgetId = hoveredPhotoDropWidgetId;\n binding.conditionUuid = hoveredPhotoDropWidgetConditionId;\n\n skipWidgetCreation = true;\n }\n\n // Handle data type of Object... sometimes actually a Datetime.. like Weather data tree\n let dataType = draggingInfo?.dataType;\n\n if (dataType === \"Object\" && draggingInfo?.children?.[0]) {\n dataType = draggingInfo?.children[0].dataType;\n }\n\n if (!skipWidgetCreation) {\n switch (dataType) {\n case \"Bool\":\n case \"String\":\n case \"Number\":\n case \"Color\":\n case \"Url\":\n case \"Object\":\n binding.property = isScalar\n ? (`content_${makeId()}` as DataBindingProperty)\n : \"content\";\n widget = CreateText(widget);\n break;\n case \"ImageUrl\":\n case \"ImageUpload\":\n widget = CreateImage(widget);\n binding.property = \"url\";\n break;\n\n case \"Date\":\n binding.property = \"datetimeValue\";\n widget.isDate = true;\n widget = CreateDatetime(widget);\n break;\n case \"Time\":\n binding.property = \"datetimeValue\";\n widget.isTime = true;\n widget.datetimeFormat = JSON.stringify({\n hour: \"numeric\",\n minute: \"numeric\",\n });\n widget = CreateDatetime(widget);\n break;\n case \"DateTime\":\n binding.property = \"datetimeValue\";\n widget = CreateDatetime(widget);\n break;\n }\n\n if (\n ![\"ImageUrl\", \"ImageUpload\", \"Date\", \"Time\", \"DateTime\"].includes(\n dataType || \"\"\n )\n ) {\n widget.content = makeDynamicTextContent(\n binding.dataUuid || \"\",\n draggingInfo?.formattedValue || \"\"\n );\n\n // Stretch: Perhaps double width until it contains largest word?\n widget.canScaleY = false;\n widget.w = 3840;\n widget.fontSize = Math.floor(cellWidth / 10);\n // eslint-disable-next-line no-case-declarations\n const size = measureText(widget, undefined, true);\n widget.h = size.h;\n widget.w = cellWidth * 0.5;\n }\n }\n\n const addWidgetPayload: any = {\n widget,\n };\n\n // If we are dropping node into a repeater...\n if (parentWid) {\n addWidgetPayload.preserveParentId = true;\n addWidgetPayload.widget.parentId = parentWid;\n }\n\n let realWidgetId = widget.wid;\n\n if (!skipWidgetCreation) {\n this.addWidgetAction({ ...addWidgetPayload, track: false });\n } else {\n realWidgetId = hoveredPhotoDropWidgetId;\n\n /**\n * Handle case of user choosing \"New Repeater\" --\n * Requires us to *transport* the existing target widget into the new repeater.\n */\n if (isNewRepeater) {\n this.addToNewRepeater(\n {\n wid: hoveredPhotoDropWidgetId as string,\n parentWidgetId: parentWid as string,\n },\n \"NO_UNDO\"\n );\n }\n\n // Remove data binding if it exists\n this.removeDataBinding(\n {\n widgetId: hoveredPhotoDropWidgetId as string,\n property: binding.property as DataBindingProperty,\n conditionUuid: hoveredPhotoDropWidgetConditionId,\n },\n \"NO_UNDO\"\n );\n\n if (binding.property === \"backgroundImageUrl\") {\n const props = {\n backgroundImageW: draggingInfo?.width,\n backgroundImageH: draggingInfo?.height,\n };\n this.setWidgetProps(\n [hoveredPhotoDropWidgetId as string],\n props,\n \"NO_UNDO\"\n );\n }\n }\n\n this.addDataBinding(binding as DataBinding, \"NO_UNDO\");\n this.replaceSelections([realWidgetId]);\n\n return repeaterWidget;\n },\n\n unbindProperty(args: {\n widgetId: string;\n conditionUuid: string;\n propertyName: string;\n value: any;\n }) {\n const { widgetId, conditionUuid, propertyName, value } = args;\n this.setWidgetProps(\n [widgetId],\n {\n [propertyName]: value, // Use freshly-unbound value for static value\n },\n \"NO_UNDO\"\n );\n this.removeDataBinding(\n {\n widgetId: widgetId,\n property: propertyName,\n conditionUuid: conditionUuid,\n },\n \"NO_UNDO\"\n );\n },\n },\n});\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","export class KeyCodes {\n // static BACKSPACE = 8;\n // static SHIFT = 16;\n // static SPACE = 32;\n // static LEFT = 37;\n // static UP = 38;\n // static RIGHT = 39;\n // static DOWN = 40;\n // static DELETE = 46;\n // static PLUS = 187;\n // static MINUS = 189;\n // static C = 67;\n // static V = 86;\n\n static BACKSPACE = \"Backspace\";\n static DELETE = \"Delete\";\n static SHIFT = \"ShiftLeft\"; // or right... so do isShift?\n static SPACE = \"Space\";\n static LEFT = \"ArrowLeft\";\n static RIGHT = \"ArrowRight\";\n static UP = \"ArrowUp\";\n static DOWN = \"ArrowDown\";\n static C = \"KeyC\";\n static V = \"KeyV\";\n static Z = \"KeyZ\";\n static Y = \"KeyY\";\n static G = \"KeyG\";\n static PLUS = \"Equal\";\n static MINUS = \"Minus\";\n static TAB = \"Tab\";\n static ESCAPE = \"Escape\";\n static ZERO = \"Digit0\";\n\n static isArrowKey(value: string) {\n return (\n value === KeyCodes.RIGHT ||\n value === KeyCodes.LEFT ||\n value === KeyCodes.UP ||\n value === KeyCodes.DOWN\n );\n }\n\n static isMultiselectKey(event: KeyboardEvent) {\n // console.log(\"ismult\", this.isMeta(event) || event.shiftKey);\n return this.isMeta(event) || event.shiftKey;\n }\n\n static isMeta(event: KeyboardEvent) {\n const isWindows = navigator.platform.indexOf(\"Win\") > -1;\n return isWindows ? event.ctrlKey : event.metaKey;\n }\n\n static isUndo(event: KeyboardEvent) {\n return this.isMeta(event) && !event.shiftKey && event.code === KeyCodes.Z;\n }\n\n static isRedo(event: KeyboardEvent) {\n return this.isMeta(event) && event.shiftKey && event.code === KeyCodes.Z;\n }\n\n static isCopy(event: KeyboardEvent) {\n return this.isMeta(event) && event.code === KeyCodes.C;\n }\n\n static isPaste(event: KeyboardEvent) {\n return this.isMeta(event) && event.code === KeyCodes.V;\n }\n\n static isGroup(event: KeyboardEvent) {\n return this.isMeta(event) && !event.shiftKey && event.code === KeyCodes.G;\n }\n\n static isUngroup(event: KeyboardEvent) {\n return this.isMeta(event) && event.shiftKey && event.code === KeyCodes.G;\n }\n\n static isArtboardReset(event: KeyboardEvent) {\n return this.isMeta(event) && event.code === KeyCodes.ZERO;\n }\n}\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","/**\n * DON'T OVERUSE THIS FUNCTION!\n *\n * Returns the component instance of a widget in the CanvasRender context.\n *\n * It recursively scans the component tree of the provided node and returns\n * the first component that has an attribute `renderWid` with the value of\n * the supplied `wid`.\n *\n * I specifically added that attribute in the following components\n * - CanvasRender\n * - RepeaterComponent\n * - GroupComponent\n *\n * This feels __really__ janky, but it works and it solved a problem where\n * WidgetResizer needs access to the component instance of the widget\n * so it can call handleResize() on it.\n */\nexport const findWidget = (node: Vue, wid: string): Vue | null => {\n if (node.$attrs?.renderWid === wid) {\n return node;\n }\n const children = node.$children;\n if (children) {\n for (const child of children) {\n const found = findWidget(child, wid);\n if (found) {\n return found;\n }\n }\n }\n return null;\n};\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","export const isFunction = (func: any) => {\n return (\n typeof func === \"function\" ||\n Object.prototype.toString.call(func) === \"[object Function]\"\n );\n};\n\nexport const matchesSelectorToParentElements = (\n el: any,\n selector: string,\n baseNode: any\n) => {\n let node = el;\n\n const matchesSelectorFunc =\n [\n \"matches\",\n \"webkitMatchesSelector\",\n \"mozMatchesSelector\",\n \"msMatchesSelector\",\n \"oMatchesSelector\"\n ].find(func => isFunction(node[func])) || \"\";\n\n if (!isFunction(node[matchesSelectorFunc])) return false;\n\n do {\n if (node[matchesSelectorFunc](selector)) return true;\n if (node === baseNode) return false;\n node = node.parentNode;\n } while (node);\n\n return false;\n};\n\nexport const addEvent = (el: any, event: string, handler: any) => {\n if (!el) {\n return;\n }\n if (el.attachEvent) {\n el.attachEvent(\"on\" + event, handler);\n } else if (el.addEventListener) {\n el.addEventListener(event, handler, true);\n } else {\n el[\"on\" + event] = handler;\n }\n};\n\nexport const removeEvent = (el: any, event: string, handler: any) => {\n if (!el) {\n return;\n }\n if (el.detachEvent) {\n el.detachEvent(\"on\" + event, handler);\n } else if (el.removeEventListener) {\n el.removeEventListener(event, handler, true);\n } else {\n el[\"on\" + event] = null;\n }\n};\n\nexport const MOUSE_EVENTS = {\n start: \"mousedown\",\n move: \"mousemove\",\n stop: \"mouseup\"\n};\n\nexport const TOUCH_EVENTS = {\n start: \"touchstart\",\n move: \"touchmove\",\n stop: \"touchend\"\n};\n\nexport const userSelectNone = {\n userSelect: \"none\",\n MozUserSelect: \"none\",\n WebkitUserSelect: \"none\",\n MsUserSelect: \"none\"\n};\n\nexport const userSelectAuto = {\n userSelect: \"auto\",\n MozUserSelect: \"auto\",\n WebkitUserSelect: \"auto\",\n MsUserSelect: \"auto\"\n};\n","\n\n\n","\n\n\n\n\n","import { Node } from \"@tiptap/vue-2\";\nimport { DATA_TOKEN_TYPE } from \"./index\";\n\nexport interface DataTokenOptions {\n HTMLAttributes: Record;\n}\n\nexport const DataToken = Node.create({\n group: \"inline\",\n inline: true,\n selectable: false,\n atom: true,\n name: DATA_TOKEN_TYPE,\n addAttributes() {\n return {\n uuid: {\n default: null,\n renderHTML: (attributes: any) => {\n return {\n \"data-uuid\": attributes.uuid,\n };\n },\n parseHTML: (element: any) => {\n return {\n uuid: element.getAttribute(\"data-uuid\"),\n };\n },\n },\n text: {\n default: \"\",\n renderHTML: (attributes: any) => {\n return {\n \"data-text\": attributes.text,\n };\n },\n parseHTML: (element: any) => {\n return {\n text: element.getAttribute(\"data-text\"),\n };\n },\n },\n };\n },\n renderHTML({ node, HTMLAttributes }) {\n return [\"span\", HTMLAttributes, node.attrs[\"text\"]];\n },\n parseHTML() {\n return [\n {\n tag: \"span[data-uuid]\",\n },\n ];\n },\n});\n","import { Node } from \"@tiptap/core\";\nimport { DataToken } from \"@/text/DataToken\";\n\nconst Document = Node.create({\n name: \"doc\",\n topNode: true,\n content: \"block+\",\n});\n\nconst Paragraph = Node.create({\n name: \"paragraph\",\n group: \"block\",\n content: \"inline*\",\n parseHTML() {\n return [{ tag: \"p\" }];\n },\n renderHTML({ HTMLAttributes }) {\n return [\"p\", HTMLAttributes, 0];\n },\n});\n\nconst Text = Node.create({\n name: \"text\",\n group: \"inline\",\n});\n\nexport const contentSchema = [Document, Paragraph, Text, DataToken];\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","import { Vue, Component, Watch, Prop } from \"vue-property-decorator\";\n\n@Component({})\nexport default class TooltipMixin extends Vue {\n @Prop({ type: String, default: \"\" }) tooltip: string;\n @Prop(String) tooltipPosition: string;\n @Prop(Boolean) tooltipDisabled: boolean;\n\n tooltipVisible = false;\n private tooltipTimer: any = null;\n\n // TODO: mouse events do not fire on disabled elements, so if you redo into disabled state, tooltip stays open\n // We could just turn off tooltipVis when disabled...\n // I guess that's consistent with what we have now\n // Possibly want to come back and allow tooltip hover on disabled buttons\n\n onMouseEnter() {\n // console.log(\"me\", this.tooltip, this.tooltipDisabled, this.tooltipVisible)\n if (this.tooltip.length > 0) {\n this.tooltipTimer = setTimeout(this.showTip, 150);\n }\n }\n\n onMouseLeave() {\n // console.log(\"ml\");\n this.hideTip();\n clearTimeout(this.tooltipTimer);\n }\n\n showTip() {\n // console.log(\"show tip\");\n if (this.tooltipDisabled) {\n this.tooltipVisible = false;\n return;\n }\n // console.log(\"show tip2\");\n\n this.tooltipVisible = true;\n }\n\n hideTip() {\n this.tooltipVisible = false;\n }\n\n @Watch(\"tooltipDisabled\")\n onTooltipDisabledChanged(isDisabled: boolean) {\n if (isDisabled) {\n this.tooltipVisible = false;\n clearTimeout(this.tooltipTimer);\n }\n }\n\n @Watch(\"disabled\")\n onDisabledChanged(isDisabled: boolean) {\n if (isDisabled) {\n this.tooltipVisible = false;\n clearTimeout(this.tooltipTimer);\n }\n }\n\n destroyed() {\n clearTimeout(this.tooltipTimer);\n }\n\n\n get bgColor() {\n return \"bg-gray-700\";\n }\n\n get borderColor() {\n return \"border-gray-700\";\n }\n\n get pos() {\n switch (this.tooltipPosition) {\n case \"t\":\n case \"r\":\n case \"l\":\n return this.tooltipPosition;\n default:\n return \"b\";\n }\n }\n\n get contentClasses() {\n if (!this.tooltipVisible) {\n return [\"invisible\"];\n }\n switch (this.pos) {\n case \"t\":\n return [this.bgColor, \"tip-offset-t\"];\n case \"r\":\n return [this.bgColor, \"tip-offset-r\"];\n case \"l\":\n return [this.bgColor, \"tip-offset-l\"];\n default:\n return [this.bgColor, \"tip-offset-b\"];\n }\n }\n\n get arrowClasses() {\n if (!this.tooltipVisible) {\n return [\"invisible\"];\n }\n switch (this.pos) {\n case \"t\":\n return [this.borderColor, \"tip-arrow-t\"];\n case \"r\":\n return [this.borderColor, \"tip-arrow-r\"];\n case \"l\":\n return [this.borderColor, \"tip-arrow-l\"];\n default:\n return [this.borderColor, \"tip-arrow-b\"];\n }\n }\n}\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n","\n\n\n","\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n","\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n","\n\n\n","\n\n\n","\n\n\n","var x=String;\nvar create=function() {return {isColorSupported:false,reset:x,bold:x,dim:x,italic:x,underline:x,inverse:x,hidden:x,strikethrough:x,black:x,red:x,green:x,yellow:x,blue:x,magenta:x,cyan:x,white:x,gray:x,bgBlack:x,bgRed:x,bgGreen:x,bgYellow:x,bgBlue:x,bgMagenta:x,bgCyan:x,bgWhite:x}};\nmodule.exports=create();\nmodule.exports.createColors = create;\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nfunction _export(target, all) {\n for(var name in all)Object.defineProperty(target, name, {\n enumerable: true,\n get: all[name]\n });\n}\n_export(exports, {\n dim: ()=>dim,\n default: ()=>_default\n});\nconst _picocolors = /*#__PURE__*/ _interopRequireDefault(require(\"picocolors\"));\nfunction _interopRequireDefault(obj) {\n return obj && obj.__esModule ? obj : {\n default: obj\n };\n}\nlet alreadyShown = new Set();\nfunction log(type, messages, key) {\n if (typeof process !== \"undefined\" && process.env.JEST_WORKER_ID) return;\n if (key && alreadyShown.has(key)) return;\n if (key) alreadyShown.add(key);\n console.warn(\"\");\n messages.forEach((message)=>console.warn(type, \"-\", message));\n}\nfunction dim(input) {\n return _picocolors.default.dim(input);\n}\nconst _default = {\n info (key, messages) {\n log(_picocolors.default.bold(_picocolors.default.cyan(\"info\")), ...Array.isArray(key) ? [\n key\n ] : [\n messages,\n key\n ]);\n },\n warn (key, messages) {\n log(_picocolors.default.bold(_picocolors.default.yellow(\"warn\")), ...Array.isArray(key) ? [\n key\n ] : [\n messages,\n key\n ]);\n },\n risk (key, messages) {\n log(_picocolors.default.bold(_picocolors.default.magenta(\"risk\")), ...Array.isArray(key) ? [\n key\n ] : [\n messages,\n key\n ]);\n }\n};\n","\"use strict\";\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nObject.defineProperty(exports, \"default\", {\n enumerable: true,\n get: ()=>_default\n});\nconst _log = /*#__PURE__*/ _interopRequireDefault(require(\"../util/log\"));\nfunction _interopRequireDefault(obj) {\n return obj && obj.__esModule ? obj : {\n default: obj\n };\n}\nfunction warn({ version , from , to }) {\n _log.default.warn(`${from}-color-renamed`, [\n `As of Tailwind CSS ${version}, \\`${from}\\` has been renamed to \\`${to}\\`.`,\n \"Update your configuration file to silence this warning.\"\n ]);\n}\nconst _default = {\n inherit: \"inherit\",\n current: \"currentColor\",\n transparent: \"transparent\",\n black: \"#000\",\n white: \"#fff\",\n slate: {\n 50: \"#f8fafc\",\n 100: \"#f1f5f9\",\n 200: \"#e2e8f0\",\n 300: \"#cbd5e1\",\n 400: \"#94a3b8\",\n 500: \"#64748b\",\n 600: \"#475569\",\n 700: \"#334155\",\n 800: \"#1e293b\",\n 900: \"#0f172a\",\n 950: \"#020617\"\n },\n gray: {\n 50: \"#f9fafb\",\n 100: \"#f3f4f6\",\n 200: \"#e5e7eb\",\n 300: \"#d1d5db\",\n 400: \"#9ca3af\",\n 500: \"#6b7280\",\n 600: \"#4b5563\",\n 700: \"#374151\",\n 800: \"#1f2937\",\n 900: \"#111827\",\n 950: \"#030712\"\n },\n zinc: {\n 50: \"#fafafa\",\n 100: \"#f4f4f5\",\n 200: \"#e4e4e7\",\n 300: \"#d4d4d8\",\n 400: \"#a1a1aa\",\n 500: \"#71717a\",\n 600: \"#52525b\",\n 700: \"#3f3f46\",\n 800: \"#27272a\",\n 900: \"#18181b\",\n 950: \"#09090b\"\n },\n neutral: {\n 50: \"#fafafa\",\n 100: \"#f5f5f5\",\n 200: \"#e5e5e5\",\n 300: \"#d4d4d4\",\n 400: \"#a3a3a3\",\n 500: \"#737373\",\n 600: \"#525252\",\n 700: \"#404040\",\n 800: \"#262626\",\n 900: \"#171717\",\n 950: \"#0a0a0a\"\n },\n stone: {\n 50: \"#fafaf9\",\n 100: \"#f5f5f4\",\n 200: \"#e7e5e4\",\n 300: \"#d6d3d1\",\n 400: \"#a8a29e\",\n 500: \"#78716c\",\n 600: \"#57534e\",\n 700: \"#44403c\",\n 800: \"#292524\",\n 900: \"#1c1917\",\n 950: \"#0c0a09\"\n },\n red: {\n 50: \"#fef2f2\",\n 100: \"#fee2e2\",\n 200: \"#fecaca\",\n 300: \"#fca5a5\",\n 400: \"#f87171\",\n 500: \"#ef4444\",\n 600: \"#dc2626\",\n 700: \"#b91c1c\",\n 800: \"#991b1b\",\n 900: \"#7f1d1d\",\n 950: \"#450a0a\"\n },\n orange: {\n 50: \"#fff7ed\",\n 100: \"#ffedd5\",\n 200: \"#fed7aa\",\n 300: \"#fdba74\",\n 400: \"#fb923c\",\n 500: \"#f97316\",\n 600: \"#ea580c\",\n 700: \"#c2410c\",\n 800: \"#9a3412\",\n 900: \"#7c2d12\",\n 950: \"#431407\"\n },\n amber: {\n 50: \"#fffbeb\",\n 100: \"#fef3c7\",\n 200: \"#fde68a\",\n 300: \"#fcd34d\",\n 400: \"#fbbf24\",\n 500: \"#f59e0b\",\n 600: \"#d97706\",\n 700: \"#b45309\",\n 800: \"#92400e\",\n 900: \"#78350f\",\n 950: \"#451a03\"\n },\n yellow: {\n 50: \"#fefce8\",\n 100: \"#fef9c3\",\n 200: \"#fef08a\",\n 300: \"#fde047\",\n 400: \"#facc15\",\n 500: \"#eab308\",\n 600: \"#ca8a04\",\n 700: \"#a16207\",\n 800: \"#854d0e\",\n 900: \"#713f12\",\n 950: \"#422006\"\n },\n lime: {\n 50: \"#f7fee7\",\n 100: \"#ecfccb\",\n 200: \"#d9f99d\",\n 300: \"#bef264\",\n 400: \"#a3e635\",\n 500: \"#84cc16\",\n 600: \"#65a30d\",\n 700: \"#4d7c0f\",\n 800: \"#3f6212\",\n 900: \"#365314\",\n 950: \"#1a2e05\"\n },\n green: {\n 50: \"#f0fdf4\",\n 100: \"#dcfce7\",\n 200: \"#bbf7d0\",\n 300: \"#86efac\",\n 400: \"#4ade80\",\n 500: \"#22c55e\",\n 600: \"#16a34a\",\n 700: \"#15803d\",\n 800: \"#166534\",\n 900: \"#14532d\",\n 950: \"#052e16\"\n },\n emerald: {\n 50: \"#ecfdf5\",\n 100: \"#d1fae5\",\n 200: \"#a7f3d0\",\n 300: \"#6ee7b7\",\n 400: \"#34d399\",\n 500: \"#10b981\",\n 600: \"#059669\",\n 700: \"#047857\",\n 800: \"#065f46\",\n 900: \"#064e3b\",\n 950: \"#022c22\"\n },\n teal: {\n 50: \"#f0fdfa\",\n 100: \"#ccfbf1\",\n 200: \"#99f6e4\",\n 300: \"#5eead4\",\n 400: \"#2dd4bf\",\n 500: \"#14b8a6\",\n 600: \"#0d9488\",\n 700: \"#0f766e\",\n 800: \"#115e59\",\n 900: \"#134e4a\",\n 950: \"#042f2e\"\n },\n cyan: {\n 50: \"#ecfeff\",\n 100: \"#cffafe\",\n 200: \"#a5f3fc\",\n 300: \"#67e8f9\",\n 400: \"#22d3ee\",\n 500: \"#06b6d4\",\n 600: \"#0891b2\",\n 700: \"#0e7490\",\n 800: \"#155e75\",\n 900: \"#164e63\",\n 950: \"#083344\"\n },\n sky: {\n 50: \"#f0f9ff\",\n 100: \"#e0f2fe\",\n 200: \"#bae6fd\",\n 300: \"#7dd3fc\",\n 400: \"#38bdf8\",\n 500: \"#0ea5e9\",\n 600: \"#0284c7\",\n 700: \"#0369a1\",\n 800: \"#075985\",\n 900: \"#0c4a6e\",\n 950: \"#082f49\"\n },\n blue: {\n 50: \"#eff6ff\",\n 100: \"#dbeafe\",\n 200: \"#bfdbfe\",\n 300: \"#93c5fd\",\n 400: \"#60a5fa\",\n 500: \"#3b82f6\",\n 600: \"#2563eb\",\n 700: \"#1d4ed8\",\n 800: \"#1e40af\",\n 900: \"#1e3a8a\",\n 950: \"#172554\"\n },\n indigo: {\n 50: \"#eef2ff\",\n 100: \"#e0e7ff\",\n 200: \"#c7d2fe\",\n 300: \"#a5b4fc\",\n 400: \"#818cf8\",\n 500: \"#6366f1\",\n 600: \"#4f46e5\",\n 700: \"#4338ca\",\n 800: \"#3730a3\",\n 900: \"#312e81\",\n 950: \"#1e1b4b\"\n },\n violet: {\n 50: \"#f5f3ff\",\n 100: \"#ede9fe\",\n 200: \"#ddd6fe\",\n 300: \"#c4b5fd\",\n 400: \"#a78bfa\",\n 500: \"#8b5cf6\",\n 600: \"#7c3aed\",\n 700: \"#6d28d9\",\n 800: \"#5b21b6\",\n 900: \"#4c1d95\",\n 950: \"#2e1065\"\n },\n purple: {\n 50: \"#faf5ff\",\n 100: \"#f3e8ff\",\n 200: \"#e9d5ff\",\n 300: \"#d8b4fe\",\n 400: \"#c084fc\",\n 500: \"#a855f7\",\n 600: \"#9333ea\",\n 700: \"#7e22ce\",\n 800: \"#6b21a8\",\n 900: \"#581c87\",\n 950: \"#3b0764\"\n },\n fuchsia: {\n 50: \"#fdf4ff\",\n 100: \"#fae8ff\",\n 200: \"#f5d0fe\",\n 300: \"#f0abfc\",\n 400: \"#e879f9\",\n 500: \"#d946ef\",\n 600: \"#c026d3\",\n 700: \"#a21caf\",\n 800: \"#86198f\",\n 900: \"#701a75\",\n 950: \"#4a044e\"\n },\n pink: {\n 50: \"#fdf2f8\",\n 100: \"#fce7f3\",\n 200: \"#fbcfe8\",\n 300: \"#f9a8d4\",\n 400: \"#f472b6\",\n 500: \"#ec4899\",\n 600: \"#db2777\",\n 700: \"#be185d\",\n 800: \"#9d174d\",\n 900: \"#831843\",\n 950: \"#500724\"\n },\n rose: {\n 50: \"#fff1f2\",\n 100: \"#ffe4e6\",\n 200: \"#fecdd3\",\n 300: \"#fda4af\",\n 400: \"#fb7185\",\n 500: \"#f43f5e\",\n 600: \"#e11d48\",\n 700: \"#be123c\",\n 800: \"#9f1239\",\n 900: \"#881337\",\n 950: \"#4c0519\"\n },\n get lightBlue () {\n warn({\n version: \"v2.2\",\n from: \"lightBlue\",\n to: \"sky\"\n });\n return this.sky;\n },\n get warmGray () {\n warn({\n version: \"v3.0\",\n from: \"warmGray\",\n to: \"stone\"\n });\n return this.stone;\n },\n get trueGray () {\n warn({\n version: \"v3.0\",\n from: \"trueGray\",\n to: \"neutral\"\n });\n return this.neutral;\n },\n get coolGray () {\n warn({\n version: \"v3.0\",\n from: \"coolGray\",\n to: \"gray\"\n });\n return this.gray;\n },\n get blueGray () {\n warn({\n version: \"v3.0\",\n from: \"blueGray\",\n to: \"slate\"\n });\n return this.slate;\n }\n};\n","let colors = require('./lib/public/colors')\nmodule.exports = (colors.__esModule ? colors : { default: colors }).default\n","import tailwindColors from \"tailwindcss/colors\";\n\nconst ignored = [\n `lightBlue`,\n `warmGray`,\n `trueGray`,\n `coolGray`,\n `blueGray`,\n `rose`,\n];\n\nconst getColors = () => {\n const colors = tailwindColors as any;\n const values = Object.keys(colors)\n .filter((groupKey, index) => {\n return (\n !ignored.includes(groupKey) &&\n index % 2 === 0 &&\n typeof colors[groupKey] !== \"string\"\n );\n })\n .map((groupKey) => {\n return Object.keys(colors[groupKey])\n .filter((key) => parseInt(key) % 100 === 0)\n .map((key) => colors[groupKey][key]);\n });\n\n return values.flatMap((v) => v);\n};\n\nexport const colors: string[] = getColors();\n","// TinyColor v1.4.2\n// https://github.com/bgrins/TinyColor\n// Brian Grinstead, MIT License\n\n(function(Math) {\n\nvar trimLeft = /^\\s+/,\n trimRight = /\\s+$/,\n tinyCounter = 0,\n mathRound = Math.round,\n mathMin = Math.min,\n mathMax = Math.max,\n mathRandom = Math.random;\n\nfunction tinycolor (color, opts) {\n\n color = (color) ? color : '';\n opts = opts || { };\n\n // If input is already a tinycolor, return itself\n if (color instanceof tinycolor) {\n return color;\n }\n // If we are called as a function, call using new instead\n if (!(this instanceof tinycolor)) {\n return new tinycolor(color, opts);\n }\n\n var rgb = inputToRGB(color);\n this._originalInput = color,\n this._r = rgb.r,\n this._g = rgb.g,\n this._b = rgb.b,\n this._a = rgb.a,\n this._roundA = mathRound(100*this._a) / 100,\n this._format = opts.format || rgb.format;\n this._gradientType = opts.gradientType;\n\n // Don't let the range of [0,255] come back in [0,1].\n // Potentially lose a little bit of precision here, but will fix issues where\n // .5 gets interpreted as half of the total, instead of half of 1\n // If it was supposed to be 128, this was already taken care of by `inputToRgb`\n if (this._r < 1) { this._r = mathRound(this._r); }\n if (this._g < 1) { this._g = mathRound(this._g); }\n if (this._b < 1) { this._b = mathRound(this._b); }\n\n this._ok = rgb.ok;\n this._tc_id = tinyCounter++;\n}\n\ntinycolor.prototype = {\n isDark: function() {\n return this.getBrightness() < 128;\n },\n isLight: function() {\n return !this.isDark();\n },\n isValid: function() {\n return this._ok;\n },\n getOriginalInput: function() {\n return this._originalInput;\n },\n getFormat: function() {\n return this._format;\n },\n getAlpha: function() {\n return this._a;\n },\n getBrightness: function() {\n //http://www.w3.org/TR/AERT#color-contrast\n var rgb = this.toRgb();\n return (rgb.r * 299 + rgb.g * 587 + rgb.b * 114) / 1000;\n },\n getLuminance: function() {\n //http://www.w3.org/TR/2008/REC-WCAG20-20081211/#relativeluminancedef\n var rgb = this.toRgb();\n var RsRGB, GsRGB, BsRGB, R, G, B;\n RsRGB = rgb.r/255;\n GsRGB = rgb.g/255;\n BsRGB = rgb.b/255;\n\n if (RsRGB <= 0.03928) {R = RsRGB / 12.92;} else {R = Math.pow(((RsRGB + 0.055) / 1.055), 2.4);}\n if (GsRGB <= 0.03928) {G = GsRGB / 12.92;} else {G = Math.pow(((GsRGB + 0.055) / 1.055), 2.4);}\n if (BsRGB <= 0.03928) {B = BsRGB / 12.92;} else {B = Math.pow(((BsRGB + 0.055) / 1.055), 2.4);}\n return (0.2126 * R) + (0.7152 * G) + (0.0722 * B);\n },\n setAlpha: function(value) {\n this._a = boundAlpha(value);\n this._roundA = mathRound(100*this._a) / 100;\n return this;\n },\n toHsv: function() {\n var hsv = rgbToHsv(this._r, this._g, this._b);\n return { h: hsv.h * 360, s: hsv.s, v: hsv.v, a: this._a };\n },\n toHsvString: function() {\n var hsv = rgbToHsv(this._r, this._g, this._b);\n var h = mathRound(hsv.h * 360), s = mathRound(hsv.s * 100), v = mathRound(hsv.v * 100);\n return (this._a == 1) ?\n \"hsv(\" + h + \", \" + s + \"%, \" + v + \"%)\" :\n \"hsva(\" + h + \", \" + s + \"%, \" + v + \"%, \"+ this._roundA + \")\";\n },\n toHsl: function() {\n var hsl = rgbToHsl(this._r, this._g, this._b);\n return { h: hsl.h * 360, s: hsl.s, l: hsl.l, a: this._a };\n },\n toHslString: function() {\n var hsl = rgbToHsl(this._r, this._g, this._b);\n var h = mathRound(hsl.h * 360), s = mathRound(hsl.s * 100), l = mathRound(hsl.l * 100);\n return (this._a == 1) ?\n \"hsl(\" + h + \", \" + s + \"%, \" + l + \"%)\" :\n \"hsla(\" + h + \", \" + s + \"%, \" + l + \"%, \"+ this._roundA + \")\";\n },\n toHex: function(allow3Char) {\n return rgbToHex(this._r, this._g, this._b, allow3Char);\n },\n toHexString: function(allow3Char) {\n return '#' + this.toHex(allow3Char);\n },\n toHex8: function(allow4Char) {\n return rgbaToHex(this._r, this._g, this._b, this._a, allow4Char);\n },\n toHex8String: function(allow4Char) {\n return '#' + this.toHex8(allow4Char);\n },\n toRgb: function() {\n return { r: mathRound(this._r), g: mathRound(this._g), b: mathRound(this._b), a: this._a };\n },\n toRgbString: function() {\n return (this._a == 1) ?\n \"rgb(\" + mathRound(this._r) + \", \" + mathRound(this._g) + \", \" + mathRound(this._b) + \")\" :\n \"rgba(\" + mathRound(this._r) + \", \" + mathRound(this._g) + \", \" + mathRound(this._b) + \", \" + this._roundA + \")\";\n },\n toPercentageRgb: function() {\n return { r: mathRound(bound01(this._r, 255) * 100) + \"%\", g: mathRound(bound01(this._g, 255) * 100) + \"%\", b: mathRound(bound01(this._b, 255) * 100) + \"%\", a: this._a };\n },\n toPercentageRgbString: function() {\n return (this._a == 1) ?\n \"rgb(\" + mathRound(bound01(this._r, 255) * 100) + \"%, \" + mathRound(bound01(this._g, 255) * 100) + \"%, \" + mathRound(bound01(this._b, 255) * 100) + \"%)\" :\n \"rgba(\" + mathRound(bound01(this._r, 255) * 100) + \"%, \" + mathRound(bound01(this._g, 255) * 100) + \"%, \" + mathRound(bound01(this._b, 255) * 100) + \"%, \" + this._roundA + \")\";\n },\n toName: function() {\n if (this._a === 0) {\n return \"transparent\";\n }\n\n if (this._a < 1) {\n return false;\n }\n\n return hexNames[rgbToHex(this._r, this._g, this._b, true)] || false;\n },\n toFilter: function(secondColor) {\n var hex8String = '#' + rgbaToArgbHex(this._r, this._g, this._b, this._a);\n var secondHex8String = hex8String;\n var gradientType = this._gradientType ? \"GradientType = 1, \" : \"\";\n\n if (secondColor) {\n var s = tinycolor(secondColor);\n secondHex8String = '#' + rgbaToArgbHex(s._r, s._g, s._b, s._a);\n }\n\n return \"progid:DXImageTransform.Microsoft.gradient(\"+gradientType+\"startColorstr=\"+hex8String+\",endColorstr=\"+secondHex8String+\")\";\n },\n toString: function(format) {\n var formatSet = !!format;\n format = format || this._format;\n\n var formattedString = false;\n var hasAlpha = this._a < 1 && this._a >= 0;\n var needsAlphaFormat = !formatSet && hasAlpha && (format === \"hex\" || format === \"hex6\" || format === \"hex3\" || format === \"hex4\" || format === \"hex8\" || format === \"name\");\n\n if (needsAlphaFormat) {\n // Special case for \"transparent\", all other non-alpha formats\n // will return rgba when there is transparency.\n if (format === \"name\" && this._a === 0) {\n return this.toName();\n }\n return this.toRgbString();\n }\n if (format === \"rgb\") {\n formattedString = this.toRgbString();\n }\n if (format === \"prgb\") {\n formattedString = this.toPercentageRgbString();\n }\n if (format === \"hex\" || format === \"hex6\") {\n formattedString = this.toHexString();\n }\n if (format === \"hex3\") {\n formattedString = this.toHexString(true);\n }\n if (format === \"hex4\") {\n formattedString = this.toHex8String(true);\n }\n if (format === \"hex8\") {\n formattedString = this.toHex8String();\n }\n if (format === \"name\") {\n formattedString = this.toName();\n }\n if (format === \"hsl\") {\n formattedString = this.toHslString();\n }\n if (format === \"hsv\") {\n formattedString = this.toHsvString();\n }\n\n return formattedString || this.toHexString();\n },\n clone: function() {\n return tinycolor(this.toString());\n },\n\n _applyModification: function(fn, args) {\n var color = fn.apply(null, [this].concat([].slice.call(args)));\n this._r = color._r;\n this._g = color._g;\n this._b = color._b;\n this.setAlpha(color._a);\n return this;\n },\n lighten: function() {\n return this._applyModification(lighten, arguments);\n },\n brighten: function() {\n return this._applyModification(brighten, arguments);\n },\n darken: function() {\n return this._applyModification(darken, arguments);\n },\n desaturate: function() {\n return this._applyModification(desaturate, arguments);\n },\n saturate: function() {\n return this._applyModification(saturate, arguments);\n },\n greyscale: function() {\n return this._applyModification(greyscale, arguments);\n },\n spin: function() {\n return this._applyModification(spin, arguments);\n },\n\n _applyCombination: function(fn, args) {\n return fn.apply(null, [this].concat([].slice.call(args)));\n },\n analogous: function() {\n return this._applyCombination(analogous, arguments);\n },\n complement: function() {\n return this._applyCombination(complement, arguments);\n },\n monochromatic: function() {\n return this._applyCombination(monochromatic, arguments);\n },\n splitcomplement: function() {\n return this._applyCombination(splitcomplement, arguments);\n },\n triad: function() {\n return this._applyCombination(triad, arguments);\n },\n tetrad: function() {\n return this._applyCombination(tetrad, arguments);\n }\n};\n\n// If input is an object, force 1 into \"1.0\" to handle ratios properly\n// String input requires \"1.0\" as input, so 1 will be treated as 1\ntinycolor.fromRatio = function(color, opts) {\n if (typeof color == \"object\") {\n var newColor = {};\n for (var i in color) {\n if (color.hasOwnProperty(i)) {\n if (i === \"a\") {\n newColor[i] = color[i];\n }\n else {\n newColor[i] = convertToPercentage(color[i]);\n }\n }\n }\n color = newColor;\n }\n\n return tinycolor(color, opts);\n};\n\n// Given a string or object, convert that input to RGB\n// Possible string inputs:\n//\n// \"red\"\n// \"#f00\" or \"f00\"\n// \"#ff0000\" or \"ff0000\"\n// \"#ff000000\" or \"ff000000\"\n// \"rgb 255 0 0\" or \"rgb (255, 0, 0)\"\n// \"rgb 1.0 0 0\" or \"rgb (1, 0, 0)\"\n// \"rgba (255, 0, 0, 1)\" or \"rgba 255, 0, 0, 1\"\n// \"rgba (1.0, 0, 0, 1)\" or \"rgba 1.0, 0, 0, 1\"\n// \"hsl(0, 100%, 50%)\" or \"hsl 0 100% 50%\"\n// \"hsla(0, 100%, 50%, 1)\" or \"hsla 0 100% 50%, 1\"\n// \"hsv(0, 100%, 100%)\" or \"hsv 0 100% 100%\"\n//\nfunction inputToRGB(color) {\n\n var rgb = { r: 0, g: 0, b: 0 };\n var a = 1;\n var s = null;\n var v = null;\n var l = null;\n var ok = false;\n var format = false;\n\n if (typeof color == \"string\") {\n color = stringInputToObject(color);\n }\n\n if (typeof color == \"object\") {\n if (isValidCSSUnit(color.r) && isValidCSSUnit(color.g) && isValidCSSUnit(color.b)) {\n rgb = rgbToRgb(color.r, color.g, color.b);\n ok = true;\n format = String(color.r).substr(-1) === \"%\" ? \"prgb\" : \"rgb\";\n }\n else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.v)) {\n s = convertToPercentage(color.s);\n v = convertToPercentage(color.v);\n rgb = hsvToRgb(color.h, s, v);\n ok = true;\n format = \"hsv\";\n }\n else if (isValidCSSUnit(color.h) && isValidCSSUnit(color.s) && isValidCSSUnit(color.l)) {\n s = convertToPercentage(color.s);\n l = convertToPercentage(color.l);\n rgb = hslToRgb(color.h, s, l);\n ok = true;\n format = \"hsl\";\n }\n\n if (color.hasOwnProperty(\"a\")) {\n a = color.a;\n }\n }\n\n a = boundAlpha(a);\n\n return {\n ok: ok,\n format: color.format || format,\n r: mathMin(255, mathMax(rgb.r, 0)),\n g: mathMin(255, mathMax(rgb.g, 0)),\n b: mathMin(255, mathMax(rgb.b, 0)),\n a: a\n };\n}\n\n\n// Conversion Functions\n// --------------------\n\n// `rgbToHsl`, `rgbToHsv`, `hslToRgb`, `hsvToRgb` modified from:\n// \n\n// `rgbToRgb`\n// Handle bounds / percentage checking to conform to CSS color spec\n// \n// *Assumes:* r, g, b in [0, 255] or [0, 1]\n// *Returns:* { r, g, b } in [0, 255]\nfunction rgbToRgb(r, g, b){\n return {\n r: bound01(r, 255) * 255,\n g: bound01(g, 255) * 255,\n b: bound01(b, 255) * 255\n };\n}\n\n// `rgbToHsl`\n// Converts an RGB color value to HSL.\n// *Assumes:* r, g, and b are contained in [0, 255] or [0, 1]\n// *Returns:* { h, s, l } in [0,1]\nfunction rgbToHsl(r, g, b) {\n\n r = bound01(r, 255);\n g = bound01(g, 255);\n b = bound01(b, 255);\n\n var max = mathMax(r, g, b), min = mathMin(r, g, b);\n var h, s, l = (max + min) / 2;\n\n if(max == min) {\n h = s = 0; // achromatic\n }\n else {\n var d = max - min;\n s = l > 0.5 ? d / (2 - max - min) : d / (max + min);\n switch(max) {\n case r: h = (g - b) / d + (g < b ? 6 : 0); break;\n case g: h = (b - r) / d + 2; break;\n case b: h = (r - g) / d + 4; break;\n }\n\n h /= 6;\n }\n\n return { h: h, s: s, l: l };\n}\n\n// `hslToRgb`\n// Converts an HSL color value to RGB.\n// *Assumes:* h is contained in [0, 1] or [0, 360] and s and l are contained [0, 1] or [0, 100]\n// *Returns:* { r, g, b } in the set [0, 255]\nfunction hslToRgb(h, s, l) {\n var r, g, b;\n\n h = bound01(h, 360);\n s = bound01(s, 100);\n l = bound01(l, 100);\n\n function hue2rgb(p, q, t) {\n if(t < 0) t += 1;\n if(t > 1) t -= 1;\n if(t < 1/6) return p + (q - p) * 6 * t;\n if(t < 1/2) return q;\n if(t < 2/3) return p + (q - p) * (2/3 - t) * 6;\n return p;\n }\n\n if(s === 0) {\n r = g = b = l; // achromatic\n }\n else {\n var q = l < 0.5 ? l * (1 + s) : l + s - l * s;\n var p = 2 * l - q;\n r = hue2rgb(p, q, h + 1/3);\n g = hue2rgb(p, q, h);\n b = hue2rgb(p, q, h - 1/3);\n }\n\n return { r: r * 255, g: g * 255, b: b * 255 };\n}\n\n// `rgbToHsv`\n// Converts an RGB color value to HSV\n// *Assumes:* r, g, and b are contained in the set [0, 255] or [0, 1]\n// *Returns:* { h, s, v } in [0,1]\nfunction rgbToHsv(r, g, b) {\n\n r = bound01(r, 255);\n g = bound01(g, 255);\n b = bound01(b, 255);\n\n var max = mathMax(r, g, b), min = mathMin(r, g, b);\n var h, s, v = max;\n\n var d = max - min;\n s = max === 0 ? 0 : d / max;\n\n if(max == min) {\n h = 0; // achromatic\n }\n else {\n switch(max) {\n case r: h = (g - b) / d + (g < b ? 6 : 0); break;\n case g: h = (b - r) / d + 2; break;\n case b: h = (r - g) / d + 4; break;\n }\n h /= 6;\n }\n return { h: h, s: s, v: v };\n}\n\n// `hsvToRgb`\n// Converts an HSV color value to RGB.\n// *Assumes:* h is contained in [0, 1] or [0, 360] and s and v are contained in [0, 1] or [0, 100]\n// *Returns:* { r, g, b } in the set [0, 255]\n function hsvToRgb(h, s, v) {\n\n h = bound01(h, 360) * 6;\n s = bound01(s, 100);\n v = bound01(v, 100);\n\n var i = Math.floor(h),\n f = h - i,\n p = v * (1 - s),\n q = v * (1 - f * s),\n t = v * (1 - (1 - f) * s),\n mod = i % 6,\n r = [v, q, p, p, t, v][mod],\n g = [t, v, v, q, p, p][mod],\n b = [p, p, t, v, v, q][mod];\n\n return { r: r * 255, g: g * 255, b: b * 255 };\n}\n\n// `rgbToHex`\n// Converts an RGB color to hex\n// Assumes r, g, and b are contained in the set [0, 255]\n// Returns a 3 or 6 character hex\nfunction rgbToHex(r, g, b, allow3Char) {\n\n var hex = [\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16))\n ];\n\n // Return a 3 character hex if possible\n if (allow3Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1)) {\n return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0);\n }\n\n return hex.join(\"\");\n}\n\n// `rgbaToHex`\n// Converts an RGBA color plus alpha transparency to hex\n// Assumes r, g, b are contained in the set [0, 255] and\n// a in [0, 1]. Returns a 4 or 8 character rgba hex\nfunction rgbaToHex(r, g, b, a, allow4Char) {\n\n var hex = [\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16)),\n pad2(convertDecimalToHex(a))\n ];\n\n // Return a 4 character hex if possible\n if (allow4Char && hex[0].charAt(0) == hex[0].charAt(1) && hex[1].charAt(0) == hex[1].charAt(1) && hex[2].charAt(0) == hex[2].charAt(1) && hex[3].charAt(0) == hex[3].charAt(1)) {\n return hex[0].charAt(0) + hex[1].charAt(0) + hex[2].charAt(0) + hex[3].charAt(0);\n }\n\n return hex.join(\"\");\n}\n\n// `rgbaToArgbHex`\n// Converts an RGBA color to an ARGB Hex8 string\n// Rarely used, but required for \"toFilter()\"\nfunction rgbaToArgbHex(r, g, b, a) {\n\n var hex = [\n pad2(convertDecimalToHex(a)),\n pad2(mathRound(r).toString(16)),\n pad2(mathRound(g).toString(16)),\n pad2(mathRound(b).toString(16))\n ];\n\n return hex.join(\"\");\n}\n\n// `equals`\n// Can be called with any tinycolor input\ntinycolor.equals = function (color1, color2) {\n if (!color1 || !color2) { return false; }\n return tinycolor(color1).toRgbString() == tinycolor(color2).toRgbString();\n};\n\ntinycolor.random = function() {\n return tinycolor.fromRatio({\n r: mathRandom(),\n g: mathRandom(),\n b: mathRandom()\n });\n};\n\n\n// Modification Functions\n// ----------------------\n// Thanks to less.js for some of the basics here\n// \n\nfunction desaturate(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.s -= amount / 100;\n hsl.s = clamp01(hsl.s);\n return tinycolor(hsl);\n}\n\nfunction saturate(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.s += amount / 100;\n hsl.s = clamp01(hsl.s);\n return tinycolor(hsl);\n}\n\nfunction greyscale(color) {\n return tinycolor(color).desaturate(100);\n}\n\nfunction lighten (color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.l += amount / 100;\n hsl.l = clamp01(hsl.l);\n return tinycolor(hsl);\n}\n\nfunction brighten(color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var rgb = tinycolor(color).toRgb();\n rgb.r = mathMax(0, mathMin(255, rgb.r - mathRound(255 * - (amount / 100))));\n rgb.g = mathMax(0, mathMin(255, rgb.g - mathRound(255 * - (amount / 100))));\n rgb.b = mathMax(0, mathMin(255, rgb.b - mathRound(255 * - (amount / 100))));\n return tinycolor(rgb);\n}\n\nfunction darken (color, amount) {\n amount = (amount === 0) ? 0 : (amount || 10);\n var hsl = tinycolor(color).toHsl();\n hsl.l -= amount / 100;\n hsl.l = clamp01(hsl.l);\n return tinycolor(hsl);\n}\n\n// Spin takes a positive or negative amount within [-360, 360] indicating the change of hue.\n// Values outside of this range will be wrapped into this range.\nfunction spin(color, amount) {\n var hsl = tinycolor(color).toHsl();\n var hue = (hsl.h + amount) % 360;\n hsl.h = hue < 0 ? 360 + hue : hue;\n return tinycolor(hsl);\n}\n\n// Combination Functions\n// ---------------------\n// Thanks to jQuery xColor for some of the ideas behind these\n// \n\nfunction complement(color) {\n var hsl = tinycolor(color).toHsl();\n hsl.h = (hsl.h + 180) % 360;\n return tinycolor(hsl);\n}\n\nfunction triad(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 120) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 240) % 360, s: hsl.s, l: hsl.l })\n ];\n}\n\nfunction tetrad(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 90) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 180) % 360, s: hsl.s, l: hsl.l }),\n tinycolor({ h: (h + 270) % 360, s: hsl.s, l: hsl.l })\n ];\n}\n\nfunction splitcomplement(color) {\n var hsl = tinycolor(color).toHsl();\n var h = hsl.h;\n return [\n tinycolor(color),\n tinycolor({ h: (h + 72) % 360, s: hsl.s, l: hsl.l}),\n tinycolor({ h: (h + 216) % 360, s: hsl.s, l: hsl.l})\n ];\n}\n\nfunction analogous(color, results, slices) {\n results = results || 6;\n slices = slices || 30;\n\n var hsl = tinycolor(color).toHsl();\n var part = 360 / slices;\n var ret = [tinycolor(color)];\n\n for (hsl.h = ((hsl.h - (part * results >> 1)) + 720) % 360; --results; ) {\n hsl.h = (hsl.h + part) % 360;\n ret.push(tinycolor(hsl));\n }\n return ret;\n}\n\nfunction monochromatic(color, results) {\n results = results || 6;\n var hsv = tinycolor(color).toHsv();\n var h = hsv.h, s = hsv.s, v = hsv.v;\n var ret = [];\n var modification = 1 / results;\n\n while (results--) {\n ret.push(tinycolor({ h: h, s: s, v: v}));\n v = (v + modification) % 1;\n }\n\n return ret;\n}\n\n// Utility Functions\n// ---------------------\n\ntinycolor.mix = function(color1, color2, amount) {\n amount = (amount === 0) ? 0 : (amount || 50);\n\n var rgb1 = tinycolor(color1).toRgb();\n var rgb2 = tinycolor(color2).toRgb();\n\n var p = amount / 100;\n\n var rgba = {\n r: ((rgb2.r - rgb1.r) * p) + rgb1.r,\n g: ((rgb2.g - rgb1.g) * p) + rgb1.g,\n b: ((rgb2.b - rgb1.b) * p) + rgb1.b,\n a: ((rgb2.a - rgb1.a) * p) + rgb1.a\n };\n\n return tinycolor(rgba);\n};\n\n\n// Readability Functions\n// ---------------------\n// false\n// tinycolor.isReadable(\"#000\", \"#111\",{level:\"AA\",size:\"large\"}) => false\ntinycolor.isReadable = function(color1, color2, wcag2) {\n var readability = tinycolor.readability(color1, color2);\n var wcag2Parms, out;\n\n out = false;\n\n wcag2Parms = validateWCAG2Parms(wcag2);\n switch (wcag2Parms.level + wcag2Parms.size) {\n case \"AAsmall\":\n case \"AAAlarge\":\n out = readability >= 4.5;\n break;\n case \"AAlarge\":\n out = readability >= 3;\n break;\n case \"AAAsmall\":\n out = readability >= 7;\n break;\n }\n return out;\n\n};\n\n// `mostReadable`\n// Given a base color and a list of possible foreground or background\n// colors for that base, returns the most readable color.\n// Optionally returns Black or White if the most readable color is unreadable.\n// *Example*\n// tinycolor.mostReadable(tinycolor.mostReadable(\"#123\", [\"#124\", \"#125\"],{includeFallbackColors:false}).toHexString(); // \"#112255\"\n// tinycolor.mostReadable(tinycolor.mostReadable(\"#123\", [\"#124\", \"#125\"],{includeFallbackColors:true}).toHexString(); // \"#ffffff\"\n// tinycolor.mostReadable(\"#a8015a\", [\"#faf3f3\"],{includeFallbackColors:true,level:\"AAA\",size:\"large\"}).toHexString(); // \"#faf3f3\"\n// tinycolor.mostReadable(\"#a8015a\", [\"#faf3f3\"],{includeFallbackColors:true,level:\"AAA\",size:\"small\"}).toHexString(); // \"#ffffff\"\ntinycolor.mostReadable = function(baseColor, colorList, args) {\n var bestColor = null;\n var bestScore = 0;\n var readability;\n var includeFallbackColors, level, size ;\n args = args || {};\n includeFallbackColors = args.includeFallbackColors ;\n level = args.level;\n size = args.size;\n\n for (var i= 0; i < colorList.length ; i++) {\n readability = tinycolor.readability(baseColor, colorList[i]);\n if (readability > bestScore) {\n bestScore = readability;\n bestColor = tinycolor(colorList[i]);\n }\n }\n\n if (tinycolor.isReadable(baseColor, bestColor, {\"level\":level,\"size\":size}) || !includeFallbackColors) {\n return bestColor;\n }\n else {\n args.includeFallbackColors=false;\n return tinycolor.mostReadable(baseColor,[\"#fff\", \"#000\"],args);\n }\n};\n\n\n// Big List of Colors\n// ------------------\n// \nvar names = tinycolor.names = {\n aliceblue: \"f0f8ff\",\n antiquewhite: \"faebd7\",\n aqua: \"0ff\",\n aquamarine: \"7fffd4\",\n azure: \"f0ffff\",\n beige: \"f5f5dc\",\n bisque: \"ffe4c4\",\n black: \"000\",\n blanchedalmond: \"ffebcd\",\n blue: \"00f\",\n blueviolet: \"8a2be2\",\n brown: \"a52a2a\",\n burlywood: \"deb887\",\n burntsienna: \"ea7e5d\",\n cadetblue: \"5f9ea0\",\n chartreuse: \"7fff00\",\n chocolate: \"d2691e\",\n coral: \"ff7f50\",\n cornflowerblue: \"6495ed\",\n cornsilk: \"fff8dc\",\n crimson: \"dc143c\",\n cyan: \"0ff\",\n darkblue: \"00008b\",\n darkcyan: \"008b8b\",\n darkgoldenrod: \"b8860b\",\n darkgray: \"a9a9a9\",\n darkgreen: \"006400\",\n darkgrey: \"a9a9a9\",\n darkkhaki: \"bdb76b\",\n darkmagenta: \"8b008b\",\n darkolivegreen: \"556b2f\",\n darkorange: \"ff8c00\",\n darkorchid: \"9932cc\",\n darkred: \"8b0000\",\n darksalmon: \"e9967a\",\n darkseagreen: \"8fbc8f\",\n darkslateblue: \"483d8b\",\n darkslategray: \"2f4f4f\",\n darkslategrey: \"2f4f4f\",\n darkturquoise: \"00ced1\",\n darkviolet: \"9400d3\",\n deeppink: \"ff1493\",\n deepskyblue: \"00bfff\",\n dimgray: \"696969\",\n dimgrey: \"696969\",\n dodgerblue: \"1e90ff\",\n firebrick: \"b22222\",\n floralwhite: \"fffaf0\",\n forestgreen: \"228b22\",\n fuchsia: \"f0f\",\n gainsboro: \"dcdcdc\",\n ghostwhite: \"f8f8ff\",\n gold: \"ffd700\",\n goldenrod: \"daa520\",\n gray: \"808080\",\n green: \"008000\",\n greenyellow: \"adff2f\",\n grey: \"808080\",\n honeydew: \"f0fff0\",\n hotpink: \"ff69b4\",\n indianred: \"cd5c5c\",\n indigo: \"4b0082\",\n ivory: \"fffff0\",\n khaki: \"f0e68c\",\n lavender: \"e6e6fa\",\n lavenderblush: \"fff0f5\",\n lawngreen: \"7cfc00\",\n lemonchiffon: \"fffacd\",\n lightblue: \"add8e6\",\n lightcoral: \"f08080\",\n lightcyan: \"e0ffff\",\n lightgoldenrodyellow: \"fafad2\",\n lightgray: \"d3d3d3\",\n lightgreen: \"90ee90\",\n lightgrey: \"d3d3d3\",\n lightpink: \"ffb6c1\",\n lightsalmon: \"ffa07a\",\n lightseagreen: \"20b2aa\",\n lightskyblue: \"87cefa\",\n lightslategray: \"789\",\n lightslategrey: \"789\",\n lightsteelblue: \"b0c4de\",\n lightyellow: \"ffffe0\",\n lime: \"0f0\",\n limegreen: \"32cd32\",\n linen: \"faf0e6\",\n magenta: \"f0f\",\n maroon: \"800000\",\n mediumaquamarine: \"66cdaa\",\n mediumblue: \"0000cd\",\n mediumorchid: \"ba55d3\",\n mediumpurple: \"9370db\",\n mediumseagreen: \"3cb371\",\n mediumslateblue: \"7b68ee\",\n mediumspringgreen: \"00fa9a\",\n mediumturquoise: \"48d1cc\",\n mediumvioletred: \"c71585\",\n midnightblue: \"191970\",\n mintcream: \"f5fffa\",\n mistyrose: \"ffe4e1\",\n moccasin: \"ffe4b5\",\n navajowhite: \"ffdead\",\n navy: \"000080\",\n oldlace: \"fdf5e6\",\n olive: \"808000\",\n olivedrab: \"6b8e23\",\n orange: \"ffa500\",\n orangered: \"ff4500\",\n orchid: \"da70d6\",\n palegoldenrod: \"eee8aa\",\n palegreen: \"98fb98\",\n paleturquoise: \"afeeee\",\n palevioletred: \"db7093\",\n papayawhip: \"ffefd5\",\n peachpuff: \"ffdab9\",\n peru: \"cd853f\",\n pink: \"ffc0cb\",\n plum: \"dda0dd\",\n powderblue: \"b0e0e6\",\n purple: \"800080\",\n rebeccapurple: \"663399\",\n red: \"f00\",\n rosybrown: \"bc8f8f\",\n royalblue: \"4169e1\",\n saddlebrown: \"8b4513\",\n salmon: \"fa8072\",\n sandybrown: \"f4a460\",\n seagreen: \"2e8b57\",\n seashell: \"fff5ee\",\n sienna: \"a0522d\",\n silver: \"c0c0c0\",\n skyblue: \"87ceeb\",\n slateblue: \"6a5acd\",\n slategray: \"708090\",\n slategrey: \"708090\",\n snow: \"fffafa\",\n springgreen: \"00ff7f\",\n steelblue: \"4682b4\",\n tan: \"d2b48c\",\n teal: \"008080\",\n thistle: \"d8bfd8\",\n tomato: \"ff6347\",\n turquoise: \"40e0d0\",\n violet: \"ee82ee\",\n wheat: \"f5deb3\",\n white: \"fff\",\n whitesmoke: \"f5f5f5\",\n yellow: \"ff0\",\n yellowgreen: \"9acd32\"\n};\n\n// Make it easy to access colors via `hexNames[hex]`\nvar hexNames = tinycolor.hexNames = flip(names);\n\n\n// Utilities\n// ---------\n\n// `{ 'name1': 'val1' }` becomes `{ 'val1': 'name1' }`\nfunction flip(o) {\n var flipped = { };\n for (var i in o) {\n if (o.hasOwnProperty(i)) {\n flipped[o[i]] = i;\n }\n }\n return flipped;\n}\n\n// Return a valid alpha value [0,1] with all invalid values being set to 1\nfunction boundAlpha(a) {\n a = parseFloat(a);\n\n if (isNaN(a) || a < 0 || a > 1) {\n a = 1;\n }\n\n return a;\n}\n\n// Take input from [0, n] and return it as [0, 1]\nfunction bound01(n, max) {\n if (isOnePointZero(n)) { n = \"100%\"; }\n\n var processPercent = isPercentage(n);\n n = mathMin(max, mathMax(0, parseFloat(n)));\n\n // Automatically convert percentage into number\n if (processPercent) {\n n = parseInt(n * max, 10) / 100;\n }\n\n // Handle floating point rounding errors\n if ((Math.abs(n - max) < 0.000001)) {\n return 1;\n }\n\n // Convert into [0, 1] range if it isn't already\n return (n % max) / parseFloat(max);\n}\n\n// Force a number between 0 and 1\nfunction clamp01(val) {\n return mathMin(1, mathMax(0, val));\n}\n\n// Parse a base-16 hex value into a base-10 integer\nfunction parseIntFromHex(val) {\n return parseInt(val, 16);\n}\n\n// Need to handle 1.0 as 100%, since once it is a number, there is no difference between it and 1\n// \nfunction isOnePointZero(n) {\n return typeof n == \"string\" && n.indexOf('.') != -1 && parseFloat(n) === 1;\n}\n\n// Check to see if string passed in is a percentage\nfunction isPercentage(n) {\n return typeof n === \"string\" && n.indexOf('%') != -1;\n}\n\n// Force a hex value to have 2 characters\nfunction pad2(c) {\n return c.length == 1 ? '0' + c : '' + c;\n}\n\n// Replace a decimal with it's percentage value\nfunction convertToPercentage(n) {\n if (n <= 1) {\n n = (n * 100) + \"%\";\n }\n\n return n;\n}\n\n// Converts a decimal to a hex value\nfunction convertDecimalToHex(d) {\n return Math.round(parseFloat(d) * 255).toString(16);\n}\n// Converts a hex value to a decimal\nfunction convertHexToDecimal(h) {\n return (parseIntFromHex(h) / 255);\n}\n\nvar matchers = (function() {\n\n // \n var CSS_INTEGER = \"[-\\\\+]?\\\\d+%?\";\n\n // \n var CSS_NUMBER = \"[-\\\\+]?\\\\d*\\\\.\\\\d+%?\";\n\n // Allow positive/negative integer/number. Don't capture the either/or, just the entire outcome.\n var CSS_UNIT = \"(?:\" + CSS_NUMBER + \")|(?:\" + CSS_INTEGER + \")\";\n\n // Actual matching.\n // Parentheses and commas are optional, but not required.\n // Whitespace can take the place of commas or opening paren\n var PERMISSIVE_MATCH3 = \"[\\\\s|\\\\(]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")\\\\s*\\\\)?\";\n var PERMISSIVE_MATCH4 = \"[\\\\s|\\\\(]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")[,|\\\\s]+(\" + CSS_UNIT + \")\\\\s*\\\\)?\";\n\n return {\n CSS_UNIT: new RegExp(CSS_UNIT),\n rgb: new RegExp(\"rgb\" + PERMISSIVE_MATCH3),\n rgba: new RegExp(\"rgba\" + PERMISSIVE_MATCH4),\n hsl: new RegExp(\"hsl\" + PERMISSIVE_MATCH3),\n hsla: new RegExp(\"hsla\" + PERMISSIVE_MATCH4),\n hsv: new RegExp(\"hsv\" + PERMISSIVE_MATCH3),\n hsva: new RegExp(\"hsva\" + PERMISSIVE_MATCH4),\n hex3: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,\n hex6: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/,\n hex4: /^#?([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})([0-9a-fA-F]{1})$/,\n hex8: /^#?([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})([0-9a-fA-F]{2})$/\n };\n})();\n\n// `isValidCSSUnit`\n// Take in a single string / number and check to see if it looks like a CSS unit\n// (see `matchers` above for definition).\nfunction isValidCSSUnit(color) {\n return !!matchers.CSS_UNIT.exec(color);\n}\n\n// `stringInputToObject`\n// Permissive string parsing. Take in a number of formats, and output an object\n// based on detected format. Returns `{ r, g, b }` or `{ h, s, l }` or `{ h, s, v}`\nfunction stringInputToObject(color) {\n\n color = color.replace(trimLeft,'').replace(trimRight, '').toLowerCase();\n var named = false;\n if (names[color]) {\n color = names[color];\n named = true;\n }\n else if (color == 'transparent') {\n return { r: 0, g: 0, b: 0, a: 0, format: \"name\" };\n }\n\n // Try to match string input using regular expressions.\n // Keep most of the number bounding out of this function - don't worry about [0,1] or [0,100] or [0,360]\n // Just return an object and let the conversion functions handle that.\n // This way the result will be the same whether the tinycolor is initialized with string or object.\n var match;\n if ((match = matchers.rgb.exec(color))) {\n return { r: match[1], g: match[2], b: match[3] };\n }\n if ((match = matchers.rgba.exec(color))) {\n return { r: match[1], g: match[2], b: match[3], a: match[4] };\n }\n if ((match = matchers.hsl.exec(color))) {\n return { h: match[1], s: match[2], l: match[3] };\n }\n if ((match = matchers.hsla.exec(color))) {\n return { h: match[1], s: match[2], l: match[3], a: match[4] };\n }\n if ((match = matchers.hsv.exec(color))) {\n return { h: match[1], s: match[2], v: match[3] };\n }\n if ((match = matchers.hsva.exec(color))) {\n return { h: match[1], s: match[2], v: match[3], a: match[4] };\n }\n if ((match = matchers.hex8.exec(color))) {\n return {\n r: parseIntFromHex(match[1]),\n g: parseIntFromHex(match[2]),\n b: parseIntFromHex(match[3]),\n a: convertHexToDecimal(match[4]),\n format: named ? \"name\" : \"hex8\"\n };\n }\n if ((match = matchers.hex6.exec(color))) {\n return {\n r: parseIntFromHex(match[1]),\n g: parseIntFromHex(match[2]),\n b: parseIntFromHex(match[3]),\n format: named ? \"name\" : \"hex\"\n };\n }\n if ((match = matchers.hex4.exec(color))) {\n return {\n r: parseIntFromHex(match[1] + '' + match[1]),\n g: parseIntFromHex(match[2] + '' + match[2]),\n b: parseIntFromHex(match[3] + '' + match[3]),\n a: convertHexToDecimal(match[4] + '' + match[4]),\n format: named ? \"name\" : \"hex8\"\n };\n }\n if ((match = matchers.hex3.exec(color))) {\n return {\n r: parseIntFromHex(match[1] + '' + match[1]),\n g: parseIntFromHex(match[2] + '' + match[2]),\n b: parseIntFromHex(match[3] + '' + match[3]),\n format: named ? \"name\" : \"hex\"\n };\n }\n\n return false;\n}\n\nfunction validateWCAG2Parms(parms) {\n // return valid WCAG2 parms for isReadable.\n // If input parms are invalid, return {\"level\":\"AA\", \"size\":\"small\"}\n var level, size;\n parms = parms || {\"level\":\"AA\", \"size\":\"small\"};\n level = (parms.level || \"AA\").toUpperCase();\n size = (parms.size || \"small\").toLowerCase();\n if (level !== \"AA\" && level !== \"AAA\") {\n level = \"AA\";\n }\n if (size !== \"small\" && size !== \"large\") {\n size = \"small\";\n }\n return {\"level\":level, \"size\":size};\n}\n\n// Node: Export function\nif (typeof module !== \"undefined\" && module.exports) {\n module.exports = tinycolor;\n}\n// AMD/requirejs: Define the module\nelse if (typeof define === 'function' && define.amd) {\n define(function () {return tinycolor;});\n}\n// Browser: Expose to window\nelse {\n window.tinycolor = tinycolor;\n}\n\n})(Math);\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n","\n\n\n\n\n","\n\n\n\n\n","\n\n\n\n\n","