{"version":3,"file":"main.2790f61ef09ba0a2.js","sources":["./src/app/modules/customers/models/customer.ts","./src/app/modules/customers/services/customers.service.ts","./src/app/modules/customers/services/filters.service.ts","./src/app/modules/customers/services/products.service.ts","./src/app/modules/network/services/network.service.ts","./src/app/modules/orders/models/order.ts","./src/app/modules/orders/services/orders.service.ts","./src/app/modules/shared/components/card/card.component.ts","./src/app/modules/shared/components/comments-ng/marked-colors.ts","./src/app/modules/shared/components/comments-ng/delete-modal/delete-modal.component.ts","./src/app/modules/shared/components/comments-ng/delete-modal/delete-modal.component.html","./src/app/modules/shared/components/comments-ng/format/format.component.ts","./src/app/modules/shared/components/comments-ng/format/format.component.html","./src/app/modules/shared/components/comments-ng/comments-ng.component.html","./src/app/modules/shared/components/comments-ng/comments-ng.component.ts","./src/app/modules/shared/pipes/interval.pipe.ts","./src/app/modules/shared/components/filter/filter.component.html","./src/app/modules/shared/components/filter/filter.component.ts","./src/app/modules/shared/components/hero-icon/hero-icon.component.ts","./src/app/modules/shared/components/image-cropper/image-cropper.component.ts","./src/app/modules/shared/components/loader/loader.component.ts","./src/app/modules/shared/pipes/currency.pipe.ts","./src/app/modules/shared/pipes/safeHtml.pipe.ts","./src/app/modules/shared/components/table-server/table-server.component.html","./src/app/modules/shared/components/table-server/table-server.component.ts","./src/app/modules/shared/pipes/highlight.pipe.ts","./src/app/modules/shared/components/table/table.component.html","./src/app/modules/shared/components/table/table.component.ts","./src/app/modules/shared/components/tabs/tab.component.ts","./src/app/modules/shared/components/tabs/tab.component.html","./src/app/modules/shared/components/tabs/tabs.component.html","./src/app/modules/shared/components/tabs/tabs.component.ts","./src/app/modules/shared/components/tooltip/tooltip.component.html","./src/app/modules/shared/components/tooltip/tooltip.component.ts","./src/app/modules/submissions/services/paperwork.service.ts","./src/app/modules/shared/components/update-document-modal/update-document-modal.component.html","./src/app/modules/shared/components/update-document-modal/update-document-modal.component.ts","./src/app/modules/shared/directives/default-img.directive.ts","./src/app/modules/shared/directives/has-feature.directive.ts","./src/app/modules/shared/directives/has-one-permission.directive.ts","./src/app/modules/shared/guards/auth.guard.ts","./src/app/modules/shared/guards/can-deactivate.guard.ts","./src/app/modules/shared/guards/config.guard.ts","./src/app/modules/shared/guards/feature.guard.ts","./src/app/modules/shared/guards/home.guard.ts","./src/app/modules/shared/guards/one-network.guard.ts","./src/app/modules/shared/guards/permission.guard.ts","./src/app/modules/shared/models/network.ts","./src/app/modules/shared/models/ticket.ts","./src/app/modules/shared/pipes/key.pipe.ts","./src/app/modules/shared/pipes/search.pipe.ts","./src/app/modules/shared/resolvers/emails.resolvers.ts","./src/app/modules/shared/resolvers/lines.resolver.ts","./src/app/modules/shared/resolvers/roles.resolver.ts","./src/app/modules/shared/resolvers/tags-mail.resolver.ts","./src/app/modules/shared/services/auth.service.ts","./src/app/modules/shared/services/cache-busting.service.ts","./src/app/modules/shared/services/color.service.ts","./src/app/modules/shared/services/export.service.ts","./src/app/modules/shared/services/feed.service.ts","./src/app/modules/shared/services/filter.service.ts","./src/app/modules/shared/services/lines.service.ts","./src/app/modules/shared/services/modal.service.ts","./src/app/modules/shared/services/pagination.service.ts","./src/app/modules/shared/services/permissions.service.ts","./src/app/modules/shared/services/queryParams.service.ts","./src/app/modules/shared/services/server-pagination.service.ts","./src/app/modules/shared/services/to-date-string.service.ts","./src/app/modules/shared/services/translation.service.ts","./src/app/modules/shared/models/upKeep.ts","./src/app/modules/shared/services/up-keep.service.ts","./src/app/modules/shared/shared.module.ts","./src/app/modules/shared/utils/filter.ts","./src/app/modules/shared/utils/observeProperty.ts","./src/app/modules/shop/models/product-item.ts","./src/app/modules/shop/services/shop.service.ts","./src/app/modules/submissions/components/submission-detail/delete-document-modal/delete-document-modal.component.ts","./src/app/modules/submissions/components/submission-detail/delete-document-modal/delete-document-modal.component.html","./src/app/modules/submissions/services/submission.service.ts","./src/app/modules/subscriptions/services/subscriptions.service.ts","./src/app/modules/users/services/users.service.ts","./src/app/utils/cache-manager.ts","./src/app/utils/escape-regexp.ts","./src/assets/icons/heroicons/index.ts","./src/assets/icons/heroicons/outline.ts","./src/assets/icons/heroicons/solid.ts","./src/environments/environment.prod.ts","./node_modules/@sentry/utils/esm/worldwide.js","./node_modules/@sentry/core/esm/constants.js","./node_modules/@sentry/utils/esm/is.js","./node_modules/@sentry/utils/esm/browser.js","./node_modules/@sentry/utils/esm/debug-build.js","./node_modules/@sentry/utils/esm/logger.js","./node_modules/@sentry/utils/esm/string.js","./node_modules/@sentry/utils/esm/object.js","./node_modules/@sentry/utils/esm/misc.js","./node_modules/@sentry/utils/esm/time.js","./node_modules/@sentry/core/esm/debug-build.js","./node_modules/@sentry/utils/esm/syncpromise.js","./node_modules/@sentry/core/esm/eventProcessors.js","./node_modules/@sentry/core/esm/session.js","./node_modules/@sentry/core/esm/utils/getRootSpan.js","./node_modules/@sentry/utils/esm/baggage.js","./node_modules/@sentry/utils/esm/tracing.js","./node_modules/@sentry/core/esm/utils/spanUtils.js","./node_modules/@sentry/core/esm/tracing/dynamicSamplingContext.js","./node_modules/@sentry/core/esm/utils/applyScopeDataToEvent.js","./node_modules/@sentry/core/esm/scope.js","./node_modules/@sentry/core/esm/version.js","./node_modules/@sentry/core/esm/hub.js","./node_modules/@sentry/utils/esm/stacktrace.js","./node_modules/@sentry/utils/esm/normalize.js","./node_modules/@sentry/utils/esm/memo.js","./node_modules/@sentry/core/esm/utils/prepareEvent.js","./node_modules/@sentry/core/esm/exports.js","./node_modules/@sentry/core/esm/integration.js","./node_modules/@sentry/core/esm/integrations/inboundfilters.js","./node_modules/@sentry/core/esm/integrations/functiontostring.js","./node_modules/@sentry/utils/esm/dsn.js","./node_modules/@sentry/core/esm/api.js","./node_modules/@sentry/utils/esm/supports.js","./node_modules/@sentry/utils/esm/vendor/supportsHistory.js","./node_modules/@sentry/utils/esm/instrument/_handlers.js","./node_modules/@sentry/utils/esm/instrument/history.js","./node_modules/@sentry/utils/esm/envelope.js","./node_modules/@sentry/utils/esm/error.js","./node_modules/@sentry/core/esm/baseclient.js","./node_modules/@sentry/core/esm/utils/sdkMetadata.js","./node_modules/@sentry/browser/esm/debug-build.js","./node_modules/@sentry/browser/esm/eventbuilder.js","./node_modules/@sentry/browser/esm/helpers.js","./node_modules/@sentry/browser/esm/client.js","./node_modules/@sentry/core/esm/envelope.js","./node_modules/@sentry/core/esm/metrics/envelope.js","./node_modules/@sentry/core/esm/metrics/utils.js","./node_modules/@sentry/utils/esm/env.js","./node_modules/@sentry/browser/esm/userfeedback.js","./node_modules/@sentry/utils/esm/clientreport.js","./node_modules/@sentry/utils/esm/instrument/console.js","./node_modules/@sentry/utils/esm/instrument/dom.js","./node_modules/@sentry/utils/esm/instrument/xhr.js","./node_modules/@sentry/utils/esm/instrument/fetch.js","./node_modules/@sentry/utils/esm/severity.js","./node_modules/@sentry/utils/esm/url.js","./node_modules/@sentry/browser/esm/integrations/breadcrumbs.js","./node_modules/@sentry/browser/esm/integrations/dedupe.js","./node_modules/@sentry/utils/esm/instrument/globalError.js","./node_modules/@sentry/utils/esm/instrument/globalUnhandledRejection.js","./node_modules/@sentry/browser/esm/integrations/globalhandlers.js","./node_modules/@sentry/browser/esm/integrations/httpcontext.js","./node_modules/@sentry/utils/esm/aggregate-errors.js","./node_modules/@sentry/browser/esm/integrations/linkederrors.js","./node_modules/@sentry/browser/esm/integrations/trycatch.js","./node_modules/@sentry/browser/esm/stack-parsers.js","./node_modules/@sentry/core/esm/transports/base.js","./node_modules/@sentry/utils/esm/promisebuffer.js","./node_modules/@sentry/utils/esm/ratelimit.js","./node_modules/@sentry/browser/esm/transports/utils.js","./node_modules/@sentry/browser/esm/transports/fetch.js","./node_modules/@sentry/browser/esm/transports/xhr.js","./node_modules/@sentry/browser/esm/sdk.js","./node_modules/@sentry/core/esm/sdk.js","./node_modules/@sentry/core/esm/semanticAttributes.js","./node_modules/@sentry/core/esm/tracing/spanstatus.js","./node_modules/@sentry/core/esm/tracing/span.js","./node_modules/@sentry/core/esm/utils/hasTracingEnabled.js","./node_modules/@sentry/core/esm/tracing/trace.js","./node_modules/@sentry/core/esm/tracing/transaction.js","./node_modules/@sentry/core/esm/tracing/idletransaction.js","./node_modules/@sentry-internal/tracing/esm/common/debug-build.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/bindReporter.js","./node_modules/@sentry-internal/tracing/esm/browser/types.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/getNavigationEntry.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/getActivationStart.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/initMetric.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/generateUniqueID.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/observe.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/onHidden.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/getVisibilityWatcher.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/lib/polyfills/interactionCountPolyfill.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/getINP.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/getLCP.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/onTTFB.js","./node_modules/@sentry-internal/tracing/esm/browser/instrument.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/getCLS.js","./node_modules/@sentry-internal/tracing/esm/browser/web-vitals/getFID.js","./node_modules/@sentry-internal/tracing/esm/browser/request.js","./node_modules/@sentry-internal/tracing/esm/common/fetch.js","./node_modules/@sentry/angular-ivy/fesm2015/sentry-angular-ivy.js","./node_modules/@sentry/core/esm/tracing/utils.js","./node_modules/@sentry/core/esm/tracing/errors.js","./node_modules/@sentry/core/esm/tracing/sampling.js","./node_modules/@sentry/core/esm/tracing/hubextensions.js","./node_modules/@sentry/core/esm/span.js","./node_modules/@sentry-internal/tracing/esm/browser/metrics/utils.js","./node_modules/@sentry-internal/tracing/esm/browser/metrics/index.js","./node_modules/@sentry-internal/tracing/esm/browser/browsertracing.js","./node_modules/@sentry-internal/tracing/esm/browser/router.js","./src/app/app.component.html","./src/app/app.component.ts","./node_modules/@angular/common/locales/extra/fr.mjs","./node_modules/@angular/common/locales/fr.mjs","./node_modules/@angular/animations/fesm2020/browser.mjs","./node_modules/@angular/platform-browser/fesm2020/animations.mjs","./src/app/modules/shared/components/search-customer-modal/search-customer-modal.component.html","./src/app/modules/shared/components/search-customer-modal/search-customer-modal.component.ts","./src/app/modules/shared/services/badge.service.ts","./src/app/services/adminlabs.service.ts","./src/app/modules/shared/components/breadcrumb/breadcrumb.component.html","./src/app/modules/shared/components/breadcrumb/breadcrumb.component.ts","./src/app/modules/shared/components/modal/modal.component.html","./src/app/modules/shared/components/modal/modal.component.ts","./src/app/app-layout/app-layout.component.html","./src/app/app-layout/app-layout.component.ts","./node_modules/@angular/cdk/fesm2020/stepper.mjs","./node_modules/@angular/material/fesm2020/stepper.mjs","./src/app/modules/customers/services/ticket.service.ts","./src/app/modules/customers/components/customer-detail/anonymization-modal/anonymization-modal.component.html","./src/app/modules/customers/components/customer-detail/anonymization-modal/anonymization-modal.component.ts","./src/app/modules/customers/components/customer-detail/attributes-modal/attributes-modal.component.html","./src/app/modules/customers/components/customer-detail/attributes-modal/attributes-modal.component.ts","./src/app/modules/customers/components/customer-detail/avatar-update-modal/avatar-update-modal.component.html","./src/app/modules/customers/components/customer-detail/avatar-update-modal/avatar-update-modal.component.ts","./src/app/modules/customers/components/customer-detail/information-modal/information-modal.component.html","./src/app/modules/customers/components/customer-detail/information-modal/information-modal.component.ts","./src/app/modules/customers/components/customer-detail/lost-phone-modal/customer-lost-phone-modal.component.ts","./src/app/modules/customers/components/customer-detail/lost-phone-modal/customer-lost-phone-modal.component.html","./src/app/modules/customers/components/customer-detail/network-dissociation-modal/network-dissociation-modal.component.ts","./src/app/modules/customers/components/customer-detail/network-dissociation-modal/network-dissociation-modal.component.html","./node_modules/date-fns/esm/isMatch/index.js","./src/app/modules/customers/components/customer-detail/ticket-generation-modal/ticket-generation-modal.component.html","./src/app/modules/customers/components/customer-detail/ticket-generation-modal/ticket-generation-modal.component.ts","./src/app/modules/customers/components/customer-detail/fusion-modal/fusion-modal.component.html","./src/app/modules/customers/components/customer-detail/fusion-modal/fusion-modal.component.ts","./src/app/modules/customers/components/customer-detail/delete-ticket-modal/delete-ticket-modal.component.html","./src/app/modules/customers/components/customer-detail/delete-ticket-modal/delete-ticket-modal.component.ts","./src/app/modules/customers/services/paperwork.service.ts","./src/app/modules/customers/components/customer-detail/document-upload-modal/document-upload-modal.component.html","./src/app/modules/customers/components/customer-detail/document-upload-modal/document-upload-modal.component.ts","./src/app/modules/customers/components/customer-detail/create-firebase-account-modal/create-firebase-account-modal.component.html","./src/app/modules/customers/components/customer-detail/create-firebase-account-modal/create-firebase-account-modal.component.ts","./src/app/modules/customers/services/shop.service.ts","./src/app/modules/customers/services/field.service.ts","./src/app/modules/customers/components/customer-detail/credit-product-modal/credit-product-modal.component.html","./src/app/modules/customers/components/customer-detail/credit-product-modal/credit-product-modal.component.ts","./src/app/modules/shared/components/card/closable-card.component.ts","./src/app/modules/customers/components/customer-detail/customer-comment/customer-comment.component.html","./src/app/modules/customers/components/customer-detail/customer-comment/customer-comment.component.ts","./src/app/modules/customers/components/customer-detail/customer-detail.component.html","./src/app/modules/customers/components/customer-detail/customer-detail.component.ts","./src/app/modules/customers/constants/resetPassword.ts","./src/app/modules/customers/components/customer-search/customer-search.component.ts","./src/app/modules/customers/components/customer-search/customer-search.component.html","./src/app/modules/customers/components/customers.component.html","./src/app/modules/customers/customers.routing.ts","./src/app/modules/customers/components/customers.component.ts","./src/app/modules/customers/customers.module.ts","./src/app/modules/orders/components/order-detail/order-comment/order-comment.component.html","./src/app/modules/orders/components/order-detail/order-comment/order-comment.component.ts","./src/app/modules/orders/components/order-detail/order-detail.component.html","./src/app/modules/orders/components/order-detail/order-detail.component.ts","./src/app/modules/orders/components/orders.component.html","./src/app/modules/orders/orders.routing.ts","./src/app/modules/orders/components/orders.component.ts","./src/app/modules/orders/orders.module.ts","./src/app/app.routing.ts","./src/app/interceptors/common.interceptor.ts","./src/app/app.module.ts","./src/main.ts","./node_modules/@sentry-internal/tracing/esm/browser/backgroundtab.js","./node_modules/@sentry/core/esm/tracing/measurement.js","./node_modules/@ngx-translate/core/__ivy_ngcc__/fesm2015/ngx-translate-core.js","./node_modules/@ngx-translate/http-loader/__ivy_ngcc__/fesm2015/ngx-translate-http-loader.js","./node_modules/@transloadit/prettier-bytes/prettierBytes.js","./node_modules/@uppy/image-editor/node_modules/cropperjs/dist/cropper.js","./node_modules/ang-jsoneditor/__ivy_ngcc__/fesm2015/ang-jsoneditor.js","./node_modules/angular2-notifications/__ivy_ngcc__/fesm2015/angular2-notifications.js","./node_modules/cropperjs/dist/cropper.js","./node_modules/date-fns/esm/_lib/addLeadingZeros/index.js","./node_modules/date-fns/esm/locale/en-US/_lib/formatDistance/index.js","./node_modules/date-fns/esm/locale/_lib/buildFormatLongFn/index.js","./node_modules/date-fns/esm/locale/en-US/_lib/formatLong/index.js","./node_modules/date-fns/esm/locale/en-US/_lib/formatRelative/index.js","./node_modules/date-fns/esm/locale/_lib/buildLocalizeFn/index.js","./node_modules/date-fns/esm/locale/_lib/buildMatchFn/index.js","./node_modules/date-fns/esm/_lib/defaultLocale/index.js","./node_modules/date-fns/esm/locale/en-US/index.js","./node_modules/date-fns/esm/locale/en-US/_lib/localize/index.js","./node_modules/date-fns/esm/locale/en-US/_lib/match/index.js","./node_modules/date-fns/esm/locale/_lib/buildMatchPatternFn/index.js","./node_modules/date-fns/esm/_lib/defaultOptions/index.js","./node_modules/date-fns/esm/_lib/format/longFormatters/index.js","./node_modules/date-fns/esm/_lib/getTimezoneOffsetInMilliseconds/index.js","./node_modules/date-fns/esm/_lib/startOfUTCISOWeekYear/index.js","./node_modules/date-fns/esm/_lib/getUTCISOWeek/index.js","./node_modules/date-fns/esm/_lib/getUTCISOWeekYear/index.js","./node_modules/date-fns/esm/_lib/startOfUTCWeekYear/index.js","./node_modules/date-fns/esm/_lib/getUTCWeek/index.js","./node_modules/date-fns/esm/_lib/getUTCWeekYear/index.js","./node_modules/date-fns/esm/_lib/protectedTokens/index.js","./node_modules/date-fns/esm/_lib/requiredArgs/index.js","./node_modules/date-fns/esm/_lib/startOfUTCISOWeek/index.js","./node_modules/date-fns/esm/_lib/startOfUTCWeek/index.js","./node_modules/date-fns/esm/_lib/toInteger/index.js","./node_modules/date-fns/esm/addMilliseconds/index.js","./node_modules/date-fns/esm/compareAsc/index.js","./node_modules/date-fns/esm/constants/index.js","./node_modules/date-fns/esm/differenceInMilliseconds/index.js","./node_modules/date-fns/esm/_lib/roundingMethods/index.js","./node_modules/date-fns/esm/differenceInMinutes/index.js","./node_modules/date-fns/esm/_lib/getUTCDayOfYear/index.js","./node_modules/date-fns/esm/_lib/format/lightFormatters/index.js","./node_modules/date-fns/esm/_lib/format/formatters/index.js","./node_modules/date-fns/esm/format/index.js","./node_modules/date-fns/esm/formatISO/index.js","./node_modules/date-fns/esm/isDate/index.js","./node_modules/date-fns/esm/isValid/index.js","./node_modules/@babel/runtime/helpers/esm/arrayLikeToArray.js","./node_modules/@babel/runtime/helpers/esm/createForOfIteratorHelper.js","./node_modules/@babel/runtime/helpers/esm/unsupportedIterableToArray.js","./node_modules/date-fns/esm/_lib/assign/index.js","./node_modules/@babel/runtime/helpers/esm/assertThisInitialized.js","./node_modules/@babel/runtime/helpers/esm/setPrototypeOf.js","./node_modules/@babel/runtime/helpers/esm/inherits.js","./node_modules/@babel/runtime/helpers/esm/getPrototypeOf.js","./node_modules/@babel/runtime/helpers/esm/isNativeReflectConstruct.js","./node_modules/@babel/runtime/helpers/esm/possibleConstructorReturn.js","./node_modules/@babel/runtime/helpers/esm/createSuper.js","./node_modules/@babel/runtime/helpers/esm/classCallCheck.js","./node_modules/@babel/runtime/helpers/esm/createClass.js","./node_modules/date-fns/esm/parse/_lib/Setter.js","./node_modules/date-fns/esm/parse/_lib/Parser.js","./node_modules/date-fns/esm/parse/_lib/parsers/EraParser.js","./node_modules/date-fns/esm/parse/_lib/constants.js","./node_modules/date-fns/esm/parse/_lib/utils.js","./node_modules/date-fns/esm/parse/_lib/parsers/YearParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/LocalWeekYearParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/ISOWeekYearParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/ExtendedYearParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/QuarterParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/StandAloneQuarterParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/MonthParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/StandAloneMonthParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/LocalWeekParser.js","./node_modules/date-fns/esm/_lib/setUTCWeek/index.js","./node_modules/date-fns/esm/parse/_lib/parsers/ISOWeekParser.js","./node_modules/date-fns/esm/_lib/setUTCISOWeek/index.js","./node_modules/date-fns/esm/parse/_lib/parsers/DateParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/DayOfYearParser.js","./node_modules/date-fns/esm/_lib/setUTCDay/index.js","./node_modules/date-fns/esm/parse/_lib/parsers/DayParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/LocalDayParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/StandAloneLocalDayParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/ISODayParser.js","./node_modules/date-fns/esm/_lib/setUTCISODay/index.js","./node_modules/date-fns/esm/parse/_lib/parsers/AMPMParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/AMPMMidnightParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/DayPeriodParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/Hour1to12Parser.js","./node_modules/date-fns/esm/parse/_lib/parsers/Hour0to23Parser.js","./node_modules/date-fns/esm/parse/_lib/parsers/Hour0To11Parser.js","./node_modules/date-fns/esm/parse/_lib/parsers/Hour1To24Parser.js","./node_modules/date-fns/esm/parse/_lib/parsers/MinuteParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/SecondParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/FractionOfSecondParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/ISOTimezoneWithZParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/ISOTimezoneParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/TimestampSecondsParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/TimestampMillisecondsParser.js","./node_modules/date-fns/esm/parse/_lib/parsers/index.js","./node_modules/date-fns/esm/parse/index.js","./node_modules/date-fns/esm/parseISO/index.js","./node_modules/date-fns/esm/getDaysInMonth/index.js","./node_modules/date-fns/esm/setMonth/index.js","./node_modules/date-fns/esm/set/index.js","./node_modules/date-fns/esm/subMilliseconds/index.js","./node_modules/date-fns/esm/toDate/index.js","./node_modules/dompurify/dist/purify.js","./node_modules/escape-string-regexp/index.js","./node_modules/exifr/dist/mini.umd.js","./node_modules/is-shallow-equal/index.js","./node_modules/jsoneditor/dist/jsoneditor.min.js","./node_modules/lodash.debounce/index.js","./node_modules/lodash.isequal/index.js","./node_modules/lodash.throttle/index.js","./node_modules/memoize-one/dist/memoize-one.esm.js","./node_modules/mime-match/index.js","./node_modules/moment/locale/af.js","./node_modules/moment/locale/ar-dz.js","./node_modules/moment/locale/ar-kw.js","./node_modules/moment/locale/ar-ly.js","./node_modules/moment/locale/ar-ma.js","./node_modules/moment/locale/ar-ps.js","./node_modules/moment/locale/ar-sa.js","./node_modules/moment/locale/ar-tn.js","./node_modules/moment/locale/ar.js","./node_modules/moment/locale/az.js","./node_modules/moment/locale/be.js","./node_modules/moment/locale/bg.js","./node_modules/moment/locale/bm.js","./node_modules/moment/locale/bn-bd.js","./node_modules/moment/locale/bn.js","./node_modules/moment/locale/bo.js","./node_modules/moment/locale/br.js","./node_modules/moment/locale/bs.js","./node_modules/moment/locale/ca.js","./node_modules/moment/locale/cs.js","./node_modules/moment/locale/cv.js","./node_modules/moment/locale/cy.js","./node_modules/moment/locale/da.js","./node_modules/moment/locale/de-at.js","./node_modules/moment/locale/de-ch.js","./node_modules/moment/locale/de.js","./node_modules/moment/locale/dv.js","./node_modules/moment/locale/el.js","./node_modules/moment/locale/en-au.js","./node_modules/moment/locale/en-ca.js","./node_modules/moment/locale/en-gb.js","./node_modules/moment/locale/en-ie.js","./node_modules/moment/locale/en-il.js","./node_modules/moment/locale/en-in.js","./node_modules/moment/locale/en-nz.js","./node_modules/moment/locale/en-sg.js","./node_modules/moment/locale/eo.js","./node_modules/moment/locale/es-do.js","./node_modules/moment/locale/es-mx.js","./node_modules/moment/locale/es-us.js","./node_modules/moment/locale/es.js","./node_modules/moment/locale/et.js","./node_modules/moment/locale/eu.js","./node_modules/moment/locale/fa.js","./node_modules/moment/locale/fi.js","./node_modules/moment/locale/fil.js","./node_modules/moment/locale/fo.js","./node_modules/moment/locale/fr-ca.js","./node_modules/moment/locale/fr-ch.js","./node_modules/moment/locale/fr.js","./node_modules/moment/locale/fy.js","./node_modules/moment/locale/ga.js","./node_modules/moment/locale/gd.js","./node_modules/moment/locale/gl.js","./node_modules/moment/locale/gom-deva.js","./node_modules/moment/locale/gom-latn.js","./node_modules/moment/locale/gu.js","./node_modules/moment/locale/he.js","./node_modules/moment/locale/hi.js","./node_modules/moment/locale/hr.js","./node_modules/moment/locale/hu.js","./node_modules/moment/locale/hy-am.js","./node_modules/moment/locale/id.js","./node_modules/moment/locale/is.js","./node_modules/moment/locale/it-ch.js","./node_modules/moment/locale/it.js","./node_modules/moment/locale/ja.js","./node_modules/moment/locale/jv.js","./node_modules/moment/locale/ka.js","./node_modules/moment/locale/kk.js","./node_modules/moment/locale/km.js","./node_modules/moment/locale/kn.js","./node_modules/moment/locale/ko.js","./node_modules/moment/locale/ku-kmr.js","./node_modules/moment/locale/ku.js","./node_modules/moment/locale/ky.js","./node_modules/moment/locale/lb.js","./node_modules/moment/locale/lo.js","./node_modules/moment/locale/lt.js","./node_modules/moment/locale/lv.js","./node_modules/moment/locale/me.js","./node_modules/moment/locale/mi.js","./node_modules/moment/locale/mk.js","./node_modules/moment/locale/ml.js","./node_modules/moment/locale/mn.js","./node_modules/moment/locale/mr.js","./node_modules/moment/locale/ms-my.js","./node_modules/moment/locale/ms.js","./node_modules/moment/locale/mt.js","./node_modules/moment/locale/my.js","./node_modules/moment/locale/nb.js","./node_modules/moment/locale/ne.js","./node_modules/moment/locale/nl-be.js","./node_modules/moment/locale/nl.js","./node_modules/moment/locale/nn.js","./node_modules/moment/locale/oc-lnc.js","./node_modules/moment/locale/pa-in.js","./node_modules/moment/locale/pl.js","./node_modules/moment/locale/pt-br.js","./node_modules/moment/locale/pt.js","./node_modules/moment/locale/ro.js","./node_modules/moment/locale/ru.js","./node_modules/moment/locale/sd.js","./node_modules/moment/locale/se.js","./node_modules/moment/locale/si.js","./node_modules/moment/locale/sk.js","./node_modules/moment/locale/sl.js","./node_modules/moment/locale/sq.js","./node_modules/moment/locale/sr-cyrl.js","./node_modules/moment/locale/sr.js","./node_modules/moment/locale/ss.js","./node_modules/moment/locale/sv.js","./node_modules/moment/locale/sw.js","./node_modules/moment/locale/ta.js","./node_modules/moment/locale/te.js","./node_modules/moment/locale/tet.js","./node_modules/moment/locale/tg.js","./node_modules/moment/locale/th.js","./node_modules/moment/locale/tk.js","./node_modules/moment/locale/tl-ph.js","./node_modules/moment/locale/tlh.js","./node_modules/moment/locale/tr.js","./node_modules/moment/locale/tzl.js","./node_modules/moment/locale/tzm-latn.js","./node_modules/moment/locale/tzm.js","./node_modules/moment/locale/ug-cn.js","./node_modules/moment/locale/uk.js","./node_modules/moment/locale/ur.js","./node_modules/moment/locale/uz-latn.js","./node_modules/moment/locale/uz.js","./node_modules/moment/locale/vi.js","./node_modules/moment/locale/x-pseudo.js","./node_modules/moment/locale/yo.js","./node_modules/moment/locale/zh-cn.js","./node_modules/moment/locale/zh-hk.js","./node_modules/moment/locale/zh-mo.js","./node_modules/moment/locale/zh-tw.js","./node_modules/moment/moment.js","./node_modules/namespace-emitter/index.js","./node_modules/ngx-moment/__ivy_ngcc__/fesm2015/ngx-moment.js","./node_modules/ngx-pagination/__ivy_ngcc__/dist/ngx-pagination.js","./node_modules/ngx-select-ex/__ivy_ngcc__/fesm2015/ngx-select-ex.js","./node_modules/preact/dist/preact.module.js","./node_modules/rxjs/dist/esm/internal/BehaviorSubject.js","./node_modules/rxjs/dist/esm/internal/Observable.js","./node_modules/rxjs/dist/esm/internal/ReplaySubject.js","./node_modules/rxjs/dist/esm/internal/util/ObjectUnsubscribedError.js","./node_modules/rxjs/dist/esm/internal/Subject.js","./node_modules/rxjs/dist/esm/internal/NotificationFactories.js","./node_modules/rxjs/dist/esm/internal/Subscriber.js","./node_modules/rxjs/dist/esm/internal/util/UnsubscriptionError.js","./node_modules/rxjs/dist/esm/internal/Subscription.js","./node_modules/rxjs/dist/esm/internal/config.js","./node_modules/rxjs/dist/esm/internal/firstValueFrom.js","./node_modules/rxjs/dist/esm/internal/observable/combineLatest.js","./node_modules/rxjs/dist/esm/internal/observable/concat.js","./node_modules/rxjs/dist/esm/internal/operators/concatAll.js","./node_modules/rxjs/dist/esm/internal/observable/defer.js","./node_modules/rxjs/dist/esm/internal/observable/empty.js","./node_modules/rxjs/dist/esm/internal/observable/forkJoin.js","./node_modules/rxjs/dist/esm/internal/operators/observeOn.js","./node_modules/rxjs/dist/esm/internal/operators/subscribeOn.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduleAsyncIterable.js","./node_modules/rxjs/dist/esm/internal/observable/from.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduled.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduleObservable.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduleArray.js","./node_modules/rxjs/dist/esm/internal/scheduled/schedulePromise.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduleIterable.js","./node_modules/rxjs/dist/esm/internal/scheduled/scheduleReadableStreamLike.js","./node_modules/rxjs/dist/esm/internal/observable/fromEvent.js","./node_modules/rxjs/dist/esm/internal/observable/innerFrom.js","./node_modules/rxjs/dist/esm/internal/observable/merge.js","./node_modules/rxjs/dist/esm/internal/observable/of.js","./node_modules/rxjs/dist/esm/internal/observable/throwError.js","./node_modules/rxjs/dist/esm/internal/observable/timer.js","./node_modules/rxjs/dist/esm/internal/util/isDate.js","./node_modules/rxjs/dist/esm/internal/operators/OperatorSubscriber.js","./node_modules/rxjs/dist/esm/internal/operators/catchError.js","./node_modules/rxjs/dist/esm/internal/operators/concatMap.js","./node_modules/rxjs/dist/esm/internal/operators/debounceTime.js","./node_modules/rxjs/dist/esm/internal/operators/defaultIfEmpty.js","./node_modules/rxjs/dist/esm/internal/operators/delayWhen.js","./node_modules/rxjs/dist/esm/internal/operators/ignoreElements.js","./node_modules/rxjs/dist/esm/internal/operators/delay.js","./node_modules/rxjs/dist/esm/internal/operators/distinctUntilChanged.js","./node_modules/rxjs/dist/esm/internal/operators/filter.js","./node_modules/rxjs/dist/esm/internal/operators/finalize.js","./node_modules/rxjs/dist/esm/internal/operators/first.js","./node_modules/rxjs/dist/esm/internal/operators/map.js","./node_modules/rxjs/dist/esm/internal/operators/mapTo.js","./node_modules/rxjs/dist/esm/internal/operators/mergeAll.js","./node_modules/rxjs/dist/esm/internal/operators/mergeMap.js","./node_modules/rxjs/dist/esm/internal/operators/mergeInternals.js","./node_modules/rxjs/dist/esm/internal/operators/scanInternals.js","./node_modules/rxjs/dist/esm/internal/operators/share.js","./node_modules/rxjs/dist/esm/internal/operators/shareReplay.js","./node_modules/rxjs/dist/esm/internal/operators/skip.js","./node_modules/rxjs/dist/esm/internal/operators/startWith.js","./node_modules/rxjs/dist/esm/internal/operators/switchMap.js","./node_modules/rxjs/dist/esm/internal/operators/take.js","./node_modules/rxjs/dist/esm/internal/operators/takeUntil.js","./node_modules/rxjs/dist/esm/internal/operators/takeWhile.js","./node_modules/rxjs/dist/esm/internal/operators/tap.js","./node_modules/rxjs/dist/esm/internal/operators/throwIfEmpty.js","./node_modules/rxjs/dist/esm/internal/scheduler/Action.js","./node_modules/rxjs/dist/esm/internal/scheduler/intervalProvider.js","./node_modules/rxjs/dist/esm/internal/scheduler/AsyncAction.js","./node_modules/rxjs/dist/esm/internal/Scheduler.js","./node_modules/rxjs/dist/esm/internal/scheduler/AsyncScheduler.js","./node_modules/rxjs/dist/esm/internal/scheduler/animationFrameProvider.js","./node_modules/rxjs/dist/esm/internal/scheduler/animationFrame.js","./node_modules/rxjs/dist/esm/internal/scheduler/AnimationFrameScheduler.js","./node_modules/rxjs/dist/esm/internal/scheduler/AnimationFrameAction.js","./node_modules/rxjs/dist/esm/internal/util/Immediate.js","./node_modules/rxjs/dist/esm/internal/scheduler/immediateProvider.js","./node_modules/rxjs/dist/esm/internal/scheduler/asap.js","./node_modules/rxjs/dist/esm/internal/scheduler/AsapScheduler.js","./node_modules/rxjs/dist/esm/internal/scheduler/AsapAction.js","./node_modules/rxjs/dist/esm/internal/scheduler/async.js","./node_modules/rxjs/dist/esm/internal/scheduler/dateTimestampProvider.js","./node_modules/rxjs/dist/esm/internal/scheduler/timeoutProvider.js","./node_modules/rxjs/dist/esm/internal/symbol/iterator.js","./node_modules/rxjs/dist/esm/internal/symbol/observable.js","./node_modules/rxjs/dist/esm/internal/util/EmptyError.js","./node_modules/rxjs/dist/esm/internal/util/args.js","./node_modules/rxjs/dist/esm/internal/util/argsArgArrayOrObject.js","./node_modules/rxjs/dist/esm/internal/util/arrRemove.js","./node_modules/rxjs/dist/esm/internal/util/createErrorClass.js","./node_modules/rxjs/dist/esm/internal/util/createObject.js","./node_modules/rxjs/dist/esm/internal/util/errorContext.js","./node_modules/rxjs/dist/esm/internal/util/executeSchedule.js","./node_modules/rxjs/dist/esm/internal/util/identity.js","./node_modules/rxjs/dist/esm/internal/util/isArrayLike.js","./node_modules/rxjs/dist/esm/internal/util/isAsyncIterable.js","./node_modules/rxjs/dist/esm/internal/util/isFunction.js","./node_modules/rxjs/dist/esm/internal/util/isInteropObservable.js","./node_modules/rxjs/dist/esm/internal/util/isIterable.js","./node_modules/rxjs/dist/esm/internal/util/isObservable.js","./node_modules/rxjs/dist/esm/internal/util/isPromise.js","./node_modules/rxjs/dist/esm/internal/util/isReadableStreamLike.js","./node_modules/rxjs/dist/esm/internal/util/isScheduler.js","./node_modules/rxjs/dist/esm/internal/util/lift.js","./node_modules/rxjs/dist/esm/internal/util/mapOneOrManyArgs.js","./node_modules/rxjs/dist/esm/internal/util/noop.js","./node_modules/rxjs/dist/esm/internal/util/pipe.js","./node_modules/rxjs/dist/esm/internal/util/reportUnhandledError.js","./node_modules/rxjs/dist/esm/internal/util/throwUnobservableError.js","./node_modules/ts-pattern/lib/guards.js","./node_modules/ts-pattern/lib/index.js","./node_modules/ts-pattern/lib/symbols.js","./node_modules/ts-pattern/lib/wildcards.js","./node_modules/wildcard/index.js","./node_modules/zone.js/dist/zone.js","./node_modules/moment/locale/ sync ^\\.\\/.*$","./node_modules/@uppy/core/lib/BasePlugin.js","./node_modules/@uppy/core/lib/Restricter.js","./node_modules/@uppy/core/lib/UIPlugin.js","./node_modules/@uppy/core/lib/Uppy.js","./node_modules/@uppy/core/lib/getFileName.js","./node_modules/@uppy/core/lib/index.js","./node_modules/@uppy/core/lib/locale.js","./node_modules/@uppy/core/lib/loggers.js","./node_modules/@uppy/core/lib/supportsUploadProgress.js","./node_modules/@uppy/dashboard/lib/Dashboard.js","./node_modules/@uppy/dashboard/lib/components/AddFiles.js","./node_modules/@uppy/dashboard/lib/components/AddFilesPanel.js","./node_modules/@uppy/dashboard/lib/components/Dashboard.js","./node_modules/@uppy/dashboard/lib/components/EditorPanel.js","./node_modules/@uppy/dashboard/lib/components/FileCard/index.js","./node_modules/@uppy/dashboard/lib/components/FileItem/Buttons/index.js","./node_modules/@uppy/dashboard/lib/components/FileItem/FileInfo/index.js","./node_modules/@uppy/dashboard/lib/components/FileItem/FilePreviewAndLink/index.js","./node_modules/@uppy/dashboard/lib/components/FileItem/FileProgress/index.js","./node_modules/@uppy/dashboard/lib/components/FileItem/MetaErrorMessage.js","./node_modules/@uppy/dashboard/lib/components/FileItem/index.js","./node_modules/@uppy/dashboard/lib/components/FileList.js","./node_modules/@uppy/dashboard/lib/components/FilePreview.js","./node_modules/@uppy/dashboard/lib/components/PickerPanelContent.js","./node_modules/@uppy/dashboard/lib/components/PickerPanelTopBar.js","./node_modules/@uppy/dashboard/lib/components/Slide.js","./node_modules/@uppy/dashboard/lib/components/VirtualList.js","./node_modules/@uppy/dashboard/lib/index.js","./node_modules/@uppy/dashboard/lib/locale.js","./node_modules/@uppy/dashboard/lib/utils/copyToClipboard.js","./node_modules/@uppy/dashboard/lib/utils/createSuperFocus.js","./node_modules/@uppy/dashboard/lib/utils/getActiveOverlayEl.js","./node_modules/@uppy/dashboard/lib/utils/getFileTypeIcon.js","./node_modules/@uppy/dashboard/lib/utils/ignoreEvent.js","./node_modules/@uppy/dashboard/lib/utils/trapFocus.js","./node_modules/@uppy/drag-drop/lib/DragDrop.js","./node_modules/@uppy/drag-drop/lib/index.js","./node_modules/@uppy/drag-drop/lib/locale.js","./node_modules/@uppy/image-editor/lib/Editor.js","./node_modules/@uppy/image-editor/lib/ImageEditor.js","./node_modules/@uppy/image-editor/lib/index.js","./node_modules/@uppy/image-editor/lib/locale.js","./node_modules/@uppy/informer/lib/FadeIn.js","./node_modules/@uppy/informer/lib/Informer.js","./node_modules/@uppy/informer/lib/TransitionGroup.js","./node_modules/@uppy/informer/lib/index.js","./node_modules/@uppy/locales/lib/fr_FR.js","./node_modules/@uppy/progress-bar/lib/ProgressBar.js","./node_modules/@uppy/progress-bar/lib/index.js","./node_modules/@uppy/status-bar/lib/Components.js","./node_modules/@uppy/status-bar/lib/StatusBar.js","./node_modules/@uppy/status-bar/lib/StatusBarStates.js","./node_modules/@uppy/status-bar/lib/_StatusBar.js","./node_modules/@uppy/status-bar/lib/calculateProcessingProgress.js","./node_modules/@uppy/status-bar/lib/index.js","./node_modules/@uppy/status-bar/lib/locale.js","./node_modules/@uppy/store-default/lib/index.js","./node_modules/@uppy/thumbnail-generator/lib/index.js","./node_modules/@uppy/thumbnail-generator/lib/locale.js","./node_modules/@uppy/utils/lib/FOCUSABLE_ELEMENTS.js","./node_modules/@uppy/utils/lib/Translator.js","./node_modules/@uppy/utils/lib/dataURItoBlob.js","./node_modules/@uppy/utils/lib/findAllDOMElements.js","./node_modules/@uppy/utils/lib/findDOMElement.js","./node_modules/@uppy/utils/lib/generateFileID.js","./node_modules/@uppy/utils/lib/getBytesRemaining.js","./node_modules/@uppy/utils/lib/getDroppedFiles/index.js","./node_modules/@uppy/utils/lib/getDroppedFiles/utils/fallbackApi.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/getFileNameAndExtension.js","./node_modules/@uppy/utils/lib/getFileType.js","./node_modules/@uppy/utils/lib/getSpeed.js","./node_modules/@uppy/utils/lib/getTextDirection.js","./node_modules/@uppy/utils/lib/getTimeStamp.js","./node_modules/@uppy/utils/lib/hasProperty.js","./node_modules/@uppy/utils/lib/isDOMElement.js","./node_modules/@uppy/utils/lib/isDragDropSupported.js","./node_modules/@uppy/utils/lib/isObjectURL.js","./node_modules/@uppy/utils/lib/isPreviewSupported.js","./node_modules/@uppy/utils/lib/mimeTypes.js","./node_modules/@uppy/utils/lib/prettyETA.js","./node_modules/@uppy/utils/lib/secondsToTime.js","./node_modules/@uppy/utils/lib/toArray.js","./node_modules/@uppy/utils/lib/truncateString.js","./node_modules/classnames/index.js","./node_modules/nanoid/non-secure/index.cjs","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/AsyncGenerator.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/AwaitValue.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/asyncGeneratorDelegate.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/asyncIterator.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/asyncToGenerator.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/awaitAsyncGenerator.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/wrapAsyncGenerator.js","./node_modules/@angular/animations/fesm2020/animations.mjs","./node_modules/@angular/cdk/fesm2020/a11y.mjs","./node_modules/@angular/cdk/fesm2020/bidi.mjs","./node_modules/@angular/cdk/fesm2020/coercion.mjs","./node_modules/@angular/cdk/fesm2020/collections.mjs","./node_modules/@angular/cdk/fesm2020/keycodes.mjs","./node_modules/@angular/cdk/fesm2020/layout.mjs","./node_modules/@angular/cdk/fesm2020/observers.mjs","./node_modules/@angular/cdk/fesm2020/overlay.mjs","./node_modules/@angular/cdk/fesm2020/platform.mjs","./node_modules/@angular/cdk/fesm2020/portal.mjs","./node_modules/rxjs/dist/esm/internal/operators/auditTime.js","./node_modules/rxjs/dist/esm/internal/operators/audit.js","./node_modules/@angular/cdk/fesm2020/scrolling.mjs","./node_modules/@angular/common/fesm2020/common.mjs","./node_modules/@angular/common/fesm2020/http.mjs","./node_modules/@angular/core/fesm2020/core.mjs","./node_modules/@angular/forms/fesm2020/forms.mjs","./node_modules/@angular/material/fesm2020/autocomplete.mjs","./node_modules/@angular/material/fesm2020/button.mjs","./node_modules/@angular/material/fesm2020/chips.mjs","./node_modules/@angular/material/fesm2020/core.mjs","./node_modules/@angular/cdk/fesm2020/dialog.mjs","./node_modules/@angular/material/fesm2020/dialog.mjs","./node_modules/@angular/material/fesm2020/form-field.mjs","./node_modules/@angular/material/fesm2020/icon.mjs","./node_modules/@angular/cdk/fesm2020/text-field.mjs","./node_modules/@angular/material/fesm2020/input.mjs","./node_modules/@angular/material/fesm2020/menu.mjs","./node_modules/@angular/material/fesm2020/select.mjs","./node_modules/@angular/platform-browser/fesm2020/platform-browser.mjs","./node_modules/rxjs/dist/esm/internal/operators/refCount.js","./node_modules/rxjs/dist/esm/internal/observable/ConnectableObservable.js","./node_modules/rxjs/dist/esm/internal/operators/scan.js","./node_modules/rxjs/dist/esm/internal/operators/takeLast.js","./node_modules/rxjs/dist/esm/internal/operators/last.js","./node_modules/@angular/router/fesm2020/router.mjs","./node_modules/rxjs/dist/esm/internal/util/argsOrArgArray.js","./node_modules/rxjs/dist/esm/internal/observable/race.js","./node_modules/rxjs/dist/esm/internal/observable/never.js","./node_modules/rxjs/dist/esm/internal/observable/zip.js","./node_modules/rxjs/dist/esm/internal/operators/withLatestFrom.js","./node_modules/@popperjs/core/lib/utils/getOppositePlacement.js","./node_modules/@popperjs/core/lib/utils/getBasePlacement.js","./node_modules/@popperjs/core/lib/utils/getOppositeVariationPlacement.js","./node_modules/@popperjs/core/lib/enums.js","./node_modules/@popperjs/core/lib/dom-utils/getWindow.js","./node_modules/@popperjs/core/lib/dom-utils/instanceOf.js","./node_modules/@popperjs/core/lib/dom-utils/getDocumentElement.js","./node_modules/@popperjs/core/lib/utils/math.js","./node_modules/@popperjs/core/lib/utils/userAgent.js","./node_modules/@popperjs/core/lib/dom-utils/isLayoutViewport.js","./node_modules/@popperjs/core/lib/dom-utils/getBoundingClientRect.js","./node_modules/@popperjs/core/lib/dom-utils/getWindowScroll.js","./node_modules/@popperjs/core/lib/dom-utils/getWindowScrollBarX.js","./node_modules/@popperjs/core/lib/dom-utils/getComputedStyle.js","./node_modules/@popperjs/core/lib/dom-utils/getNodeName.js","./node_modules/@popperjs/core/lib/dom-utils/getParentNode.js","./node_modules/@popperjs/core/lib/dom-utils/isScrollParent.js","./node_modules/@popperjs/core/lib/dom-utils/getScrollParent.js","./node_modules/@popperjs/core/lib/dom-utils/listScrollParents.js","./node_modules/@popperjs/core/lib/dom-utils/isTableElement.js","./node_modules/@popperjs/core/lib/dom-utils/getOffsetParent.js","./node_modules/@popperjs/core/lib/dom-utils/contains.js","./node_modules/@popperjs/core/lib/utils/rectToClientRect.js","./node_modules/@popperjs/core/lib/dom-utils/getClippingRect.js","./node_modules/@popperjs/core/lib/dom-utils/getViewportRect.js","./node_modules/@popperjs/core/lib/dom-utils/getDocumentRect.js","./node_modules/@popperjs/core/lib/utils/getVariation.js","./node_modules/@popperjs/core/lib/utils/getMainAxisFromPlacement.js","./node_modules/@popperjs/core/lib/utils/computeOffsets.js","./node_modules/@popperjs/core/lib/utils/mergePaddingObject.js","./node_modules/@popperjs/core/lib/utils/getFreshSideObject.js","./node_modules/@popperjs/core/lib/utils/expandToHashMap.js","./node_modules/@popperjs/core/lib/utils/detectOverflow.js","./node_modules/@popperjs/core/lib/modifiers/flip.js","./node_modules/@popperjs/core/lib/utils/computeAutoPlacement.js","./node_modules/@popperjs/core/lib/utils/within.js","./node_modules/@popperjs/core/lib/dom-utils/getLayoutRect.js","./node_modules/@popperjs/core/lib/modifiers/preventOverflow.js","./node_modules/@popperjs/core/lib/utils/getAltAxis.js","./node_modules/@popperjs/core/lib/modifiers/arrow.js","./node_modules/@popperjs/core/lib/dom-utils/getCompositeRect.js","./node_modules/@popperjs/core/lib/dom-utils/getNodeScroll.js","./node_modules/@popperjs/core/lib/dom-utils/getHTMLElementScroll.js","./node_modules/@popperjs/core/lib/utils/orderModifiers.js","./node_modules/@popperjs/core/lib/utils/debounce.js","./node_modules/@popperjs/core/lib/createPopper.js","./node_modules/@popperjs/core/lib/utils/mergeByName.js","./node_modules/@popperjs/core/lib/modifiers/eventListeners.js","./node_modules/@popperjs/core/lib/modifiers/computeStyles.js","./node_modules/@popperjs/core/lib/popper-lite.js","./node_modules/@popperjs/core/lib/modifiers/popperOffsets.js","./node_modules/@popperjs/core/lib/modifiers/applyStyles.js","./node_modules/@ng-bootstrap/ng-bootstrap/fesm2020/ng-bootstrap.mjs","./node_modules/rxjs/dist/esm/internal/operators/endWith.js","./node_modules/@uppy/angular/fesm2020/uppy-angular.mjs","./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/systemZone.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","./node_modules/luxon/src/luxon.js","./node_modules/marked/lib/marked.esm.js","./node_modules/@angular-devkit/build-angular/node_modules/@babel/runtime/helpers/esm/asyncToGenerator.js","./node_modules/@babel/runtime/helpers/esm/defineProperty.js","./node_modules/@babel/runtime/helpers/esm/toPropertyKey.js","./node_modules/@babel/runtime/helpers/esm/toPrimitive.js","./node_modules/@babel/runtime/helpers/esm/typeof.js","./node_modules/tslib/tslib.es6.mjs"],"sourceRoot":"webpack:///","sourcesContent":["import { ArticleBooking } from '@app/modules/orders/models/order';\nimport { Address } from '@app/modules/shared/models/address';\nimport { Filters } from '@app/modules/shared/models/filters';\nimport { Ticket } from '@app/modules/shared/models/ticket';\nimport { Voucher } from '@app/modules/shared/models/voucher';\nimport { SubscriptionDetails } from '@app/modules/subscriptions/models/subscription.interface';\n\nexport interface CustomerComment {\n id: number;\n user_id: number;\n author_id: number;\n network_id: number;\n text: string;\n created_at: string;\n deleted_at?: any;\n}\n\nexport interface Media {\n id: string;\n code: string;\n type: string;\n user_id: string;\n network_id: string;\n provider_id: string;\n created_at: string;\n}\n\nexport interface Customer {\n customer_id?: string;\n network_ids?: number[];\n firstname?: string;\n lastname?: string;\n fullname?: string;\n email?: string;\n phone?: string;\n address?: Address;\n picture?: string;\n date?: string; // date\n birthday?: string;\n fb_url?: string;\n documents?: Document[];\n medias?: Media[];\n subscriptions?: SubscriptionDetails[];\n wallet?: Ticket[];\n parent?: Customer;\n profiles?: Customer[];\n isParent?: boolean;\n status?: string;\n universal?: boolean;\n active?: any;\n vouchers?: Voucher[];\n attributes?: Attribute[];\n provider?: 'airweb' | 'sncf';\n anonymized_at?: string;\n verifiedAt?: string;\n authLastAsGuest?: boolean;\n}\n\nexport interface CustomerFusion {\n id: number;\n firstName: string;\n lastName: string;\n birthdate: string;\n email: string;\n phone: string;\n address?: {\n lines?: string[];\n city?: string;\n zip?: string;\n state?: string;\n };\n attributeValues?: {\n attributeId?: number;\n attributeKey?: string;\n attributeType?: string;\n attributeNetworkId?: number;\n value?: string;\n }[];\n updatedAt: string;\n isFusionable: boolean;\n}\n\nexport interface CustomerFusionConfiguration {\n email: 'FORMER' | 'RECIPIENT';\n birthdate: 'FORMER' | 'RECIPIENT';\n phone: 'FORMER' | 'RECIPIENT';\n address: 'FORMER' | 'RECIPIENT';\n attributeValues?: {\n [key: number]: 'FORMER' | 'RECIPIENT';\n };\n}\n\nexport interface Document {\n id: string;\n deleted_at: string;\n file_deleted_at: string;\n expires_at: string;\n examined_at: string;\n label: string;\n name: string;\n status: string;\n type_id: string;\n url: string;\n file_id?: string;\n file_label?: string;\n}\n\nexport interface Attribute {\n key: string;\n network_id: number;\n type: string;\n value: any;\n label?: string;\n}\n\nexport interface SetCustomerAttributeResponse {\n response: {\n errorMessage?: string;\n attribute_id: number;\n id: number;\n user_id: number;\n value: any;\n };\n}\n\nexport interface AnonymizeResponse {\n response: {\n errorMessage?: string;\n errors: {\n code: number;\n message: string;\n }[];\n customers: number[];\n };\n}\n\nexport interface SendResetPasswordEmailResponse {\n response: {\n errorMessage?: string;\n error: {\n code: number;\n message: string;\n }[];\n success: boolean;\n };\n}\n\nexport interface UpdateInfoResponse {\n response: {\n errorMessage?: string;\n error: {\n code: number;\n message: string;\n }[];\n success: boolean;\n };\n}\n\nexport interface CustomerFilters extends Filters {\n provider?: string;\n attribute?: number;\n attribute_search?: string;\n}\n\nexport interface Contract {\n blockedAt?: string;\n blueprintId: string;\n blueprintName: string;\n code?: string;\n createdAt: string;\n customerId: string;\n data?: string;\n deletedAt?: string;\n endingAt?: string;\n id: string;\n networkId?: string;\n network?: string;\n owner?: string;\n startingAt?: string;\n state: ContractState;\n updatedAt: string;\n vendor: string;\n transferredAt?: string;\n orderId: string;\n orderCode: string;\n operatorId: string;\n trip?: ContractTrip;\n subscriptionId: number | null;\n}\n\nexport interface ContractState {\n contractEndDate: string;\n contractStartDate: string;\n inspectionEndDate: string;\n validationEndDate: string;\n validationRemainingPunches: number;\n validationRemainingTransfers: number;\n validationStartDate: string;\n validationType: string;\n}\n\nexport interface ContractTrip {\n origin: {\n url: string;\n label: string;\n };\n destination: {\n url: string;\n label: string;\n };\n}\n\nexport enum ContractStatus {\n CANCELED = 'CANCELED',\n EXPIRED = 'EXPIRED',\n REMAINING = 'REMAINING',\n PENDING = 'PENDING',\n EMPTY = 'EMPTY',\n ACTIVE = 'ACTIVE',\n}\n\nexport interface Transfer {\n id: string;\n status: string;\n networkId: number;\n issuerCustomerId: string;\n recipientCustomerId: string;\n providerId: number;\n providerSlug: string;\n contractLabel: string;\n contractCode: string;\n contractType: string;\n updatedAt: string;\n createdAt: string;\n direction?: 'IN' | 'OUT';\n issuerCustomerLabel?: string;\n recipientCustomerLabel?: string;\n}\n\nexport interface TransfersHistory {\n customers: [\n {\n [key: string]: {\n id: string;\n avatarUrl: string;\n firstName: string;\n lastName: string;\n email: string;\n };\n }\n ];\n transfers: [Transfer];\n}\n\nexport interface CustomerArticle {\n id: string;\n productId: string;\n productName: string;\n price: number;\n recipient: {\n id: string;\n firstName: string;\n lastName: string;\n };\n origin: string;\n destination: string;\n reservation: ArticleBooking | null;\n quantity: number;\n}\nexport interface CustomerOrders {\n id: string;\n code: string;\n purchaseDate: string;\n customerId: string;\n networkId: number;\n articles: CustomerArticle[];\n status: CustomerOrderStatus;\n}\n\nexport enum CustomerOrderStatus {\n CANCELLED = 'CANCELLED',\n COMPLETED = 'COMPLETED',\n ERROR = 'ERROR',\n PENDING = 'PENDING',\n REFUNDED = 'REFUNDED',\n REGULARIZATION = 'REGULARIZATION',\n TO_REFUND = 'TO_REFUND',\n}\n\nexport interface Session {\n originType: SessionOriginType | null;\n originAgent: string | null;\n date: string;\n active: boolean;\n providerId: string | null;\n}\n\nexport enum SessionOriginType {\n APP = 'APP',\n ESHOP = 'ESHOP',\n SCHOOL = 'SCHOOL',\n RETAIL = 'RETAIL',\n}\n\nexport interface CustomerValidation {\n clientSlug: string;\n isTransfer: boolean;\n itemId: number;\n itemLabel: string;\n lineCode: null;\n mediaId: null;\n mediaType: null;\n networkId: number;\n occuredAt: string;\n passengers: number;\n provider: string;\n status: string;\n stopCode: string;\n ticketId: number;\n tripDestinationLabel: string;\n tripOriginLabel: string;\n universal: boolean;\n validationNetworkId: number;\n vehicleCode: string;\n vehicleId: number;\n}\n\nexport interface CustomerWalletEvent {\n id: string;\n contractId: string;\n type: CustomerWalletEventType;\n contractNetworkId: string;\n passengers: number;\n data: { [key: string]: any };\n occurredAt: string;\n createdAt: string;\n location: {\n lineCode: string;\n lineName: string;\n linePicto: string;\n mediaCode: string;\n mediaType: string;\n stationCode: string;\n stationName: string;\n vehicleCode: string;\n vehicleType: string;\n tripDirection: string;\n };\n}\n\nexport enum CustomerWalletEventType {\n PUNCH = 'PUNCH',\n TRANSFER = 'TRANSFER',\n CHECK = 'CHECK',\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { Address } from '@app/modules/shared/models/address';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { Ticket } from '@app/modules/shared/models/ticket';\nimport { Voucher } from '@app/modules/shared/models/voucher';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { SubscriptionDetails } from '@app/modules/subscriptions/models/subscription.interface';\nimport { CacheManager } from '@app/utils/cache-manager';\nimport { environment as env } from '@env/environment';\nimport { BehaviorSubject, firstValueFrom, Observable, of } from 'rxjs';\nimport { catchError, map, switchMap } from 'rxjs/operators';\nimport { match } from 'ts-pattern';\nimport {\n AnonymizeResponse,\n Attribute,\n Contract,\n Customer,\n CustomerFusion,\n CustomerFusionConfiguration,\n CustomerOrders,\n CustomerOrderStatus,\n CustomerValidation,\n CustomerWalletEvent,\n CustomerWalletEventType,\n Document,\n SendResetPasswordEmailResponse,\n Session,\n SetCustomerAttributeResponse,\n TransfersHistory,\n UpdateInfoResponse,\n} from '../models/customer';\nimport { CustomersFiltersService } from './filters.service';\nimport { Identity } from '../models/identity';\n\nexport interface $$Comment {\n id: number;\n customer_id: number;\n author_id?: number;\n network_id?: number;\n text: string;\n created_at: string;\n deleted_at?: string;\n author?: unknown;\n}\n\nexport interface $$Ticket {\n id: string;\n ticket_id: string;\n user_id: string;\n order_identifier?: string;\n order_id?: string;\n order_date?: string;\n product_id: string;\n start_date?: string;\n end_date?: string;\n label: string;\n status: string;\n network_id: string;\n dematerialized: string;\n category_id?: string;\n cms_user_id?: string;\n subscription_id?: number;\n created_at: string;\n}\n\nexport interface $$Profile {\n customer_id: string;\n isParent: boolean;\n firstname: string;\n lastname: string;\n status: string;\n picture: string;\n active: boolean;\n birthdate: string;\n anonymized_at?: any;\n}\n\nexport interface $$Customer {\n customer_id: string;\n network_ids: number[];\n firstname: string;\n lastname: string;\n email: string;\n phone: string;\n address: {\n streetNumber: string;\n route: string;\n city: string;\n zipCode: string;\n country: string;\n };\n picture: string;\n date: string;\n birthday: string;\n active: string;\n lastActivityAt?: string | null;\n anonymized_at?: any;\n universal: boolean;\n provider: string;\n provider_uid: string;\n locale: string | null;\n verified_at?: string;\n emailBouncedAt: string | null;\n emailBouncedReason: EmailBouncedReason | null;\n emailBouncedUntil: string | null;\n isFusionable: boolean;\n firebaseUid: string | null;\n}\n\nexport enum EmailBouncedReason {\n TEMPORARY_MAILBOX_FULL = 'TEMPORARY_MAILBOX_FULL',\n TEMPORARY_ERROR = 'TEMPORARY_ERROR',\n PERMANENT_EMAIL_ADDRESS_UNKNOWN = 'PERMANENT_EMAIL_ADDRESS_UNKNOWN',\n PERMANENT_EMAIL_ADDRESS_INCORRECT = 'PERMANENT_EMAIL_ADDRESS_INCORRECT',\n PERMANENT_ERROR = 'PERMANENT_ERROR',\n UNKNOWN_ERROR = 'UNKNOWN_ERROR',\n}\n\nexport interface $$GlobalAttribute {\n id: string;\n key: string;\n network_id: string;\n label: string;\n type: 'ENUM' | 'BOOLEAN';\n values: string[];\n}\n\n@Injectable()\nexport class CustomersService {\n private cache = new CacheManager('customers');\n\n constructor(\n private http: HttpClient,\n private authService: AuthService,\n private router: Router,\n private globalFilters: FilterService,\n private customerFilters: CustomersFiltersService\n ) {}\n\n public getCustomers(): Observable {\n let version = this.cache.getVersion();\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get>(\n env.config.feedRoot + `Customer/list.json`,\n {\n params: {\n v: version,\n ...this.globalFilters.filtersWithID,\n ...this.customerFilters.getCleanValue(),\n },\n }\n )\n ),\n\n map((payload) => {\n // handle empty response as empty array\n if (!payload) return [];\n\n return payload.response.customers;\n })\n );\n }\n\n public deactivateCustomer(customerId: string): Promise {\n const http$ = this.http\n .put>(env.config.feedRoot + 'Customer/deactivate', {\n id: customerId,\n })\n .pipe(\n map(({ response }) => {\n return response.success;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public reactivateCustomer(customerId: string): Promise {\n const http$ = this.http\n .put>(env.config.feedRoot + 'Customer/reactivate', {\n id: customerId,\n })\n .pipe(\n map(({ response }) => {\n return response.success;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public $$getProfiles(customerId: string): Promise<$$Profile[]> {\n const http$ = this.http\n .get>(env.config.feedRoot + `Customer/profiles`, {\n params: { customerId },\n })\n .pipe(map(({ response }) => response.profiles));\n\n return firstValueFrom(http$);\n }\n\n public getProfiles(customerId: string) {\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + `Customer/profiles.json`,\n {\n params: { customerId },\n }\n )\n .pipe(map(({ response }) => response.profiles))\n );\n }\n\n public $$getCustomer(customerId: string): Promise<$$Customer> {\n const http$ = this.http\n .get>(env.config.feedRoot + `Customer/details`, {\n params: { customerId },\n })\n .pipe(\n map(({ response }) => {\n if (response.customer === null) {\n this.router.navigate(['/403']);\n }\n\n return response.customer;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getCustomerDetails(customerId: string) {\n const version = this.cache.getVersion();\n\n const http$ = this.http\n .get>(`${env.config.feedRoot}Customer/details.json`, {\n params: { version, customerId },\n })\n .pipe(\n map(({ response }) => {\n if (response.customer === null) {\n this.router.navigate(['/403']);\n }\n\n return response.customer;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public statusType = {\n PENDING: 'Validé',\n ACTIVE: 'A valider',\n REMAINING: 'Restant',\n EXPIRED: 'Expiré',\n };\n\n public $$getWallet(customerId: string): Promise<$$Ticket[]> {\n const http$ = this.http\n .get>(env.config.feedRoot + `Customer/wallet`, {\n params: { customerId },\n })\n .pipe(\n map(({ response }) => {\n return response.wallet.sort(function (a, b) {\n // Turn your strings into dates, and then subtract them\n // to get a value that is either negative, positive, or zero.\n return new Date(b.created_at).getTime() - new Date(a.created_at).getTime();\n });\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getWallet(customerId: string) {\n return firstValueFrom(\n this.http\n .get>(env.config.feedRoot + `Customer/wallet.json`, {\n params: { customerId },\n })\n .pipe(map(({ response }) => response.wallet))\n );\n }\n\n public $$getContracts(customerId: string): Promise {\n const http$ = this.http\n .get>(\n `${env.config.feedRoot}Wallet/contracts.json`,\n {\n params: { customerId },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.contracts;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getContracts(customerId: number) {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Wallet/contracts.json`,\n {\n params: { customerId },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.contracts;\n })\n )\n );\n }\n\n public $$getVouchers(customerId: string): Promise {\n const http$ = this.http\n .get>(env.config.feedRoot + `Customer/vouchers`, {\n params: { customerId },\n })\n .pipe(map(({ response }) => response.vouchers));\n\n return firstValueFrom(http$);\n }\n\n public $$getDocuments(customerId: string): Promise {\n const http$ = this.http\n .get>(env.config.feedRoot + `Customer/documents`, {\n params: { customerId },\n })\n .pipe(\n map(({ response }) => {\n return response.documents;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getCustomerAttributes(customerId: string): Promise {\n return firstValueFrom(\n this.http\n .get>(env.config.feedRoot + `Customer/attributeValues`, {\n params: { customerId },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n public uploadImage(id: string, file: File) {\n let formData: FormData = new FormData();\n formData.append('media', file);\n formData.append('id', id);\n\n return this.http.post(env.config.feedRoot + `Customer/updateAvatar`, formData).pipe(\n map(({ response: r }) => {\n this.cache.incrementVersion();\n return r;\n })\n );\n }\n\n public searchCustomer(query: string): Observable {\n const searchUrl = new URL(`${env.config.feedRoot}Customer/search`);\n\n const searchParams = new URLSearchParams({\n ...this.globalFilters.filtersWithID,\n query,\n limit: '5',\n });\n searchUrl.search = searchParams.toString();\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get>(searchUrl.toString())\n ),\n\n map(({ response }) => {\n return response.customers;\n })\n );\n }\n\n public getSearchedCustomers(query: string): Observable {\n const searchUrl = new URL(`${env.config.feedRoot}Customer/searchedCustomers`);\n\n const searchParams = new URLSearchParams({ ...this.globalFilters.filtersWithID, query });\n searchUrl.search = searchParams.toString();\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get>(searchUrl.toString())\n ),\n\n map(({ response }) => {\n return response.customers;\n })\n );\n }\n\n public $$getGlobalAttributes(): Promise<$$GlobalAttribute[]> {\n const http$ = this.http\n .get>(\n `${env.config.feedRoot}Customer/attributes`\n )\n .pipe(map(({ response }) => response.attributes));\n\n return firstValueFrom(http$);\n }\n\n public getAttributes(): Promise {\n return firstValueFrom(\n this.http.get(`${env.config.feedRoot}Customer/attributes.json`)\n ).then(({ response }) => response.attributes);\n }\n\n public setCustomerAttribute(customerId, attributeId, value) {\n const body = {\n customerId,\n attributeId,\n value,\n };\n\n return firstValueFrom(\n this.http\n .post(\n `${env.config.feedRoot}Customer/setAttributeValue.json`,\n body\n )\n .pipe(map(({ response }: any) => response))\n );\n }\n\n public $$getSubscriptions(\n parentProfileId: string,\n customerId: string\n ): Promise {\n const http$ = this.http\n .get>(\n env.config.feedRoot + `Customer/getSubscriptions`,\n {\n params: { customerId: parentProfileId },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.filter((subscription) => +subscription.recipient_user_id === +customerId);\n })\n );\n\n return firstValueFrom(http$);\n }\n\n // TODO: Type this\n public $$getMedias(customerId: string) {\n const http$ = this.http\n .get(env.config.feedRoot + `Customer/medias`, {\n params: { customerId },\n })\n .pipe(map(({ response }) => response.medias));\n\n return firstValueFrom(http$);\n }\n\n public $$getComments(\n customerId: string,\n filters?: { networkId: string | number }\n ): Promise<$$Comment[]> {\n const http$ = this.http\n .get>(\n `${env.config.feedRoot}Customer/getCommentsAboutCustomer`,\n {\n params: { customerId, ...filters, antiCache: Math.random().toString(36).slice(2, 5) },\n }\n )\n .pipe(\n map(({ response }) => (response.errorMessage ? [] : response)),\n map((comments) =>\n filters?.networkId !== undefined\n ? comments.filter((comment) => +comment.network_id === +filters.networkId)\n : comments\n )\n );\n\n return firstValueFrom(http$);\n }\n\n public getCustomerComments(customerId: string) {\n const value$ = new BehaviorSubject(null);\n\n const http$ = this.http.get(\n `${env.config.feedRoot}Customer/getCommentsAboutCustomer.json`,\n {\n params: { customerId },\n }\n );\n\n const fetch = (filters: { networkId?: string } = {}) => {\n firstValueFrom(\n http$.pipe(\n // If we have an error, just return an empty array..\n map(({ response }) => (response.errorMessage ? [] : response)),\n map((comments) =>\n comments && filters?.networkId !== undefined\n ? comments.filter((comment) => +comment.network_id === +filters.networkId)\n : comments\n )\n )\n ).then((comments) => value$.next(comments));\n };\n\n return { value$, fetch } as const;\n }\n\n public addCustomerComment(customerId: string, body: { networkId: string; text: string }) {\n return this.http\n .post(`${env.config.feedRoot}Customer/addCommentAboutCustomer.json`, {\n customerId,\n ...body,\n })\n .pipe(map(({ response }) => response));\n }\n\n public async editCustomerComment(\n commentId: number,\n customerId: string,\n body: { networkId: string; text: string }\n ) {\n const deleteRequest = await this.deleteCustomerComment(String(commentId));\n if (!deleteRequest) return deleteRequest;\n\n return firstValueFrom(this.addCustomerComment(customerId, body));\n }\n\n public deleteCustomerComment(commentId: string) {\n const http$ = this.http\n .delete(`${env.config.feedRoot}Customer/removeCommentAboutCustomer`, {\n params: { commentId },\n })\n .pipe(map(({ response }) => response));\n\n return firstValueFrom(http$);\n }\n\n public anonymizeCustomer(customerId: string) {\n return firstValueFrom(\n this.http\n .delete(`${env.config.feedRoot}Customer/anonymize.json`, {\n params: { customerId },\n })\n .pipe(\n map((response) => {\n return response;\n })\n )\n );\n }\n\n public sendResetPasswordEmail(customerId: string, firebaseEmail?: string) {\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Customer/resetPassword.json`, {\n customerId,\n firebaseEmail,\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public updateInfo(\n customerId: string,\n params: {\n firstname?: string;\n lastname?: string;\n birthday?: string;\n email?: string;\n phone?: string;\n address?: Address;\n }\n ) {\n if (!params) return;\n\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Customer/updateInfo.json`, {\n customerId,\n ...params,\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public linkNetwork(customerId: number, networkId: number) {\n return firstValueFrom(\n this.http\n .post>(\n `${env.config.feedRoot}Customer/linkNetwork.json`,\n {\n customerId,\n networkId,\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public unlinkNetwork(customerId: number, networkId: number) {\n return firstValueFrom(\n this.http\n .delete>(\n `${env.config.feedRoot}Customer/unlinkNetwork.json`,\n {\n params: { customerId, networkId },\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public disableContract(contractId: string) {\n return firstValueFrom(\n this.http\n .put>(\n `${env.config.feedRoot}Wallet/updateContract.json`,\n {\n contractId,\n blockedAt: new Date().toISOString(),\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public reactivateContract(contractId: string) {\n return firstValueFrom(\n this.http\n .put>(\n `${env.config.feedRoot}Wallet/updateContract.json`,\n {\n contractId,\n blockedAt: null,\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public sendVerificationMail(customerId: string) {\n return firstValueFrom(\n this.http\n .post>(\n `${env.config.feedRoot}Customer/sendVerificationEmail`,\n {\n customerId: customerId,\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public releaseSessions(customerId: string) {\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Customer/releaseSessions`, {\n customerId,\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public getHistory(customerId: string): Promise {\n const http = this.http\n .get(`${env.config.feedRoot}Customer/transfers`, {\n params: {\n customerId,\n status: 'DONE',\n },\n })\n .pipe(map(({ response }) => response));\n return firstValueFrom(http);\n }\n\n public getCustomerOrders(customerId: string, status?: CustomerOrderStatus) {\n let params: { customerId: string; status?: CustomerOrderStatus } = {\n customerId,\n };\n\n if (status) {\n params = {\n status,\n ...params,\n };\n }\n\n const http = this.http\n .get>(`${env.config.feedRoot}Customer/orders`, {\n params,\n })\n .pipe(\n map(({ response }) => {\n return response?.orders;\n })\n );\n\n return firstValueFrom(http);\n }\n\n public getCustomerSessions(customerId: string) {\n return firstValueFrom(\n this.http\n .get>(`${env.config.feedRoot}Customer/sessions`, {\n params: { id: customerId },\n })\n .pipe(\n map(({ response }) => {\n return response.sessions;\n })\n )\n );\n }\n\n public getFusionList(\n customerId: string,\n query: string\n ): Promise> {\n const http$ = this.http\n .get>(\n env.config.feedRoot + `Customer/searchFusionable`,\n {\n params: { id: customerId, query },\n }\n )\n .pipe(map((response) => response));\n\n return firstValueFrom(http$);\n }\n\n public fusionCustomers(\n recipientCustomerId: number,\n formerCustomerId: number,\n configuration: CustomerFusionConfiguration\n ): Promise {\n const http$ = this.http\n .post>(`${env.config.feedRoot}Customer/fusion`, {\n recipientCustomerId: recipientCustomerId,\n formerCustomerId: formerCustomerId,\n configuration: JSON.stringify(configuration),\n })\n .pipe(\n catchError((error) => {\n console.error('Error on Customer/fusion flux', error);\n return of({ response: { success: false } });\n }),\n map(({ response }) => response?.success || false)\n );\n\n return firstValueFrom(http$);\n }\n\n public getValidations(customerId: number): Promise {\n const http$ = this.http\n .get>(\n `${env.config.feedRoot}Customer/validations`,\n { params: { customerId } }\n )\n .pipe(\n catchError((error) => {\n console.error('Error on Customer/validations', error);\n return of({ response: { validations: [] } });\n }),\n map(({ response }) => response.validations)\n );\n\n return firstValueFrom(http$);\n }\n\n public getWalletEvents(\n customerId: number,\n type?: CustomerWalletEventType[]\n ): Promise {\n const params = { customerId };\n\n if (type) {\n params['type'] = type.join(',');\n }\n\n const http$ = this.http\n .get>(\n `${env.config.feedRoot}Customer/walletEvents`,\n { params }\n )\n .pipe(\n catchError((error) => {\n console.error('Error on Customer/walletEvents', error);\n return of({ response: { events: [] } });\n }),\n map(({ response }) => response.events)\n );\n\n return firstValueFrom(http$);\n }\n\n public checkAccount(\n customerId: string\n ): Promise<{ success: boolean; error?: { code: string; message: string } }> {\n const http$ = this.http\n .post<{ response: { success: boolean; error?: { code: string; message: string } } }>(\n `${env.config.feedRoot}Customer/verify`,\n { customerId }\n )\n .pipe(\n catchError((error) => {\n console.error('Error on Customer/verify flux', error);\n const message = match(error?.error?.code)\n .with(\n 'CUSTOMER_VERIFICATION_CUSTOMER_INACTIVE',\n () => `pages.customer_details.account_check_validation_error_inactive`\n )\n .with(\n 'CUSTOMER_VERIFICATION_CUSTOMER_ACCOUNT_ALREADY_VERIFIED',\n () => `pages.customer_details.account_check_already_verified`\n )\n .with(\n 'CUSTOMER_VERIFICATION_CUSTOMER_ACCOUNT_INVALID',\n () => `pages.customer_details.account_check_account_invalid`\n )\n .otherwise(() => `otherslabels.unknown_error`);\n return of({ response: { success: false, error: { code: error?.error?.code, message } } });\n }),\n map(({ response }) => {\n return {\n success: response?.success || false,\n error: response?.error || null,\n };\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public identities(customerId: string): Promise {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}/Customer/identities`,\n {\n params: { customerId },\n }\n )\n .pipe(map(({ response }) => response.identities))\n );\n }\n\n public async deliverProduct(\n customerId: number,\n productId: number,\n support: string,\n quantity: number,\n generationReason: string,\n generationComment: string,\n data?: {\n tripToken?: string;\n validityStartDate?: string;\n validityEndDate?: string;\n isSubscription?: boolean;\n }\n ): Promise {\n return firstValueFrom(\n this.http\n .post>(`${env.config.feedRoot}Customer/deliverProduct`, {\n customerId,\n productId,\n support,\n quantity,\n generationReason,\n generationComment,\n ...data,\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n}\n","import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\n\n@Injectable()\nexport class CustomersFiltersService extends BehaviorSubject {\n constructor() {\n super({\n attribute: null,\n attribute_search: null,\n virtual_provider: null,\n });\n }\n\n /**\n * Getter retrieving value object without falsy keys\n */\n getCleanValue() {\n return Object.entries(this.value).reduce((a, [k, v]) => (v ? { ...a, [k]: v } : a), {});\n }\n}\n\nexport interface ICustomersFilters {\n attribute: string;\n attribute_search: string;\n virtual_provider: string;\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { Blueprint } from '@app/modules/shop/models/blueprint/blueprint';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, Observable, of } from 'rxjs';\nimport { map, switchMap } from 'rxjs/operators';\n\nexport interface $$Network {\n id: string;\n name: string;\n categories: $$Category[];\n}\n\nexport interface $$Category {\n id: string;\n name: string;\n items: $$Item[];\n}\n\nexport interface $$Item {\n id: string;\n name: string;\n is_od: boolean;\n products: $$Product[];\n}\n\nexport interface $$Product {\n id: string;\n name: string;\n photoMandatory: number;\n dematerialized?: null | string;\n}\n\n@Injectable()\nexport class ProductsService {\n constructor(private http: HttpClient, private authService: AuthService) { }\n\n public getProducts(): Observable<$$Network[]> {\n return this.http.get>(env.config.feedRoot + 'Product/getProducts.json').pipe(\n map(({ response }) => {\n return response.networks;\n })\n );\n }\n\n public $$getProducts(): Promise<$$Item[]> {\n const http$ = this.http\n .get>(env.config.feedRoot + 'Product/getProducts')\n .pipe(\n map(({ response }) => {\n return response.networks\n .sort((a, b) => a.name.localeCompare(b.name))\n .reduce((items, network) => {\n for (const category of network.categories) {\n items.push(...category.items);\n }\n\n return items;\n }, []);\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getNetworkItems(network_id: string) {\n return this.http\n .get(env.config.feedRoot + `Product/getNetworkItems.json`, {\n params: { network_id },\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n );\n }\n\n public getNetworkCategories(network_id: string) {\n return this.http\n .get(env.config.feedRoot + `Product/getNetworkCategories.json`, {\n params: { network_id },\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n );\n }\n\n public getNetworksBlueprints(params: Partial<{ network: string[] }> = {}) {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(`${env.config.feedRoot}Wallet/blueprints`, {\n params: {\n network: `[${params.network}]`,\n deleted: 0,\n },\n })\n ),\n\n map(({ response }) => {\n return response ? (response as Blueprint[]) : null;\n })\n );\n }\n\n public getDatePreview(params: {\n pivot_type: string;\n pivot_value: string;\n start_proposals: number;\n timezone?: string;\n }) {\n return this.http.get(`${env.config.feedRoot}product/getProductStartDates`, {\n params,\n });\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { IDocument } from '@app/modules/orders/models/document';\nimport { NetworkPaymentOptions, PayzenOptions } from '@app/modules/payment/models/commission';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { CacheManager } from '@app/utils/cache-manager';\nimport { environment as env } from '@env/environment';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { firstValueFrom, Observable } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { BillingInfo, Domain, NetworkInfo, NetworkProvider } from '../models/network-info';\nimport { NetworkFeature, NetworkFeatureHandle } from '@app/modules/shared/models/network';\nimport { NetworkInspectionConfiguration } from '../models/network-configuration';\n\n@Injectable()\nexport class NetworkService {\n private cache = new CacheManager('network');\n\n constructor(\n private http: HttpClient,\n private authService: AuthService,\n private notification: NotificationsService,\n private readonly translate: TranslateService\n ) {}\n\n public getNetworks({ withId = false }): Observable {\n let version = this.cache.getVersion();\n const file = withId ? `Network/getNetworks?withId=true&v=` : `Network/getNetworks?v=`;\n\n return this.http\n .get>(env.config.feedRoot + file + version)\n .pipe(\n map(({ response }) => {\n return response.networks;\n })\n );\n }\n\n public getNetworksDetails(id: number | string): Observable {\n const version = this.cache.getVersion();\n\n return this.http\n .get>(\n env.config.feedRoot + 'Network/getNetworks',\n {\n params: { id, version },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.networks[0];\n })\n );\n }\n\n public updateLocales(\n networkId: number | string,\n localesAsArray: { locale: string; equivalences: string[] | null }[]\n ) {\n const locales = localesAsArray.reduce((locales, { locale, equivalences }, index) => {\n if (!locale) return locales;\n\n locales[locale] = {\n equivalences: equivalences?.length ? equivalences : null,\n default: index === 0,\n priority: index + 1,\n };\n\n return locales;\n }, {});\n\n const response$ = this.http\n .put(`${env.config.feedRoot}Network/updateLocales`, { networkId, locales })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n throw new Error(response.errorMessage);\n }\n\n return response.success;\n })\n );\n\n return firstValueFrom(response$);\n }\n\n public updateNetwork(networkInfo: NetworkInfo, id: number, translations: any) {\n if ('subscriptions' in networkInfo.configuration) {\n // @ts-expect-error: The update feed that a different type than the get feed\n networkInfo.configuration.has_subscriptions = networkInfo.configuration.subscriptions;\n }\n\n if ('data' in networkInfo && networkInfo.data !== null) {\n networkInfo.data = JSON.parse(networkInfo.data);\n }\n\n return this.http\n .post>(env.config.feedRoot + `Network/editNetworks`, {\n networkInfo,\n id,\n translations,\n })\n .pipe(\n map(({ response }) => {\n this.cache.incrementVersion();\n\n if (response.errorMessage) {\n throw new Error(response.errorMessage);\n }\n\n return response.success;\n })\n );\n }\n\n public createNetwork(name: string): Observable<{ id: number }> {\n return this.http\n .post>(env.config.feedRoot + `Network/createNetwork`, {\n name,\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n if (response.errorCode !== null) throw new Error(response.errorCode);\n } else {\n this.cache.incrementVersion();\n return response;\n }\n })\n );\n }\n\n public getDocuments(networkId: number | string): Observable {\n return this.http\n .get(env.config.feedRoot + `Network/getDocuments`, {\n params: {\n networkId,\n },\n })\n .pipe(\n map(({ response }) => {\n return response.documents;\n })\n );\n }\n\n public getMails(): Observable {\n return this.http.get(env.config.feedRoot + `Network/getMails`).pipe(\n map(({ response }) => {\n return response.emails;\n })\n );\n }\n\n public getTags(): Observable {\n return this.http.get(env.config.feedRoot + `Network/getTags`).pipe(\n map(({ response }) => {\n return response.tags;\n })\n );\n }\n\n public setNetworkMailTags(datas: any): Observable {\n return this.http.post(env.config.feedRoot + `Network/setNetworkMailTags`, datas).pipe(\n map(({ response }) => {\n return response.success;\n })\n );\n }\n\n public editNetworkClients(networkId: string, isTixiPass: boolean, clients: number[]) {\n const body = { networkId, isTixiPass, clients };\n\n return this.http\n .post(`${env.config.feedRoot}Network/editNetworkClients`, body)\n .pipe(map(({ response }) => response));\n }\n\n public getBillingInfo(networkId: string) {\n return this.http\n .get>(\n env.config.feedRoot + 'Network/getBillingInfo',\n {\n params: { networkId },\n }\n )\n .pipe(map(({ response }) => response.billingInfos));\n }\n\n public addBillingInfo(networkId: string, info: Partial) {\n const body = { networkId: +networkId, ...info };\n return this.http\n .post(`${env.config.feedRoot}Network/addBillingInfo`, null, {\n params: body,\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n throw new Error(response.errorCode);\n } else {\n this.notification.success(\n this.translate.instant('pages.network_details.billing_info_added_notification')\n );\n return response;\n }\n })\n );\n }\n\n public updateBillingInfo(info: BillingInfo) {\n return this.http\n .put(`${env.config.feedRoot}Network/updateBillingInfo`, null, {\n params: { ...info },\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n throw new Error(response.errorCode);\n } else {\n this.notification.success(\n this.translate.instant('pages.network_details.billing_info_updated_notification')\n );\n return response;\n }\n })\n );\n }\n\n public deleteBillingInfo(id: number) {\n //id = billing field id\n return this.http\n .delete(`${env.config.feedRoot}Network/removeBillingInfo`, {\n params: { id },\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n throw new Error(response.errorCode);\n } else {\n this.notification.success(\n this.translate.instant('pages.network_details.billing_info_deleted_notification')\n );\n return response;\n }\n })\n );\n }\n\n public getDomains(networkId: number) {\n return firstValueFrom(\n this.http\n .get>(`${env.config.feedRoot}Network/getDomains`, {\n params: { networkId },\n })\n .pipe(\n map(({ response }) => {\n return response.domains;\n })\n )\n );\n }\n\n public addDomain(domainInfo: { networkId: string | number; domain: string; default: boolean }) {\n return firstValueFrom(\n this.http\n .post>(\n `${env.config.feedRoot}Network/addDomain`,\n domainInfo\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public editDomain(domainInfo: { networkId: string | number; domain: string; default: boolean }) {\n return firstValueFrom(\n this.http\n .put>(\n `${env.config.feedRoot}Network/editDomain`,\n domainInfo\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public deleteDomain(networkId: string | number, domain: string) {\n return firstValueFrom(\n this.http\n .delete>(\n `${env.config.feedRoot}Network/deleteDomain`,\n {\n params: { networkId, domain },\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public getProviders(networkId: number | string | undefined, onlyTicketing = false) {\n return firstValueFrom(\n this.http\n .get>(`${env.config.feedRoot}Network/providers`, {\n params: { networkId, onlyTicketing },\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n throw new Error(response.errorCode);\n } else {\n return response;\n }\n })\n )\n );\n }\n\n public updateNetworkProviders(networkId: number, providerIds: number[]) {\n return firstValueFrom(\n this.http\n .put>(\n `${env.config.feedRoot}Network/updateProviders`,\n {\n networkId,\n providerIds,\n }\n )\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n throw new Error(response.errorCode);\n } else {\n return response;\n }\n })\n )\n );\n }\n\n public getNetworksPaymentOptions(network_id: number): Observable {\n let version = this.cache.getVersion();\n return this.http\n .get>(\n env.config.feedRoot + 'Payment/getPaymentOptions',\n {\n params: { network_id, v: version },\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n );\n }\n\n //THIS IS WAR : Flux delete field configuration if empty value... Need to send them ALL :(\n public updateNetworksPayzenOptions(\n network_id: number,\n options: {\n saveCards: boolean;\n payment_company: string;\n paymentMethods: string[];\n payzen: PayzenOptions;\n }\n ): Observable {\n return this.http\n .post(env.config.feedRoot + `Payment/setPaymentOptions`, {\n network_id,\n options: {\n saveCards: options.saveCards ? '1' : '0',\n payment_company: options.payment_company,\n paymentMethods: options.paymentMethods,\n payzen: {\n shop: options.payzen.shop_id,\n descriptor: options.payzen.descriptor,\n transferEnabled: options.payzen.transfer_enabled,\n transferDayOfMonth: options.payzen.transfer_day_of_month,\n subscriptionEnabled: options.payzen.subscription_enabled,\n subscriptionAsMultiplePayment: options.payzen.subscription_as_multiple_payment,\n },\n },\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n if (response.errorCode !== null) throw new Error(response.errorCode);\n } else {\n this.cache.incrementVersion();\n return response;\n }\n })\n );\n }\n\n public updateExportDatasets(\n networkId: number,\n datasetsKeys: string[]\n ): Promise<{ success: boolean }> {\n return firstValueFrom(\n this.http\n .put>(\n `${env.config.feedRoot}Network/updateExportDatasets`,\n {\n networkId,\n datasetHandles: JSON.stringify(datasetsKeys),\n }\n )\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) {\n throw new Error(response.errorCode);\n } else {\n return response;\n }\n })\n )\n );\n }\n\n public getNetworkFeatures(networkId: number): Promise {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Network/getNetworkFeatures`,\n {\n params: {\n networkId,\n },\n }\n )\n .pipe(map(({ response }) => response.features))\n );\n }\n\n public activateFeature(networkId: number, featureHandle: NetworkFeatureHandle): Promise {\n return firstValueFrom(\n this.http\n .post>(\n `${env.config.feedRoot}Network/activateFeature`,\n {\n networkId,\n featureHandle,\n }\n )\n .pipe(map(({ response }) => response.activatedFeature))\n );\n }\n\n public deactivateFeature(\n networkId: number,\n featureHandle: NetworkFeatureHandle\n ): Promise {\n return firstValueFrom(\n this.http\n .delete>(\n `${env.config.feedRoot}Network/deactivateFeature`,\n {\n params: {\n networkId,\n featureHandle,\n },\n }\n )\n .pipe(map(({ response }) => response.success))\n );\n }\n\n public getInspectionConfiguration(networkId: number): Promise {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Network/getInspectionConfiguration`,\n {\n params: {\n networkId,\n },\n }\n )\n .pipe(map(({ response }) => response.settings))\n );\n }\n\n public editInspectionConfiguration(\n networkId: number,\n settings: NetworkInspectionConfiguration\n ): Promise {\n return firstValueFrom(\n this.http\n .patch>(\n `${env.config.feedRoot}Network/editInspectionConfiguration`,\n {\n networkId,\n settings,\n }\n )\n .pipe(map(({ response }) => response.settings))\n );\n }\n}\n","import { Address } from '@app/modules/shared/models/address';\nimport { Filters } from '@app/modules/shared/models/filters';\nimport { Attribute, Customer } from '../../customers/models/customer';\nimport { Item } from './item';\n\nexport interface Order {\n id: string;\n order_identifier: string;\n network_id: string;\n network_logo: string;\n status: OrderStatus;\n process_status: string;\n total_base: string;\n total_vat: string;\n date: string;\n email_sent: string;\n sent_at: string;\n user: Customer;\n items: Item[];\n vats: Array<{ vat: string; price: number }>;\n installments: string;\n inst_schedule?: any[];\n shipping_mode: string;\n shipping_address?: Address;\n origin_type?: string;\n origin_agent?: string;\n hasVoucher?: boolean;\n voucher?: string;\n universal?: boolean;\n payment_provider?: string;\n mandate_id?: string;\n issuer?: 'airweb' | 'sncf';\n item_subscription_count?: string;\n client: {\n id: string;\n name: string;\n slug: string;\n } | null;\n discounts: OrderDiscount[];\n}\n\nexport interface SchoolOrder {\n id: number;\n code: string;\n date: string;\n status: OrderStatus;\n processStatus: OrderProcessStatus | null;\n taxFreeAmount: string;\n totalAmount: string;\n installments: string;\n paymentMethod: OrderPaymentMethod;\n paymentProvider: string;\n voucher: string;\n networkId: number;\n originType: OrderOriginType;\n customer?: SchoolOrderCustomer | null;\n articles: SchoolOrderArticle[];\n shippingAddress?: OrderShippingAddress | null;\n discounts: OrderDiscount[];\n checkouts: Checkout[];\n}\n\nexport interface Checkout {\n executionStatus: string;\n executionDuration: number;\n initiatorType: string;\n initiatorAgent: string;\n createdAt: string;\n}\n\nexport interface SchoolOrderArticle {\n id: number;\n unitPrice: string;\n taxFreeUnitPrice: string;\n submissionId?: number;\n customerLastSubmissionId: number;\n customerLastSubmissionStatus: string;\n product: {\n id: number;\n name: string;\n toProcess: boolean;\n };\n customer?: SchoolOrderCustomer | null;\n fields?: SchoolComplementaryField[];\n trip?: Trip | null;\n reservation?: ArticleBooking | null;\n}\n\nexport interface Trip {\n origin: {\n label: string;\n url: string;\n };\n destination: {\n label: string;\n url: string;\n };\n via?: {\n label: string;\n url: string;\n };\n}\n\nexport interface SchoolComplementaryField {\n fieldId: number;\n valueId?: any;\n valueLabel?: any;\n value: string;\n}\n\nexport enum OrderStatus {\n completed = 'COMPLETED',\n pending = 'PENDING',\n canceled = 'CANCELED',\n error = 'ERROR',\n regularization = 'REGULARIZATION',\n preauthorized = 'PREAUTHORIZED',\n waiting_for_payment = 'WAITING_FOR_PAYMENT',\n to_refund = 'TO_REFUND',\n refunded = 'REFUNDED',\n defaultPayment = 'PAYMENT_ISSUE',\n processing = 'PROCESSING',\n}\n\nexport enum OrderProcessStatus {\n processed = 'PROCESSED',\n processing = 'PROCESSING',\n to_process = 'TO_PROCESS',\n autoprocessed = 'AUTOPROCESSED',\n incomplete = 'INCOMPLETE',\n}\n\nexport enum OrderPaymentMethod {\n CARD = 'CARD',\n TRANSFER = 'TRANSFER',\n DEPOSIT = 'DEPOSIT',\n CASH = 'CASH',\n}\n\nexport enum OrderOriginType {\n APP = 'APP',\n ESHOP = 'ESHOP',\n SCHOOL = 'SCHOOL',\n RETAIL = 'RETAIL',\n}\n\nexport interface OrderFilters extends Filters {\n status?: string;\n processingStatus?: string;\n supports?: string;\n installments?: number;\n total?: number;\n provider?: string;\n paymentMethod?: string;\n waiting?: boolean;\n cat_id?: number;\n originType?: string;\n attribute?: number;\n attributeSearch?: string;\n productId?: string;\n}\n\nexport interface SchoolOrderCustomer {\n id: number;\n firstname: string | null;\n lastname: string | null;\n birthdate?: string | null;\n email?: string | null;\n phone?: string | null;\n avatar?: string | null;\n address?: Address | null;\n attributes?: Attribute[] | null;\n}\n\nexport interface OrderShippingAddress {\n streetNumber: string;\n route: string;\n city: string;\n zipCode: string;\n country: string;\n}\n\nexport interface OrderShippingAddress {\n streetNumber: string;\n route: string;\n city: string;\n zipCode: string;\n country: string;\n}\n\nexport interface OrderDiscount {\n id: number;\n sourceType: string | null;\n sourceCode: string | null;\n type: OrderDiscountType;\n value: string | null;\n}\n\nexport enum OrderDiscountType {\n AMOUNT = 'AMOUNT',\n RATIO = 'RATIO',\n ARTICLE_AMOUNT = 'ARTICLE_AMOUNT',\n ARTICLE_RATIO = 'ARTICLE_RATIO',\n FREE = 'FREE',\n GIFT = 'GIFT',\n}\n\nexport interface ArticleBooking {\n type: BookingTrip;\n message: string | null;\n tripSeatCount: string;\n tripDepartureDate: string;\n tripDepartureLocationLabel: string;\n tripDepartureLocationRef: string;\n tripArrivalDate: string;\n tripArrivalLocationLabel: string;\n tripArrivalLocationRef: string;\n}\n\nexport enum BookingTrip {\n TRIP = 'TRIP',\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { environment as env } from '@env/environment';\nimport { NotificationsService } from 'angular2-notifications';\nimport { firstValueFrom, Observable, of } from 'rxjs';\nimport { map, switchMap } from 'rxjs/operators';\nimport { Comment } from '../models/comment';\nimport { Order, OrderProcessStatus, OrderStatus, SchoolOrder } from '../models/order';\n\n@Injectable({\n providedIn: 'root',\n})\nexport class OrdersService {\n public order: Order;\n\n constructor(\n private http: HttpClient,\n private authService: AuthService,\n private router: Router,\n private filters: FilterService,\n private notification: NotificationsService\n ) {}\n\n public getOrders(cid: () => number = () => null, otherFilter: any = null): Observable {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Order/getOrder.json?cat_id=` + cid(), {\n params: otherFilter !== null ? otherFilter : this.filters.filtersWithID,\n })\n ),\n\n map(({ response: r }) => {\n return r.orders as Order[];\n })\n );\n }\n\n /**\n * @deprecated - This function should not be used anymore. Use `getOrderDetails` instead.\n */\n public getLegacyOrderDetails(orderId: string | number): Promise {\n const legacyOrder$ = this.http\n .get>(`${env.config.feedRoot}Order/getOrder`, {\n params: { id: orderId },\n })\n .pipe(\n map(({ response }) => {\n if (Array.isArray(response.orders) && response.orders.length > 0) {\n return response.orders[0];\n }\n\n return null;\n })\n );\n\n return firstValueFrom(legacyOrder$);\n }\n\n public getOrderDetails(orderId: string | number): Promise {\n const order$ = this.http\n .get>(`${env.config.feedRoot}Order/details`, {\n params: { id: orderId },\n })\n .pipe(\n map(({ response }) => {\n return response as SchoolOrder;\n })\n );\n\n return firstValueFrom(order$);\n }\n\n public getOrderInvoice(id: string, options: { legacy: boolean; version?: string }) {\n let endpoint = options.legacy ? 'InvoicePrinter/getInvoice' : 'Order/printInvoice';\n\n let responseType = 'blob';\n\n if (options?.version) {\n if (options.version === 'V1') {\n responseType = 'html';\n endpoint = 'invoicePrinter/getInvoice';\n }\n\n if (options.version === 'V2') {\n endpoint = 'Order/printInvoice';\n }\n\n if (options.version === 'V3') {\n endpoint = 'Order/printInvoiceV3';\n }\n }\n\n return this.http\n .post(\n env.config.feedRoot + endpoint,\n {\n orderId: id,\n },\n {\n responseType: responseType as 'json',\n }\n )\n .pipe(\n map(async (response) => {\n if (options?.version === 'V1') {\n return response;\n }\n\n return new Blob([response], { type: 'application/pdf' });\n })\n );\n }\n\n public updateArticleFieldValue(\n orderId: string,\n orderArticleId: string,\n orderItemId: string,\n fieldId: string,\n value: unknown\n ) {\n const body = {\n orderId,\n orderArticleId,\n orderItemId,\n fieldId,\n value,\n };\n\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Order/updateArticleFieldValue.json`, body)\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public updateStatus(order_id: number | string, status: string, isPayment = false) {\n return this.http\n .post(env.config.feedRoot + 'Order/updateOrderStatus.json', {\n order_id,\n status,\n isPayment,\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) console.log({ response });\n\n return response;\n })\n );\n }\n\n public updateProcessStatus(order_id: number, process_status: string) {\n return this.http\n .post(env.config.feedRoot + 'Order/updateOrderProcessStatus.json', {\n order_id,\n process_status,\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) console.log({ response });\n\n return response;\n })\n );\n }\n\n public getOrderComments(id: number, version: number) {\n return this.http\n .get(env.config.feedRoot + `Order/getOrderComments.json?order_id=${id}&v=${version}`)\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) console.log({ response });\n\n return response.comments as Comment[];\n })\n );\n }\n\n public postOrderComment(order_id: number, comment: string) {\n return this.http\n .post(env.config.feedRoot + 'Order/postOrderComment.json', {\n order_id,\n comment,\n })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) console.log({ response });\n\n return response;\n })\n );\n }\n\n public editOrderComment(id: number, comment: string) {\n return this.http\n .post(env.config.feedRoot + 'Order/editOrderComment.json', { id, comment })\n .pipe(\n map(({ response }) => {\n if (response.errorMessage) console.log({ response });\n\n return response;\n })\n );\n }\n\n public deleteOrderComment(id: number) {\n return this.http.post(env.config.feedRoot + 'Order/deleteOrderComment.json', { id }).pipe(\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public updateNbOrderToProcess(): Promise {\n return firstValueFrom(\n this.http\n .get>(`${env.config.feedRoot}Order/count`, {\n params: {\n max: 11,\n status: OrderStatus.completed,\n processingStatus: `${OrderProcessStatus.to_process},${OrderProcessStatus.processing}`,\n },\n })\n .pipe(\n map((response) => {\n const toProcess = response?.response?.count || 0;\n return toProcess;\n })\n )\n );\n }\n\n /**\n * Start the checkout process of a given order manually\n * @param orderId - The order id to checkout\n * @param force - This force the checkout even if the order is already checked out. Throw otherwise.\n */\n public checkoutOrder(orderId: number | string, force: boolean = true): Promise {\n const response$ = this.http\n .post>(`${env.config.feedRoot}/order/checkout`, {\n orderId,\n force,\n })\n .pipe(\n map(({ response }) => {\n return response.success;\n })\n );\n\n return firstValueFrom(response$);\n }\n}\n","import { Component, Input } from '@angular/core';\n\ntype Status = 'default' | 'info' | 'warning' | 'success' | 'error';\n\n@Component({\n selector: 'tu-card',\n styleUrls: ['./card.component.scss'],\n template: `\n \n \n \n \n \n\n
\n \n
\n\n
\n \n
\n \n `,\n})\nexport class CardComponent {\n @Input('icon')\n public icon?: string;\n\n @Input('withHeader')\n public withHeader = false;\n\n @Input('withFooter')\n public withFooter = false;\n\n @Input('klass')\n public klass = '';\n\n @Input('status')\n public status: Status = 'default';\n}\n","const COLORS = {\n red: 'tw-text-red-500',\n rouge: 'tw-text-red-500',\n blue: 'tw-text-blue-500',\n bleu: 'tw-text-blue-500',\n green: 'tw-text-green-500',\n vert: 'tw-text-green-500',\n orange: 'tw-text-orange-500',\n yellow: 'tw-text-yellow-500',\n jaune: 'tw-text-yellow-500',\n violet: 'tw-text-violet-500',\n pink: 'tw-text-pink-500',\n rose: 'tw-text-rose-500',\n};\n\n// https://marked.js.org/using_pro#extensions\nexport const fontColor = {\n name: 'fontColor',\n level: 'inline', // Is this a block-level or inline-level tokenizer?\n start(src) {\n return src.match(/\\[([a-z]+)\\]/)?.index;\n }, // Hint to Marked.js to stop and check for a match\n tokenizer(src) {\n const rule = /^\\[([a-z]+)\\]((?:.|\\n)+)\\[\\/\\1\\]/; // Regex for the complete token\n\n const match = rule.exec(src);\n\n if (!match) return;\n const color = match[1].trim();\n\n if (!Object.keys(COLORS).includes(color)) return;\n\n const token = {\n // Token to generate\n type: 'fontColor', // Should match \"name\" above\n raw: match[0], // Text to consume from the source\n text: this.lexer.inlineTokens(match[2].replace(/^ +| +$/gm, '')), // Removes the leading and trailing white spaces\n color,\n };\n\n return token;\n },\n\n renderer(token) {\n return `${this.parser.parseInline(\n token.text\n )}`;\n },\n};\n","import { Component } from '@angular/core';\nimport { MatDialogRef } from '@angular/material/dialog';\n\n@Component({\n selector: 'tu-delete-modal',\n templateUrl: './delete-modal.component.html',\n styleUrls: ['./delete-modal.component.scss'],\n})\nexport class DeleteModalComponent {\n constructor(private modalRef: MatDialogRef) {}\n\n public closeModal(success: boolean): void {\n this.modalRef.close({ success });\n }\n}\n","
\n

{{ 'pages.shared.confirm_comment_delete' | translate }}

\n
\n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n {{ 'otherslabels.btn_delete' | translate }}\n \n
\n
\n","import { Component } from '@angular/core';\n\n@Component({\n selector: 'tu-comments-ng-format',\n templateUrl: './format.component.html',\n styleUrls: ['./format.component.scss'],\n})\nexport class FormatComponent {}\n","
\n

{{ 'pages.shared.format_direction' | translate }}

\n

{{ 'pages.shared.markdown_use' | translate }}

\n\n

{{ 'pages.shared.reminders' | translate }} :

\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
{{ 'pages.shared.reminders_list' | translate }}
\n {{ 'pages.shared.reminder_style' | translate }}\n \n {{ 'pages.shared.reminder_syntax' | translate }}\n \n {{ 'pages.shared.reminder_exemple' | translate }}\n \n {{ 'pages.shared.reminder_result' | translate }}\n
\n {{ 'pages.shared.color' | translate }}\n \n {{ 'pages.shared.color_hint' | translate }}\n
\n {{ 'pages.shared.color_options' | translate }}\n {{ 'pages.shared.color_options_label' | translate }}\n
\n {{ 'pages.shared.color_hint_red' | translate }}\n \n {{ 'pages.shared.color_red' | translate }}\n
\n {{ 'pages.shared.bold' | translate }}\n ** ** ou __ __\n {{ 'pages.shared.bold_exemple' | translate }}\n \n {{ 'pages.shared.bold_text' | translate }}\n
\n {{ 'pages.shared.italic' | translate }}\n * * ou _ _\n {{ 'pages.shared.italic_exemple' | translate }}\n \n {{ 'pages.shared.bold_text' | translate }}\n
\n {{ 'pages.shared.strikethrough' | translate }}\n ~~ ~~\n {{ 'pages.shared.strikethrough_exemple' | translate }}\n \n {{ 'pages.shared.strikethrough_text' | translate }}\n
\n {{ 'pages.shared.link' | translate }}\n \n {{ 'pages.shared.link_exemple' | translate }}\n \n {{ 'pages.shared.link_test' | translate }}\n \n {{ 'pages.shared.link_test' | translate }}\n
\n {{ 'pages.shared.title' | translate }}\n \n {{ 'pages.shared.title_hierarchy' | translate }}\n \n # {{ 'pages.shared.title_main' | translate }}\n
\n ### {{ 'pages.shared.title_minor' | translate }}\n
\n

{{ 'pages.shared.title_main' | translate }}

\n

{{ 'pages.shared.title_minor' | translate }}

\n
\n
\n","
\n
\n \n
\n
\n \n
\n {{ 'pages.shared.comments' | translate }}\n 0\" class=\"tw-py-1 tw-px-3 tw-bg-red-500 tw-text-white tw-rounded-full\">{{ comments.length }}\n
\n \n \n \n \n
\n \n
\n \n \n
\n \n \n \n \n \n \n \n
\n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n \n \n {{ (commentForm.controls.id.value ? 'Editer le commentaire' : 'pages.order_details.comment') | translate }}\n \n \n
\n \n \n
\n \n
\n \n
    0\" class=\"tw-flex tw-flex-col tw-gap-2 tw-px-4 tw-pb-4 tw-grow tw-overflow-y-auto tw-min-h-28\">\n
  • \n \n
  • \n
\n \n
\n

{{ 'pages.order_details.no_comment' | translate }}

\n
\n
\n
\n \n
\n
\n","import {\n Component,\n EventEmitter,\n Input,\n OnInit,\n Output,\n TemplateRef,\n ViewEncapsulation,\n} from '@angular/core';\nimport { UntypedFormBuilder } from '@angular/forms';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { marked } from 'marked';\nimport DOMPurify from 'dompurify';\nimport { fontColor } from './marked-colors';\nimport { CommentFormValues } from '../../models/comment';\nimport { NotificationsService } from 'angular2-notifications';\nimport { TranslateService } from '@ngx-translate/core';\nimport { MatDialog } from '@angular/material/dialog';\nimport { DeleteModalComponent } from './delete-modal/delete-modal.component';\nimport { Network } from '../../models/network';\n\ntype ComputeCommentResponse = Promise<{ success: boolean; errorMessage?: string }>;\n\n@Component({\n selector: 'tu-comments-ng',\n templateUrl: './comments-ng.component.html',\n styleUrls: ['./comments-ng.component.scss'],\n encapsulation: ViewEncapsulation.None,\n})\nexport class CommentsNgComponent implements OnInit {\n public heroicons = heroicons;\n\n public commentForm = this.fb.group({\n id: null,\n message: '',\n networkId: null,\n });\n\n public showFormat: boolean = false;\n public submitLoading: boolean = false;\n public isLoading: boolean = false;\n\n @Input()\n public comments: Array | null = null;\n\n @Input()\n public commentTemplate: TemplateRef;\n\n @Input()\n public postComment: (comment: CommentFormValues) => ComputeCommentResponse;\n\n @Input()\n public editComment: (comment: CommentFormValues) => ComputeCommentResponse;\n\n @Input()\n public deleteComment: (id: number) => ComputeCommentResponse;\n\n @Input() public requiresNetwork = false;\n\n @Input() public networks: Network[] = [];\n\n @Output() public selectedNetwork = new EventEmitter();\n\n constructor(\n private fb: UntypedFormBuilder,\n private notification: NotificationsService,\n private translate: TranslateService,\n private modal: MatDialog\n ) {}\n\n get hasMessage(): boolean {\n return (\n Boolean(this.commentForm.controls.message.value) &&\n // No only space\n !/^\\s+$/.test(this.commentForm.controls.message.value)\n );\n }\n\n get sortedComments(): Array {\n if (this.comments) {\n return this.comments;\n }\n\n return [];\n }\n\n public ngOnInit(): void {\n this.commentForm.controls.networkId.valueChanges.subscribe((networkId) => {\n this.selectedNetwork.emit(networkId);\n this.commentForm.controls.id.reset();\n });\n\n if (this.requiresNetwork && this.networks.length === 1) {\n this.commentForm.controls.networkId.setValue(this.networks[0].id);\n }\n }\n\n public toggleFormat(): void {\n this.showFormat = !this.showFormat;\n }\n\n public async submitComment(): Promise {\n try {\n this.isLoading = true;\n this.submitLoading = true;\n\n const body = this.commentForm.value;\n\n const response = await (body.id ? this.editComment(body) : this.postComment(body));\n\n if (!response.success) {\n throw new Error(response.errorMessage);\n }\n\n this.cancelEdition();\n } catch {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n } finally {\n this.isLoading = false;\n this.submitLoading = false;\n }\n }\n\n public cancelEdition(): void {\n this.commentForm.controls.message.reset();\n this.commentForm.controls.id.reset();\n }\n\n public setEditableComment = (id: number, message: string) => {\n this.commentForm.controls.id.setValue(id);\n this.commentForm.controls.message.setValue(message);\n };\n\n public deleteCommentModal = (id: number) => {\n const modalRef = this.modal.open(DeleteModalComponent);\n\n modalRef.afterClosed().subscribe(async (response) => {\n if (response?.success === true) {\n try {\n this.isLoading = true;\n await this.deleteComment(id);\n } catch {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n } finally {\n this.isLoading = false;\n }\n }\n });\n };\n\n public markdownToHTML(markdown: string): string {\n marked.setOptions({ breaks: true });\n marked.use({ extensions: [fontColor] });\n\n //https://marked.js.org/\n const dirtyHTML = marked.parse(markdown);\n\n //https://github.com/cure53/DOMPurify\n return DOMPurify.sanitize(dirtyHTML);\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\n\n@Pipe({ name: 'interval' })\nexport class IntervalPipe implements PipeTransform {\n static sqlDateRegexp = /\\d{4}-\\d{2}-\\d{2} \\d{2}:\\d{2}:\\d{2}/;\n\n public transform(data: T[], from?: string | Date, to?: string | Date, key?: string): T[] {\n if ([data, from, to].some(this.isFalsy)) return data;\n\n const fromDate = new Date(from);\n const toDate = new Date(to);\n\n return data.filter((val) => {\n const value = key ? key.split('.').reduce((v, k) => v[k], val) : key['date'];\n const isSqlDate = IntervalPipe.sqlDateRegexp.test(value);\n const date = isSqlDate ? value.replace(/ /g, 'T') : value;\n const valueDate = new Date(date);\n\n return valueDate >= fromDate && valueDate <= toDate;\n });\n }\n\n private isFalsy(value: unknown) {\n return !Boolean(value);\n }\n}\n","\n
\n {{ 'pages.order.filters' | translate }}\n\n
\n \n \n
\n\n 1\"\n class=\"input-group tw-flex\"\n >\n \n\n \n\n \n \n
{{ name }}
\n \n
\n\n
\n \n \n \n
\n \n
\n\n \n {{ 'pages.shared.dates_filter' | translate }}\n\n \n
\n {{ 'pages.order.from' | translate }}\n \n
\n\n
\n {{ 'pages.order.to' | translate }}\n \n
\n \n\n
\n \n \n
\n\n \n\n \n {{ 'pages.order.export' | translate }}:\n \n \n \n Excel\n \n\n \n CSV\n \n \n\n \n \n {{ format }}\n \n \n \n\n \n \n\n \n \n\n","import { DatePipe } from '@angular/common';\nimport { Component, EventEmitter, Input, OnDestroy, OnInit, Output } from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl, UntypedFormGroup } from '@angular/forms';\nimport { environment as env } from '@env/environment';\nimport { formatISO, parse, set } from 'date-fns';\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { debounceTime, distinctUntilChanged, map, startWith, switchMap, tap } from 'rxjs/operators';\nimport { IntervalPipe } from '../../pipes/interval.pipe';\nimport { KeyPipe } from '../../pipes/key.pipe';\nimport { SearchPipe } from '../../pipes/search.pipe';\nimport { AuthService } from '../../services/auth.service';\nimport { FilterService } from '../../services/filter.service';\nimport { QueryParamsService } from '../../services/queryParams.service';\nimport { observeProperty } from '../../utils/observeProperty';\nimport { getDefaultFrom, getDefaultTo } from '../../utils/filter';\nimport { Network } from '../../models/network';\n\n@Component({\n selector: 'tu-filter',\n templateUrl: './filter.component.html',\n styleUrls: ['./filter.component.scss'],\n providers: [SearchPipe, IntervalPipe, DatePipe, KeyPipe],\n})\nexport class FilterComponent implements OnInit, OnDestroy {\n constructor(\n private searchPipe: SearchPipe,\n private intervalPipe: IntervalPipe,\n private datePipe: DatePipe,\n private keyPipe: KeyPipe,\n private fb: UntypedFormBuilder,\n public filterService: FilterService,\n private queryParamsService: QueryParamsService,\n public authService: AuthService\n ) {}\n\n public static sub: Subscription;\n public exportLinkHref = '';\n\n private _filtered: BehaviorSubject = new BehaviorSubject(null);\n private _content: Object[];\n public loading = false;\n\n public filterForm: UntypedFormGroup = null;\n public networksName: string[] = [];\n public networkNameCtrl = new UntypedFormControl();\n public filteredNetworksName: Observable = this.networkNameCtrl.valueChanges.pipe(\n startWith(null),\n map((name: string | null) => (name ? this._filter(name) : this.networksName.slice()))\n );\n\n @Input() defaultFilters: Filters;\n @Input() content: Observable | Object[];\n @Input() config: FiltersConfig;\n @Input() exportInformation: ExportInformation;\n @Input() isLoading = false;\n @Input() networks: Network[] | null = null;\n public isLoading$ = observeProperty(this, 'isLoading');\n\n @Output() filtered: EventEmitter> = new EventEmitter<\n BehaviorSubject\n >();\n @Output() filter: EventEmitter> = new EventEmitter>();\n @Output() exportDatasEvent: EventEmitter = new EventEmitter();\n @Output() filtersChanged = new EventEmitter>();\n @Output() filtersReset = new EventEmitter();\n\n private filtersSub: Subject = new Subject();\n private filtersBoSub: Subject = new Subject();\n\n get isExportEnabled() {\n if (!this.exportInformation) return false;\n\n const hasCheckedRows = this.exportInformation.checkedRowsIds.length > 0;\n const hasAllCheckedRows = this.exportInformation.hasCheckedAll;\n\n return hasCheckedRows || hasAllCheckedRows;\n }\n\n get exportFormats() {\n if (!this.exportInformation) return this.config?.export_labels ?? undefined;\n if (!this.config || !this.config?.export_labels) return [];\n\n // If export information exist, we filter 'CSV' from export formats\n if (typeof this.config.export_labels === 'string') {\n if (this.config.export_labels === 'CSV') {\n return [];\n }\n return this.config.export_labels;\n }\n\n return this.config.export_labels.filter((label) => label !== 'CSV');\n }\n\n ngOnDestroy() {\n if (FilterComponent.sub) FilterComponent.sub.unsubscribe();\n\n this.isLoading$.subscribe((isLoading) => {\n if (isLoading) {\n this.filterForm.disable();\n } else {\n this.filterForm.enable();\n }\n });\n }\n\n private getNetworkByName(name: string) {\n return this.authService.networks.find((network) => network.name === name);\n }\n\n async ngOnInit() {\n this.initFilterService();\n\n const filters = this.filterService.filters;\n if (this.networks === null) {\n this.authService.networks.forEach((n) => this.networksName.push(n.name));\n } else {\n this.networks.forEach((n) => this.networksName.push(n.name));\n }\n\n const network = this.authService.getNetwork(filters.network);\n\n this.filterForm = this.fb.group({\n query: filters.query,\n from: filters.from ? this.formatDate(new Date(filters.from)) : null,\n to: filters.to ? this.formatDate(new Date(filters.to)) : null,\n network: filters.network,\n onlyActive: filters.onlyActive,\n });\n\n this.filter.emit(this.filtersSub);\n this.filtersSub.next(this.filterForm.value);\n\n this.networkNameCtrl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((name) => {\n if (!name) this.filterForm.controls.network.setValue(null);\n\n const network = this.getNetworkByName(name);\n if (network) this.filterForm.controls.network.setValue(network.id);\n\n this.filterService.networkName = name;\n });\n\n if (network) {\n this.networkNameCtrl.setValue(network.name);\n }\n\n if (!this.content) return;\n\n if (this.content instanceof Observable) {\n FilterComponent.sub = this.filtersBoSub\n .pipe(\n startWith(this.filterForm.value),\n tap(() => (this.loading = true)),\n switchMap(() => this.content as Observable)\n )\n .subscribe((c) => {\n this._content = c;\n this._filtered.next(this.filterContent(this._content));\n this.loading = false;\n });\n } else {\n this._content = this.content;\n this._filtered.next(this.filterContent(this._content));\n }\n\n this.filtered.emit(this._filtered);\n }\n\n public onClickApplyFilters() {\n //trick to add proper timezone to date\n const to = this.filterForm.value.to?.replaceAll('-', '/') ?? null;\n const from = this.filterForm.value.from?.replaceAll('-', '/') ?? null;\n const onlyActive = this.filterForm.value.onlyActive || null;\n\n this.filterService.filters = { ...this.filterForm.value, to, from, onlyActive };\n if (this.config && this.config.boFilter !== true)\n this._filtered.next(this.filterContent(this._content));\n else this.filtersBoSub.next(this.filterForm.value);\n this.filtersSub.next(this.filterForm.value);\n this.filtersChanged.emit(this.filterForm.value);\n }\n\n public onClickResetFilters() {\n //reset from & to date to current day\n const from = getDefaultFrom();\n const to = getDefaultTo();\n\n this.filterForm.setValue({\n query: null,\n from: this.defaultFilters?.from\n ? this.formatDate(new Date(this.defaultFilters.from))\n : this.formatDate(from),\n to: this.defaultFilters?.to\n ? this.formatDate(new Date(this.defaultFilters.to))\n : this.formatDate(to),\n network: null,\n onlyActive: null,\n });\n\n this.networkNameCtrl.reset();\n this.filtersReset.emit();\n this.onClickApplyFilters();\n }\n\n private initFilterService() {\n const filters = this.filterService.filters;\n\n //Init filters depending on config & previous filter state\n if (this.config?.dates === false) {\n filters.from = null;\n filters.to = null;\n } else {\n //init date if allowed but null\n filters.from = this.defaultFilters?.from || filters.from || new Date().toISOString();\n filters.to = this.defaultFilters?.to || filters.to || new Date().toISOString();\n }\n\n if (this.config?.network === false) {\n filters.network = null;\n filters.networkName = null;\n }\n if (this.config?.query === false) {\n filters.query = null;\n }\n filters.onlyActive = this.defaultFilters?.onlyActive || null;\n //force filters removed by config\n\n const isSameObject = (obj1: Object, obj2: Object) => {\n return Object.entries(obj1).every(([key, value]) => obj2[key] === value);\n };\n\n if (!isSameObject(filters, this.filterService.filters)) {\n this.filterService.filters = filters;\n }\n this.queryParamsService.globalFilterParams = filters;\n }\n\n private filterContent(content) {\n if (this.config && this.config.boFilter !== true) {\n content = this.keyPipe.transform(content, 'network_id', this.filterService.network); // If single id passed\n content = this.keyPipe.transform(content, 'network_ids', this.filterService.network); // If id array passed\n if (!this.config || this.config.dates !== false) {\n const key = this.config && this.config.date_key ? this.config.date_key : null;\n content = this.intervalPipe.transform(\n content,\n this.filterService.from,\n this.filterService.to,\n key\n );\n }\n }\n\n return this.searchPipe.transform(content, this.filterForm.controls['query'].value);\n }\n\n private formatDate(date) {\n return this.datePipe.transform(date, 'yyyy-MM-dd');\n }\n\n public exportFile(event, type): void {\n this.exportDatasEvent.emit(type);\n }\n\n private _filter(value: string): string[] {\n const filterValue = value.toLowerCase();\n\n return this.networksName.filter((name) => name.toLowerCase().includes(filterValue));\n }\n\n public computeLinkHref(format: string): void {\n try {\n // STEP 1: Create export URL base with export dataset ID and export format (for now, CSV)\n const { exportId } = this.exportInformation;\n\n const exportUrl = new URL(`${env.config.exportRoot}datasets/${exportId}/export`);\n\n // STEP 2: Create searchParams with selected filters\n const timezone = Intl.DateTimeFormat().resolvedOptions().timeZone;\n\n // The '@' character is a convention established in order to identify operator tokens\n const token = `@${this.authService.token}`;\n\n const filters: Record = {\n format: format || 'csv',\n timezone,\n token,\n ...this.exportInformation.filters,\n };\n\n const { from, to } = this.filterForm.value;\n\n if (from && to) {\n const from = parse(this.filterForm.value.from, 'yyyy-MM-dd', new Date());\n const to = parse(this.filterForm.value.to, 'yyyy-MM-dd', new Date());\n\n const start = set(from, { hours: 0 });\n const end = set(to, { hours: 23, minutes: 59, seconds: 59 });\n\n filters.from = formatISO(start);\n filters.to = formatISO(end);\n }\n\n const searchParams = new URLSearchParams(filters);\n\n // STEP 3-a: If all rows have been selected, add general optional filters\n if (this.filterService.network) {\n searchParams.set('networkId', this.filterService.network.toString());\n } else {\n const userNetworkIdList = this.authService.networks.reduce((networksList, { id }) => {\n if (!networksList) return id;\n\n return networksList + ',' + id;\n }, '');\n\n searchParams.set('networkId', userNetworkIdList);\n }\n\n // STEP 3-b: Check if the user has selected some rows, but not all rows, then export data\n if (!this.exportInformation.hasCheckedAll && this.exportInformation.checkedRowsIds.length) {\n const idsString = this.exportInformation.checkedRowsIds.join(',');\n\n searchParams.set('ids', idsString);\n exportUrl.search = searchParams.toString();\n\n this.exportLinkHref = exportUrl.toString();\n }\n\n if (this.filterForm.controls['query'].value) {\n searchParams.set('query', this.filterForm.controls['query'].value);\n }\n\n // STEP 4-b: Add filters to export URL and export data\n exportUrl.search = searchParams.toString();\n\n this.exportLinkHref = exportUrl.toString();\n } catch (error) {\n console.log(error);\n this.exportLinkHref = '';\n }\n }\n}\n\nexport interface Filters {\n query?: string;\n from?: string;\n to?: string;\n network?: number;\n onlyActive?: boolean;\n}\n\nexport interface FiltersConfig {\n query?: boolean;\n override_style?: boolean;\n dates?: boolean;\n network?: boolean;\n export?: boolean;\n export_labels?: string | string[];\n date_key?: string;\n boFilter?: boolean;\n onlyActive?: boolean;\n}\n\nexport interface ExportInformation {\n checkedRowsIds: number[];\n hasCheckedAll: boolean;\n exportId: string;\n filters: {};\n}\n","import { Component, Input } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\n@Component({\n selector: 'tu-hero-icon',\n template: `\n \n `,\n})\nexport class HeroIconComponent {\n constructor(private sanitizer: DomSanitizer) { }\n\n public content: { outline: boolean; html: SafeHtml } | null = null;\n\n @Input('klass') klass = '';\n @Input('title') title = '';\n\n @Input('path') set path(content: { outline: boolean; path: string }) {\n if (!content?.path) return;\n\n const html = this.sanitizer.bypassSecurityTrustHtml(content.path);\n\n this.content = {\n outline: content.outline,\n html,\n };\n }\n}\n","import {\n AfterViewInit,\n Component,\n ElementRef,\n EventEmitter,\n Input,\n Output,\n ViewChild,\n} from '@angular/core';\nimport Cropper from 'cropperjs';\n\n@Component({\n selector: 'tu-image-cropper',\n template: `\n
\n \n
\n `,\n})\nexport class ImageCropperComponent implements AfterViewInit {\n @ViewChild('image', { static: false })\n public imageElement: ElementRef;\n\n @Input('src')\n public imageUrl: string;\n\n @Output() newCroppedImageUri = new EventEmitter();\n\n //cropperjs doc : https://github.com/fengyuanchen/cropperjs/blob/main/README.md\n // angular integration guide : https://www.thepolyglotdeveloper.com/2019/06/image-cropping-zooming-scaling-angular-javascript/\n private cropper: Cropper;\n\n ngAfterViewInit() {\n this.cropper = new Cropper(this.imageElement.nativeElement, {\n zoomable: false,\n scalable: false,\n checkCrossOrigin: false,\n viewMode: 2,\n aspectRatio: 1,\n minContainerHeight: 0,\n minContainerWidth: 0,\n checkOrientation: false,\n crop: () => {\n const canvas = this.cropper.getCroppedCanvas();\n this.newCroppedImageUri.emit(canvas.toDataURL('image/png'));\n },\n });\n }\n}\n","import { Component, Input } from '@angular/core';\n\n@Component({\n selector: 'tu-loader',\n template: `\n
\n \n \n \n
\n `,\n})\nexport class LoaderComponent {\n @Input('klass')\n public klass: string = '';\n\n get iconClass(): string {\n let classList = ['tw-mx-auto', 'tw-animate-spin', 'tw-fill-current'];\n\n if (this.klass) {\n classList = classList.concat(this.klass.split(' '));\n } else {\n classList = classList.concat(['tw-h-10', 'tw-w-10']);\n }\n\n return classList.join(' ');\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\n\n@Pipe({ name: 'currency2' })\nexport class CurrencyPipe implements PipeTransform {\n transform(\n value: number,\n locale: string | undefined = 'fr-FR',\n currency: string | undefined = 'EUR'\n ) {\n return parseFloat(value.toString()).toLocaleString(locale, {\n style: 'currency',\n currency,\n });\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\nimport { DomSanitizer, SafeHtml } from '@angular/platform-browser';\n\n@Pipe({ name: 'safeHtml' })\nexport class SafeHtmlPipe implements PipeTransform {\n constructor(private sanitizer: DomSanitizer) {}\n\n transform(html: string): SafeHtml {\n return this.sanitizer.bypassSecurityTrustHtml(html);\n }\n}\n","\n
\n \n \n\n \n \n \n \n \n {{ column.label }}\n\n \n\n \n\n \n \n\n \n {{ column.label }}\n \n \n \n \n \n\n \n \n \n \n \n \n
\n \n \n \n
\n
\n\n \n \n \n {{ cell.value }}\n \n \n {{ cell.value }}\n \n \n\n \n \n \n \n \n \n\n \n \n {{\n isFunction(cell.config[cell.value]?.label)\n ? cell.config[cell.value]?.label(cell.row)\n : cell.config[cell.value]?.label\n }}\n \n \n\n \n \n \n\n \n \n \n \n \n\n \n \n\n \n \n \n \n {{ action.label }}\n \n\n \n \n \n \n \n\n \n {{ cell.value }}\n\n \n \n \n \n
\n
\n \n \n \n \n \n \n\n \n {{ cell.value }}\n \n \n \n
\n\n \n \n \n \n \n \n\n \n \n \n \n \n
Tableau générique d'affichage de données
\n \n
\n \n
\n \n \n \n \n \n {{ subColumn.label }}\n \n \n \n\n \n \n \n \n \n {{ subRow[itemTable.key] }}\n \n \n {{ subRow[itemTable.key] }}\n \n \n\n \n \n {{\n subRow[itemTable.key]\n | currency2\n : itemTable?.config?.locale()\n : itemTable?.config?.currency(row.row)\n }}\n \n \n\n \n \n {{\n subRow[itemTable.config.quantityKey] * subRow[itemTable.config.priceKey]\n | currency2\n : itemTable?.config?.locale()\n : itemTable?.config?.currency(row.row)\n }}\n \n \n\n \n {{ subRow[itemTable.key] }}\n \n \n \n \n
Tableau \"placeholder\" pour les loaders
\n
\n
\n
\n
\n\n
\n
\n \n\n \n \n \n \n \n \n \n \n {{ pagination.offset + 1 }} -\n {{\n pagination.offset + pagination.limit < pagination.count\n ? pagination.offset + pagination.limit\n : pagination.count\n }}\n sur\n {{ pagination.count }}\n \n
\n\n
    \n
  • \n \n \n \n
  • \n
  • \n \n \n \n
  • \n
  • \n = pagination.pages\"\n type=\"button\"\n (click)=\"goto(pagination.currentPage + 1)\"\n class=\"tw-text-gray-700 disabled:tw-cursor-not-allowed disabled:tw-text-gray-400\"\n >\n \n \n
  • \n
  • \n = pagination.pages\"\n type=\"button\"\n (click)=\"goto(pagination.pages)\"\n class=\"tw-text-gray-700 disabled:tw-cursor-not-allowed disabled:tw-text-gray-400\"\n >\n \n \n
  • \n
\n
\n
\n\n\n
\n
{{ 'otherslabels.no_data' | translate }}
\n
\n
","import { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { format, parseISO } from 'date-fns';\nimport {\n ColumnConfig,\n DetailsRowConfig,\n DisplaybleDataRow,\n Pagination,\n PaginationConfig,\n Row,\n ServerConfig,\n} from '../../models/server-pagination';\nimport { QueryParamsService } from '../../services/queryParams.service';\nimport { observeProperty } from '../../utils/observeProperty';\n\n@Component({\n selector: 'tu-table-server',\n templateUrl: './table-server.component.html',\n})\nexport class TableServerComponent implements OnInit {\n constructor(private queryParamsService: QueryParamsService) { }\n\n @Input() columnsConfig: ColumnConfig[] = [];\n\n @Input() data?: Row[] = [];\n @Input() idKey?: string;\n\n @Input('antiCache')\n public antiCache = false;\n\n @Input('hasCheckbox')\n public hasCheckbox = false;\n\n @Input('hasCheckboxForAll')\n public hasCheckboxForAll = true;\n\n private _filters: Record;\n\n @Input() set filters(filters: Record) {\n if (filters.resetPagination) {\n this.resetPagination();\n delete filters.resetPagination;\n }\n this._filters = filters;\n\n //\n\n if (this.displayableData) this.retrieveData();\n }\n get filters() {\n return this._filters;\n }\n\n @Input() rowLink?: (row: Row) => string;\n\n @Input() serverConfig: ServerConfig | false = false;\n @Input() fetchConfig: RequestInit = { method: 'GET' };\n\n public allSelected = false;\n public selectedRows: Row[] = [];\n\n @Input() paginationConfig: PaginationConfig = { limit: 10, offset: 0 };\n\n @Input() setQueryParams: boolean = true;\n\n public _pagination: Pagination = {\n sortableKeys: [],\n limit: 10,\n currentPage: 1,\n offset: 0,\n count: 0,\n pages: 10,\n };\n\n public get pagination() {\n return this._pagination;\n }\n public set pagination(config) {\n this._pagination = config;\n\n if (this.setQueryParams) {\n this.queryParamsService.paginationParams = { limit: config.limit, offset: config.offset };\n }\n }\n\n @Input() updateContent = true;\n\n public resetPagination() {\n if (this.setQueryParams) {\n this.queryParamsService.paginationParams = { offset: 0, limit: this.pagination.limit };\n }\n\n this.pagination.offset = 0;\n\n // Temporarly disabling that\n // this.retrieveData();\n }\n\n // The icon library: https://heroicons.com/\n public heroicons = heroicons;\n\n // The actual data that will be displayed in the table\n public displayableData: DisplaybleDataRow[];\n\n // The actualy columns that will be displayed in the table\n public get displayableColumns(): ColumnConfig[] {\n return this.columnsConfig.filter((column) => column.isDisplayed?.() ?? true);\n }\n\n // This controller is used to Abort data loading when we are changing\n // the page but didn't complete the current network request\n private abortController: AbortController;\n public isLoading = false;\n public isLoading$ = observeProperty(this, 'isLoading');\n\n /**\n * This method mostly just validate that all the critical configuration\n * is properly configured before proceeding to fetch the data from the server\n *\n * @param server {ServerConfig} - The server configuration\n */\n private validateServerConfiguration(server: ServerConfig) {\n try {\n new URL(server.url);\n } catch {\n throw new Error('INVALID_SERVER_URL');\n }\n\n const keys = ['url', 'sortableKeys', 'limit', 'data', 'offset', 'count', 'pages'];\n\n for (const key of keys) {\n if (server[key]) continue;\n throw new Error(`MISSING_MANDATORY_KEY:${key}`);\n }\n }\n\n /**\n * Given a server configuration, retrieve the data from that configuration\n * with the current parameters set in this.pagination.\n *\n * It aborts previous calls upon new requests if the previous one is unfinished.\n *\n * @param server {ServerConfig}\n */\n private async getDataFromServer(server: ServerConfig) {\n let data: Row[] = [];\n try {\n this.validateServerConfiguration(server);\n\n const url = new URL(server.url);\n\n const pagination = {\n offset: this.pagination.offset,\n limit: this.pagination.limit,\n };\n\n url.searchParams.set('pagination', JSON.stringify(pagination));\n\n if (this.filters) {\n Object.entries(this.filters).forEach(([key, value]) => {\n if (value === null) return;\n\n url.searchParams.set(key, value);\n });\n }\n\n if (this.pagination.sortBy && this.pagination.sortIn) {\n url.searchParams.set('sortIn', this.pagination.sortIn);\n url.searchParams.set('sortBy', this.pagination.sortBy);\n }\n\n if (this.antiCache) {\n // Ideally it would also set the following headers:\n // - Cache-Control: no-cache, no-store, must-revalidate\n // - Pragma: no-cache\n // - Expires: 0\n // But right this moment, it would be annoying with the current implementation\n url.searchParams.set('t', Date.now().toString());\n }\n\n if (this.abortController) this.abortController.abort();\n\n this.abortController = new AbortController();\n const { signal } = this.abortController;\n\n this.isLoading = true;\n\n const response = await fetch(url.toString(), { ...this.fetchConfig, signal });\n\n const sortableKeys = server.sortableKeys(response);\n const offset = server.offset(response);\n const limit = server.limit(response);\n const currentPage = offset / limit + 1;\n const count = server.count(response);\n const pages = Math.ceil(count / limit);\n const body = await response.json();\n data = server.data(body);\n\n this.pagination = {\n ...this.pagination,\n ...{ currentPage, sortableKeys, offset, limit, count, pages },\n };\n\n return data;\n } catch (e) {\n if (e.name === 'AbortError') return data;\n throw e;\n } finally {\n this.isLoading = false;\n }\n }\n\n @Output() onReady = new EventEmitter();\n\n public async ngOnInit() {\n if (this.setQueryParams) {\n const params = this.queryParamsService.paginationParams;\n\n const limit = params.limit || this.paginationConfig.limit;\n const offset = params.offset || this.paginationConfig.offset;\n\n this.pagination = { ...this.pagination, limit, offset };\n } else {\n this.pagination = {\n ...this.pagination,\n limit: this.paginationConfig.limit,\n offset: this.paginationConfig.offset,\n };\n }\n\n this.onReady.emit(true);\n\n this.retrieveData();\n }\n\n /**\n * This just generate a bunch of fake empty rows that serves\n * in to iterrate on the markup and show a placeholder when data\n * is still loading in on first run.\n */\n public get fakeRows() {\n return new Array(this.pagination.limit).fill('');\n }\n\n /**\n * Fetch and format the rows\n */\n public async retrieveData() {\n if (this.displayableData && !this.updateContent) return;\n\n try {\n const data = this.serverConfig ? await this.getDataFromServer(this.serverConfig) : this.data;\n this.displayableData = data.map((row) => {\n const cells = this.displayableColumns.map((column) => {\n // TODO: Add try catch if we don't find the value\n // Given the following row: { user: { lastname: 'test' } }\n // and the following key: 'user.lastname'\n // we get: ['user', 'lastname'] and reduce it to row['user']['lastname']\n const rawValue = column.key?.split('.').reduce((value, path) => value?.[path], row) ?? '';\n const cell = column.modifier?.(rawValue, row) ?? rawValue;\n\n switch (column.type) {\n case 'date': {\n const value = rawValue\n ? format(parseISO(rawValue), column.config?.format ?? 'dd/MM/yyyy HH:mm')\n : '-';\n return { value, config: column?.config };\n }\n\n // TODO: Add a fallback when the currency isn't native. eg: bitcoin\n case 'price': {\n try {\n const locale = column?.config?.locale(row) ?? 'fr-FR';\n const currency = column?.config?.currency(row) ?? 'EUR';\n\n // check if the cell value is a value able to be parsed as float\n const price = parseFloat(cell);\n\n if (Number.isNaN(price)) {\n throw new Error('INVALID_PRICE_VALUE');\n }\n\n const value = price.toLocaleString(locale, {\n style: 'currency',\n currency,\n });\n\n return { value, config: column?.config, row };\n } catch {\n return { value: '-', config: column?.config, row };\n }\n }\n\n case 'text':\n return { value: cell, config: column?.config, row };\n\n case 'input':\n return { value: cell, config: column?.config, row, key: column.key };\n case 'link':\n case 'htmlLink':\n case 'image':\n case 'badge':\n case 'icon':\n case 'actions':\n default:\n return { value: cell, config: column?.config, row };\n }\n });\n\n return { row, cells };\n });\n } catch (e) {\n console.error(e);\n this.displayableData = [];\n }\n }\n\n /**\n * Sort the rows and re-fetch the data.\n * TODO : check from pagination.sortableKeys\n * @param columnKey {string}\n */\n public sort(columnKey: string) {\n if (this.pagination.sortBy === columnKey && this.pagination.sortIn === 'ASC') {\n this.pagination.sortIn = 'DESC';\n } else if (this.pagination.sortBy === columnKey && this.pagination.sortIn === 'DESC') {\n this.pagination.sortIn = undefined;\n } else {\n this.pagination.sortIn = 'ASC';\n }\n\n this.pagination.sortBy = !this.pagination.sortIn ? undefined : columnKey;\n this.retrieveData();\n }\n\n public onChangePaginationLimit(value: string) {\n const offset = 0;\n this.pagination = { ...this.pagination, limit: parseInt(value), offset };\n\n this.retrieveData();\n }\n\n /**\n * Navigate to a specific page in the dataset and fetch the data\n *\n * @param page {number}\n */\n public goto(page: number) {\n this.pagination.offset = (page - 1) * this.pagination.limit;\n this.pagination.currentPage = page;\n\n this.retrieveData();\n }\n\n /**\n * This is used to know if a link should use or \n *\n * @param link {string}\n */\n public isInternalLink(link?: string) {\n if (!link) return false;\n return link.startsWith('/');\n }\n\n public toggleAllSelected(allSelected: boolean) {\n this.allSelected = allSelected;\n this.selectedRows = [];\n }\n public stopPropagation(event: Event) {\n event.stopPropagation();\n }\n public isRowSelected(row: Row) {\n const firstRowKey = String(Object.keys(row)[0]);\n const idKey = this.idKey || firstRowKey;\n\n return this.selectedRows.findIndex((selected) => selected[idKey] === row[idKey]);\n }\n\n public toggleSelectedRow(row: Row) {\n this.allSelected = false;\n const alreadySelected = this.isRowSelected(row);\n\n if (alreadySelected < 0) {\n this.selectedRows.push(row);\n } else {\n this.selectedRows.splice(alreadySelected, 1);\n }\n }\n\n public isFunction(value: unknown): value is Function {\n const isFunction = typeof value === 'function';\n return isFunction;\n }\n\n public activeDetails: number | null = null;\n\n public get detailsColumnsConfig(): DetailsRowConfig | null {\n const conf = this.displayableColumns.find((conf) => conf.type === 'details');\n\n if (conf) {\n return conf as DetailsRowConfig;\n }\n\n return null;\n }\n\n public toggleDetails(index: number): void {\n if (this.activeDetails === index) {\n this.activeDetails = null;\n } else {\n this.activeDetails = index;\n }\n }\n\n public trackByKey(_: number, equivalenceItem: { key: string }): string {\n return equivalenceItem.key;\n }\n\n public inputChange(event: Event, row: Row, change: (value: string, row: Row) => void): void {\n const target = event.target as HTMLInputElement;\n change(target.value, row);\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\nimport { escapeRegExp } from '@app/utils/escape-regexp';\n\n@Pipe({\n name: 'highlight',\n})\nexport class HighlightPipe implements PipeTransform {\n transform(value: string, str: string): any {\n if (value) {\n str = escapeRegExp(str);\n let re = new RegExp(`(${str})`, 'gi');\n return str ? value.toString().replace(re, '$1') : value;\n } else {\n return value;\n }\n }\n}\n","
\n
{{ 'otherslabels.loading' | translate }}
\n
\n
\n
{{ 'otherslabels.no_data' | translate }}
\n
\n\n
0\" class=\"tw-w-full tw-overflow-auto\">\n \n \n\n \n \n \n \n \n
\n \n {{ conf.label | translate }}\n \n \n \n\n \n \n {{ conf.label | translate }}\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 {{ action.label | translate }}\n \n \n \n \n
\n \n \n \n {{ (getSelectedOption(c.value, c.config.options).label) | translate }}\n \n \n \n \n {{ option.label | translate }}\n \n \n \n
\n \n \n
\n \n
Tableau générique d'affichage de données
\n \n\n \n
\n \n\n \n \n \n \n \n
\n\n \n
\n\n\n","import { DatePipe } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output, SecurityContext } from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { PaginationService as NGXPaginationService } from 'ngx-pagination';\nimport { isObservable, Observable, Subscription } from 'rxjs';\nimport { AuthService } from '../../services/auth.service';\nimport { PaginationService } from '../../services/pagination.service';\nimport { Filters } from '../filter/filter.component';\n\n@Component({\n selector: 'tu-table',\n templateUrl: './table.component.html',\n styleUrls: ['./table.component.scss'],\n providers: [DatePipe],\n})\nexport class TableComponent implements OnInit {\n public _filter: string;\n public type = TableElementType;\n public tagType = TagType;\n public sortType: { key: any; sort: Sort };\n public allChecked: boolean;\n public custom_fields: Array;\n public config: Array;\n public selectedPage = 1;\n public submenuOpen: string | false = false;\n\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n private __content: any = null;\n private __sortedContent: any = null;\n\n @Input('config') set _config(c: Array) {\n this.sortType = { key: null, sort: Sort.NONE };\n this.config = c.map((config) => {\n return {\n ...config,\n class: {\n header: config?.class?.header || '',\n row: config?.class?.row || '',\n },\n };\n });\n }\n @Input() filterObs: Observable;\n\n @Input() headers = true;\n\n @Input() componentName: string;\n\n @Input() id: string = location.pathname;\n\n /**\n * Subcription object if content is an observable\n */\n private subscription: Subscription = null;\n\n @Input() set content(content: Observable | Object[]) {\n // unsubscribe if subscribed to anything\n if (this.subscription) {\n this.subscription.unsubscribe();\n this.subscription = null;\n }\n\n // If null or empty, display nothing\n if (!content) {\n this._content = [];\n return;\n }\n\n // if only an array just update\n if (Array.isArray(content)) {\n this._content = content as any[];\n return;\n } // Converted to DataRow by setter (????)\n\n // If an observable, subscribe to it and store subscription\n if (isObservable(content)) {\n this.subscription = content.subscribe((c) => (this._content = c as any));\n return;\n }\n }\n\n @Input() fromDate: string;\n @Input() toDate: string;\n @Input() isClicked: boolean;\n @Input() perPage = 10;\n\n @Input() checkboxes: boolean;\n @Input() simpleCheckboxes = false;\n @Input() pagination: boolean;\n @Output() rowClicked: EventEmitter = new EventEmitter();\n\n @Output() checkedRows: EventEmitter = new EventEmitter();\n @Output() hasCheckedAll: EventEmitter = new EventEmitter();\n\n public tags = [\n 'info',\n 'warning',\n 'success',\n 'default',\n 'primary',\n 'danger',\n 'reg',\n 'waiting_payment',\n ];\n\n get _content(): DataRow[] {\n return this.__sortedContent;\n }\n set _content(values) {\n if (values) {\n this.__content = values.map((elem) => {\n return {\n elem,\n values: this.config.map((conf: ConfigBase) => {\n let value = conf.key.split('.').reduce((o, i) => (o[i] ? o[i] : ''), elem);\n value = conf.modifier ? conf.modifier(value, elem) : value;\n return Object.assign({}, conf, { value });\n }),\n };\n });\n this.__sortedContent = this.sort(this.__content);\n\n /**\n * The end of this function is mostly to handle a specific case\n * where the selected page that's stored on the pagination service\n * is actually higher than the total number of page.\n *\n * If we somehow detect that case, then we redirect the user to the\n * page number 1.\n *\n * The implemantation is faaaar from being clean and might not even be\n * bug free. That being said, spending an entire morning on this kind\n * of shit when that would be implemanted in a fraction of seconds\n * in ANY other technology is really worrisome, either for my career\n * or for Angular.\n */\n const instance = this.ngxPaginationService.getInstance();\n\n // This calcul was directly taken from the original implemantion\n // in the ngx-pagination repo:\n // https://github.com/michaelbromley/ngx-pagination/blob/master/src/pagination-controls.directive.ts#L168\n const totalPages = Math.max(\n Math.ceil(this.__sortedContent.length / instance.itemsPerPage),\n 1\n );\n\n if (totalPages < this.paginationService.getSelectedPage()) {\n // Why is the selected page information at THREE place???\n // I guess we will never find out...\n this.paginationService.setSelectedPage(1);\n this.selectedPage = 1;\n }\n }\n }\n constructor(\n private sanitizer: DomSanitizer,\n public paginationService: PaginationService,\n private notification: NotificationsService,\n private translate: TranslateService,\n private authService: AuthService,\n private datePipe: DatePipe,\n private ngxPaginationService: NGXPaginationService\n ) {}\n\n public trustStyle(value: string) {\n return this.sanitizer.sanitize(SecurityContext.STYLE, value);\n }\n\n public isFunction(val) {\n return typeof val === 'function';\n }\n\n public pageChanged(page) {\n this.paginationService.setSelectedPage(page, this.id);\n }\n\n public onRowClicked(elem: Object): void {\n // this.selectedPage = this.paginationService.selectedPage;\n this.rowClicked.emit(elem);\n }\n\n public onRowChecked($event, elem: Object): void {\n elem['_checked'] = $event.target.checked;\n if (!$event.target.checked) {\n this.allChecked = false;\n }\n\n this.emitCheckedRows();\n }\n\n public changeSort(key) {\n if (this.sortType.key === key)\n this.sortType.sort = (this.sortType.sort + 1) % (Object.keys(Sort).length / 2);\n else this.sortType.sort = 1;\n\n this.sortType.key = key;\n this.__sortedContent = this.sort(this.__content);\n }\n\n private sort(content: Array): Array {\n if (!content) return null;\n if (!content.length) return [];\n\n let temp = Object.assign([], content),\n val_key = this.sortType.key,\n val_sort = this.sortType.sort,\n index = null;\n\n try {\n if (temp[0] !== undefined) {\n if ((val_sort === 1 || val_sort === 2) && val_key !== null) {\n index = temp[0].values.findIndex((v) => v.key === val_key);\n const type = temp[0].values[index].type;\n\n let t = temp.sort((a, b) => {\n a = a.values[index].value;\n b = b.values[index].value;\n\n switch (type) {\n case TableElementType.Link:\n a = a.label;\n b = b.label;\n break;\n\n case TableElementType.ToolTip:\n a = a.enabledTag ? a.tag.label : a.text;\n b = b.enabledTag ? b.tag.label : b.text;\n break;\n\n case TableElementType.Price:\n a = +a;\n b = +b;\n break;\n }\n\n if (['number', 'boolean'].includes(typeof a)) return +a - +b;\n else if (a !== null) return a.localeCompare(b);\n else return a;\n });\n if (val_sort === 2) return t.reverse();\n else return t;\n } else return this.__content;\n }\n } catch (e) {\n console.error(e);\n }\n }\n\n ngOnInit() {\n if (typeof this.filterObs !== 'undefined') {\n this.filterObs.subscribe((filter) => {\n this._filter = filter.query;\n });\n }\n\n this.selectedPage = this.paginationService.getSelectedPage(this.id);\n }\n\n public checkAll(checked) {\n this._content.forEach((cont) => (cont.elem['_checked'] = checked));\n this.allChecked = checked;\n\n this.emitCheckedRows();\n }\n\n private emitCheckedRows() {\n this.checkedRows.emit(this._content.filter((cont) => cont.elem['_checked']));\n this.hasCheckedAll.emit(this.allChecked);\n }\n\n public exportToFile(_type: string) {\n const file = this.componentName || 'exports';\n\n const customFields = this.config\n .filter((column) => {\n const isExportable = this.isFunction(column.exportable)\n ? column.exportable()\n : column.exportable;\n\n // We only want to filter out the columns that have been specifically\n // marked as `false`. `undefined` and `true` both count as exportable columns.\n // The person that came up with this stupid idea doesn't work here probably.\n return isExportable !== false;\n })\n .reduce>((columns, column) => {\n columns[column.key] = column;\n return columns;\n }, {});\n\n const filteredContent = this._content\n .map((row) => row.elem)\n .filter((row) => row['_checked'] === true);\n\n if (!filteredContent.length) {\n return this.notification.warn(this.translate.instant(`otherslabels.notif_data_export`));\n }\n\n const csvData = this.generateCsv(filteredContent, customFields);\n\n const now = new Date();\n const fileName = `${file}_${this.parseDate(now)}.csv`;\n const blob = new Blob(['\\ufeff', csvData], { type: 'text/csv' });\n\n const a = document.createElement('a');\n const url = URL.createObjectURL(blob);\n\n a.setAttribute('style', 'display: none');\n document.body.appendChild(a);\n a.href = url;\n a.download = fileName;\n a.click();\n a.remove();\n\n this.notification.info(\n this.translate.instant(`otherslabels.notif_download`),\n this.translate.instant(`otherslabels.notif_download_progress`)\n );\n }\n\n public isExportable(key: string, fields: string[]): boolean {\n return fields.includes(key);\n }\n\n public generateCsv(data: Record[], fields: Record): string {\n const fieldsKey = Object.keys(fields);\n\n const isObject = (maybeObject: unknown): maybeObject is Record => {\n return typeof maybeObject === 'object';\n };\n\n const rows = isObject(data) ? data : JSON.parse(data);\n\n let str = '';\n let row = '';\n\n for (const [headerKey, headerValue] of Object.entries(data[0])) {\n if (headerValue && isObject(headerValue)) {\n if (this.isExportable(headerKey, fieldsKey)) {\n row += `\"${headerKey}\";`;\n }\n\n for (const columnKey of Object.keys(headerValue)) {\n const key = `${headerKey}.${columnKey}`;\n\n if (this.isExportable(key, fieldsKey)) {\n row += `\"${columnKey}\";`;\n\n if (fields[key].type === TableElementType.Price) {\n row += `\"${columnKey}_currency\";`;\n }\n }\n }\n } else if (this.isExportable(headerKey, fieldsKey)) {\n row += `\"${headerKey}\";`;\n\n if (fields[headerKey].type === TableElementType.Price) {\n row += `\"${headerKey}_currency\";`;\n }\n }\n }\n\n row = row.slice(0, -1);\n // append header row with line break\n str += row + '\\r\\n';\n\n for (let i = 0; i < rows.length; i++) {\n let line = '';\n\n for (const index in rows[i]) {\n if (rows[i][index] && isObject(data[0][index])) {\n if (this.isExportable(index, fieldsKey)) {\n if (line !== '') line += ';';\n const value = rows[i][index] ? rows[i][index] : '';\n\n try {\n line += `\"${fields[`${index}`].modifier?.(value) ?? value}\"`;\n } catch {\n line += `\"${value}\"`;\n }\n\n if (fields[`${index}`].type === TableElementType.Price) {\n const network = this.authService.networks.find(\n (network) => +network.id === +rows[i]['network_id']\n );\n\n if (network?.currency) line += `;\"${network.currency}\"`;\n }\n }\n\n for (const key in rows[i][index]) {\n if (this.isExportable(`${index}.${key}`, fieldsKey)) {\n if (line !== '') line += ';';\n const value = rows[i][index][key] ? rows[i][index][key] : '';\n line += `\"${value}\"`;\n\n if (fields[`${index}.${key}`].type === TableElementType.Price) {\n const network = this.authService.networks.find(\n (network) => +network.id === +rows[i]['network_id']\n );\n\n if (network?.currency) line += `;\"${network.currency}\"`;\n }\n }\n }\n } else {\n if (this.isExportable(index, fieldsKey)) {\n if (line !== '') line += ';';\n const regex = new RegExp(\n '^(-?(?:[1-9][0-9]*)?[0-9]{4})-(1[0-2]|0[1-9])-(3[01]|0[1-9]|[12][0-9])T(2[0-3]|[01][0-9]):([0-5][0-9]):([0-5][0-9])'\n );\n if (regex.test(rows[i][index])) {\n rows[i][index] = this.datePipe.transform(rows[i][index], 'yyyy-MM-dd HH:mm:ss');\n }\n const value = rows[i][index] ? rows[i][index] : '';\n line += `\"${value}\"`;\n\n if (fields[index].type === TableElementType.Price) {\n if (rows[i]['currency']) line += `;${rows[i]['currency']}`;\n else {\n const network = this.authService.networks.find(\n (network) => +network.id === +rows[i]['network_id']\n );\n\n if (network?.currency) line += `;\"${network.currency}\"`;\n }\n }\n }\n }\n }\n\n str += line + '\\r\\n';\n }\n\n return str;\n }\n\n public external(link: string): boolean {\n const regexp =\n /(ftp|http|https):\\/\\/(\\w+:{0,1}\\w*@)?(\\S+)(:[0-9]+)?(\\/|\\/([\\w#!:.?+=&%@!\\-\\/]))?/;\n return regexp.test(link);\n }\n\n public getTag(c: any): string {\n if (\n c.config.tags[c.value] &&\n typeof c.config.tags[c.value]['type'] !== 'undefined' &&\n this.tags[c.config.tags[c.value]['type']]\n ) {\n try {\n return `tag-${this.tags[c.config.tags[c.value]['type']]}`;\n } catch (e) {\n console.log({ e });\n return `tag-${this.tags[TagType.Default]}`;\n }\n } else {\n console.error(`Missing tag type for ${c.value} in`, c.config.tags, c.config.tags[c.value]);\n return `tag-${this.tags[TagType.Default]}`;\n }\n }\n\n public getTagContent(c: any, element: any): string {\n try {\n const value = `${c.value}`;\n\n if (c.config.tags[value] && typeof c.config.tags[value]['label'] !== 'undefined') {\n const label = c.config.tags[value].label;\n if (!label) return '';\n\n if (this.isFunction(label)) {\n return label(element);\n } else {\n return this.translate.instant(label);\n }\n } else {\n console.error(`Missing Content tag for ${value} in`, c.config.tags, c.config.tags[value]);\n return this.translate.instant('otherslabels.unkown');\n }\n } catch (error) {\n console.error({ error, c, element });\n return this.translate.instant('otherslabels.unkown');\n }\n }\n\n public stringToArray(str: string | string[]) {\n return Array.isArray(str) ? str : [str];\n }\n\n private parseDate(date: Date): string {\n let day = `0${date.getDate()}`.slice(-2),\n month = `0${date.getMonth() + 1}`.slice(-2),\n year = date.getFullYear().toString();\n\n return `${day}_${month}_${year}`;\n }\n\n public getSelectedOption(value: string, options: ConfigSelectOption[]): ConfigSelectOption {\n return options.find((opt) => opt.value === value);\n }\n\n public getSelectedTagOption(value: string, options: ConfigSelectOption[]): TagType {\n const optValue = this.getSelectedOption(value, options);\n\n if (optValue) {\n return optValue.type;\n }\n\n return TagType.Default;\n }\n\n public selectIsDisabled(config: ConfigSelect, value: string): boolean {\n if (typeof config.config.disabled === 'boolean') {\n return config.config.disabled;\n }\n\n if (typeof config.config.disabled === 'function') {\n return config.config.disabled(value);\n }\n\n return false;\n }\n}\n\nexport enum TableElementType {\n Avatar,\n Date,\n Email,\n Function,\n Icon,\n Image,\n Link,\n Price,\n Submenu,\n Tag,\n Text,\n ToolTip,\n Select,\n}\n\nexport enum TagType {\n Info,\n Warning,\n Success,\n Default,\n Primary,\n Danger,\n Reg,\n waiting_payment,\n}\n\nexport enum Sort {\n NONE,\n ASC,\n DESC,\n}\n\nexport type Config =\n | ConfigBase\n | ConfigIcon\n | ConfigTag\n | ConfigImage\n | ConfigAvatar\n | ConfigToolTip\n | ConfigPrice\n | ConfigSelect;\n\nexport interface ConfigBase extends Record {\n key: string;\n label: string;\n type: TableElementType;\n exportable?: boolean | { (): boolean };\n displayed?: boolean | { (): boolean };\n modifier?: { (value: any, element: any): any };\n format?: string;\n style?: string | Function;\n isSortable?: boolean;\n class?: {\n header?: string;\n row?: string;\n };\n}\n\nexport interface ConfigAvatar extends ConfigBase {\n type: TableElementType.Avatar;\n imgDefault: string;\n}\n\nexport interface ConfigImage extends ConfigBase {\n type: TableElementType.Image;\n imgDefault?: string;\n config?: {\n height?: string;\n width?: string;\n };\n}\n\nexport interface ConfigIcon extends ConfigBase {\n type: TableElementType.Icon;\n config?: {\n source?: string;\n icons?: { [icon: string]: string | string[] };\n label?: string;\n };\n}\n\nexport interface ConfigTag extends ConfigBase {\n type: TableElementType.Tag;\n config?: {\n tags?: { [tag: string]: { type: TagType; label: string | Function } };\n };\n}\n\nexport interface ConfigPrice extends ConfigBase {\n type: TableElementType.Price;\n currency?: (element: any) => string | string;\n locale?: (element: any) => string | string;\n}\n\nexport interface ConfigToolTip extends ConfigBase {\n type: TableElementType.ToolTip;\n placement?: string;\n tooltip?: string;\n text?: string;\n enabledTag?: boolean;\n tag?: { type: TagType; label: string };\n}\n\nexport interface ConfigWithValue extends ConfigBase {\n value: any;\n}\n\nexport interface ConfigSelect extends ConfigBase {\n type: TableElementType.Select;\n config: {\n options: ConfigSelectOption[];\n disabled: boolean | ((value: string) => boolean);\n };\n change: (value: string, element: any) => void;\n}\n\nexport interface ConfigSelectOption {\n label: string;\n value: string;\n type: TagType;\n disabled?: boolean;\n}\n\nexport interface DataRow {\n elem: any;\n values: ConfigWithValue[];\n}\n","import { Component, Input } from '@angular/core';\n\n@Component({\n selector: 'tu-tab',\n templateUrl: './tab.component.html',\n})\nexport class TabComponent {\n @Input('tabTitle') title: string;\n @Input() active = false;\n @Input() disabled = false;\n}\n","
\n \n
\n","
    \n \n
  • \n \n {{ tab.title }}\n \n
  • \n
    \n
\n\n","import {\n AfterContentInit,\n Component,\n ContentChildren,\n EventEmitter,\n Output,\n QueryList,\n} from '@angular/core';\nimport { TabComponent } from './tab.component';\n\n@Component({\n selector: 'tu-tabs',\n templateUrl: './tabs.component.html',\n styleUrls: ['./tabs.component.scss'],\n})\nexport class TabsComponent implements AfterContentInit {\n @ContentChildren(TabComponent) tabs: QueryList;\n\n @Output()\n public tabChange = new EventEmitter();\n\n constructor() { }\n\n ngAfterContentInit() {\n let activeTabs = this.tabs.filter((tab) => tab.active);\n if (activeTabs.length === 0) {\n this.selectTab(this.tabs.first);\n }\n }\n\n selectTab(tab: TabComponent) {\n if (!tab) return;\n\n this.tabs.toArray().forEach((bat) => (bat.active = false));\n tab.active = true;\n this.tabChange.emit();\n }\n}\n","\n \n \n \n \n\n","import { Component, Input } from '@angular/core';\n\nexport type TOOLTIP_DESIGN = 'info' | 'warning' | 'definition';\n\n@Component({\n selector: 'tu-tooltip',\n templateUrl: './tooltip.component.html',\n})\nexport class TooltipComponent {\n @Input() design: TOOLTIP_DESIGN = 'info';\n @Input() message: string = null;\n @Input() config: {\n theme?: string;\n animation?: string;\n position?: string;\n width?: string;\n classList?: string;\n bounded?: boolean; //Block position relative to component\n };\n\n //Click display\n private _clickDisplay = {\n show: false,\n timeout: null,\n delay: 5000,\n };\n public get clickDisplay() {\n return this._clickDisplay.show;\n }\n public set clickDisplay(show: boolean) {\n this._clickDisplay.show = show;\n }\n public toggleTooltip() {\n clearTimeout(this._clickDisplay.timeout);\n this.clickDisplay = !this.clickDisplay;\n if (this.clickDisplay) {\n this._clickDisplay.timeout = setTimeout(() => {\n this.clickDisplay = false;\n }, this._clickDisplay.delay);\n }\n }\n\n public get classIcon() {\n let classIcon = `tw-group tw-ml-1 tw-text-lg fa \n ${this?.config?.bounded === false ? null : `tw-relative`} \n `;\n return `${classIcon} ${this._design}`;\n }\n\n public get classBlock() {\n return `tw-block tw-absolute tw-z-50 ${this._theme} ${this._width} ${this._position} ${this._animation} ${this.classList}`;\n }\n\n private get _design() {\n switch (this.design) {\n case 'warning':\n return `fa-exclamation-triangle tw-text-yellow-500`;\n case 'definition':\n return `fa-question-circle-o tw-text-gray-500`;\n default: //Infos\n return `fa-info-circle tw-text-blue-500`;\n }\n }\n private get _theme() {\n if (!this?.config?.theme) return `tw-bg-black tw-text-white tw-p-4 tw-font-sans tw-text-xs`;\n return this.config.theme;\n }\n private get _animation() {\n return `tw-transition-transform tw-origin-top tw-scale-y-0 group-hover:tw-scale-y-100 ${\n this.clickDisplay ? 'tw-scale-y-100' : ''\n }`;\n }\n private get _position() {\n if (!this?.config?.position) return `tw-left-0 tw-top-full`; //left bottom\n\n if (this.config.position === 'custom') {\n return '';\n }\n\n const positions = [];\n //Top or Bottom\n if (this.config.position.includes('top')) positions.push('tw-bottom-full');\n else positions.push('tw-top-full');\n //Left, Center or Right\n if (this.config.position.includes('left')) positions.push('tw-left-0');\n else if (this.config.position.includes('right')) positions.push('tw-right-0');\n else positions.push('tw-left-1/2 -tw-translate-x-1/2');\n\n return positions.join(' ');\n }\n private get _width() {\n if (!this?.config?.width) return `tw-w-72`;\n return this.config.width;\n }\n\n private get classList(): string {\n return this.config?.classList ?? '';\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, map } from 'rxjs';\n\n@Injectable({\n providedIn: 'root',\n})\nexport class PaperworkService {\n constructor(private http: HttpClient) {}\n\n public updateDocument(\n documentId: number,\n content: {\n status?: string;\n startingAt?: string;\n expiresAt?: string;\n comment?: string;\n examined?: string;\n }\n ) {\n return firstValueFrom(\n this.http\n .patch(`${env.config.feedRoot}Paperwork/updateDocument`, {\n documentId,\n ...content,\n })\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n}\n","

{{ 'pages.shared.document_modal_title' | translate }}

\n\n\n \n\n

\n {{ 'pages.shared.document_modal_error' | translate }}\n

\n
\n\n\n \n\n \n {{ 'otherslabels.btn_save' | translate }}\n \n\n","import { Uppy } from '@uppy/core';\nimport { firstValueFrom } from 'rxjs';\nimport ImageEditor from '@uppy/image-editor';\nimport French from '@uppy/locales/lib/fr_FR';\nimport { Component, Inject, OnInit } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { SubmissionService } from '@app/modules/submissions/services/submission.service';\nimport { PaperworkService } from '@app/modules/submissions/services/paperwork.service';\n\nexport interface DialogData {\n documentTypeId: string;\n userId: string;\n submissionId?: string;\n ignoreDocumentStatus?: boolean;\n documentId?: number;\n documentStatus?: string;\n}\n\n@Component({\n selector: 'tu-update-document-modal',\n templateUrl: './update-document-modal.component.html',\n})\nexport class UpdateDocumentModalComponent implements OnInit {\n public uppy: Uppy;\n public isLoading = false;\n public errorInitializingUppy = false;\n\n constructor(\n @Inject(MAT_DIALOG_DATA) public data: DialogData,\n private submissionService: SubmissionService,\n private paperworkService: PaperworkService,\n public modalRef: MatDialogRef\n ) {}\n\n public ngOnInit() {\n try {\n const uppy = new Uppy({\n debug: true,\n autoProceed: true,\n locale: French,\n allowMultipleUploadBatches: false,\n restrictions: {\n maxNumberOfFiles: 1,\n minNumberOfFiles: 1,\n allowedFileTypes: ['image/*', 'application/pdf'],\n },\n });\n\n uppy.use(ImageEditor, {\n quality: 0.8,\n });\n\n this.uppy = uppy;\n } catch (error) {\n console.error(error);\n this.errorInitializingUppy = true;\n }\n }\n\n public get hasFiles(): boolean {\n return Object.keys(this.uppy.state.files).length > 0;\n }\n\n public async updateDocument() {\n this.isLoading = true;\n\n try {\n const [file] = Object.values(this.uppy.state.files);\n\n // If ignore doc status -> set status as INVALID -> send the document -> reset previous document status\n if (this.data.ignoreDocumentStatus && this.data.documentId && this.data.documentStatus) {\n await this.paperworkService.updateDocument(this.data.documentId, {\n status: 'INVALID',\n });\n\n await this.sendDocument(file.data as File);\n\n await this.paperworkService.updateDocument(this.data.documentId, {\n status: this.data.documentStatus,\n });\n } else {\n await this.sendDocument(file.data as File);\n }\n\n if (this.data.submissionId) {\n await firstValueFrom(\n this.submissionService.updateSubmission(this.data.submissionId, 'PENDING', [])\n );\n }\n\n this.modalRef.close(true);\n } catch {\n this.uppy.reset();\n } finally {\n this.isLoading = false;\n }\n }\n\n private async sendDocument(file: File) {\n return this.submissionService.updateDocument({\n ...this.data,\n file: file,\n });\n }\n}\n","import { Directive, ElementRef, HostBinding, Input } from '@angular/core';\nimport { from, Subject } from 'rxjs';\nimport { map, mergeMap } from 'rxjs/operators';\n\n@Directive({\n selector: 'img[default]',\n})\nexport class DefaultImageDirective {\n private loadingSrc = '/assets/img/load.gif';\n private imgObs = new Subject();\n\n @Input() default;\n\n @HostBinding('src') imgSrc: string;\n @Input('src') set src(src: string) {\n this.imgObs.next(src);\n }\n\n constructor(private el: ElementRef) {\n this.imgSrc = this.loadingSrc;\n\n this.imgObs\n .pipe(\n mergeMap((src) => from(this.loadImage(src))),\n map((img) => img || { src: this.default })\n )\n .subscribe((img) => (this.imgSrc = img.src));\n }\n\n private loadImage(src: string): Promise {\n if (!src) return Promise.resolve(null);\n\n return new Promise((resolve) => {\n const img = Object.assign(new Image(), {\n src,\n onload: () => resolve(img),\n onerror: () => resolve(null),\n });\n });\n }\n}\n","import { Directive, ElementRef, Input, OnInit } from '@angular/core';\nimport { FeatureGuard } from '../guards/feature.guard';\n\n@Directive({\n selector: '[hasFeature]',\n})\nexport class HasFeatureDirective implements OnInit {\n @Input('hasFeature') feature: string;\n\n constructor(private featureGuard: FeatureGuard, private element: ElementRef) {}\n\n ngOnInit() {\n if (this.feature) {\n const allowed = this.featureGuard.hasFeature(this.feature);\n\n if (!allowed) {\n this.element.nativeElement.remove();\n }\n }\n }\n}\n","import { Directive, ElementRef, Input, OnInit } from '@angular/core';\nimport { PermissionGuard } from '../guards/permission.guard';\n\n@Directive({\n selector: '[hasOnePermission], [hasAllPermission]',\n})\nexport class HasPermissionDirective implements OnInit {\n @Input('hasOnePermission') onePer: Array;\n @Input('hasAllPermission') allPer: Array;\n\n constructor(public permissionGuard: PermissionGuard, private element: ElementRef) {}\n\n ngOnInit() {\n if (this.notEmpty(this.onePer)) {\n let allowed = this.permissionGuard.isOnePermissionAllowed(this.onePer);\n if (!allowed) this.element.nativeElement.remove();\n }\n\n if (this.notEmpty(this.allPer)) {\n let allowed = this.permissionGuard.isAllPermissionAllowed(this.allPer);\n if (!allowed) this.element.nativeElement.remove();\n }\n }\n\n private notEmpty(value: any): boolean {\n if (Array.isArray(value)) return value.length > 0;\n return !!value;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { CanActivate, CanLoad, Router } from '@angular/router';\nimport { AuthService } from '../services/auth.service';\n\n@Injectable()\nexport class AuthGuard implements CanActivate, CanLoad {\n constructor(private authService: AuthService, private router: Router) {}\n\n async canActivate(route) {\n // If we have a token on the URL, we try to login with the token\n if (route.queryParams.token) {\n this.authService.tokenLoader = true;\n await this.authService.loginWithToken(route.queryParams.token);\n\n this.authService.tokenLoader = false;\n // Redirect on the same url without token\n return this.router.navigate([route._routerState.url], {\n queryParams: {\n token: null,\n },\n queryParamsHandling: 'merge',\n });\n }\n\n return this.authService.checkLogin();\n }\n\n async canLoad() {\n const currentRoute = this.router.getCurrentNavigation().extractedUrl;\n\n // If we have a token on the URL, we try to login with the token\n if (currentRoute.queryParams.token) {\n this.authService.tokenLoader = true;\n await this.authService.loginWithToken(currentRoute.queryParams.token);\n\n let currentPath = [];\n\n if (\n currentRoute.root.children['primary'] &&\n currentRoute.root.children['primary'].segments.length\n ) {\n currentPath = currentRoute.root.children['primary'].segments.map(({ path }) => path);\n }\n\n this.authService.tokenLoader = false;\n // Redirect on the same url without token\n return this.router.navigate(currentPath, {\n queryParams: {\n ...currentRoute.queryParams,\n token: null,\n },\n queryParamsHandling: 'merge',\n });\n }\n\n return this.authService.checkLogin();\n }\n}\n","import { Injectable } from '@angular/core';\nimport { CanDeactivate } from '@angular/router';\nimport { Observable } from 'rxjs';\n\nexport interface CanComponentDeactivate {\n canDeactivate: () => Observable | Promise | boolean;\n}\n\n@Injectable()\nexport class CanDeactivateGuard implements CanDeactivate {\n canDeactivate(component: CanComponentDeactivate) {\n return component.canDeactivate ? component.canDeactivate() : true;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivate, CanActivateChild, Router } from '@angular/router';\n\n@Injectable()\nexport class ConfigGuard implements CanActivate, CanActivateChild {\n constructor(private router: Router) {}\n\n canActivate(route: ActivatedRouteSnapshot) {\n const features = route.data.features as string[];\n\n const canActivate = features.every((feature) => {\n return Boolean(window['__CONFIG__']?.features?.[feature]);\n });\n\n if (!canActivate) {\n this.router.navigate(['/403']);\n\n return canActivate;\n }\n\n return canActivate;\n }\n\n canActivateChild(route: ActivatedRouteSnapshot) {\n const features = route.data.features as string[];\n\n const canActivateChild = features.every((feature) => {\n return Boolean(window['__CONFIG__']?.features?.[feature]);\n });\n\n if (!canActivateChild) {\n this.router.navigate(['/403']);\n\n return canActivateChild;\n }\n\n return canActivateChild;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivate, CanActivateChild, Router } from '@angular/router';\nimport { AuthService } from '../services/auth.service';\nimport { NetworkFeatureHandle } from '../models/network';\n\n@Injectable()\nexport class FeatureGuard implements CanActivate, CanActivateChild {\n constructor(private router: Router, private authService: AuthService) {}\n\n canActivate() {\n return true;\n }\n\n canActivateChild(route: ActivatedRouteSnapshot) {\n const data = route.data.feature as {\n handle: string;\n redirectTo: string;\n };\n if (typeof data === 'undefined') return true;\n\n const hasFeature = this.hasFeature(data.handle);\n\n if (hasFeature) {\n return true;\n }\n\n if (data.redirectTo) {\n this.router.navigate([data.redirectTo]);\n }\n\n return false;\n }\n\n hasFeature(handle: string): boolean {\n if (Object.keys(NetworkFeatureHandle).includes(handle)) {\n return this.authService.hasFeature(handle as NetworkFeatureHandle);\n } else {\n throw new Error(`Unknown ${handle} feature`);\n }\n }\n}\n","import { Injectable } from '@angular/core';\nimport { CanActivate, Router } from '@angular/router';\nimport { AuthService } from '../services/auth.service';\n\n@Injectable()\nexport class HomeGuard implements CanActivate {\n private homes = [\n {\n permission: 'ACCESS_DASHBOARD',\n route: '/dashboard-ng',\n },\n {\n permission: 'ACCESS_ORDERS',\n route: '/orders',\n },\n {\n permission: 'ACCESS_SCHOOL_ORDERS',\n route: '/school',\n },\n {\n permission: 'ACCESS_COLORS',\n route: '/colors',\n },\n {\n permission: 'ACCESS_SUBMISSIONS',\n route: '/submissions',\n },\n {\n permission: 'ACCESS_REPORTS',\n route: '/reports',\n },\n {\n permission: 'TAPNPROTECT',\n route: '/tap-and-protect/dashboard',\n },\n {\n permission: 'TAPNPROTECT',\n route: '/tap-and-protect/dashboard',\n },\n {\n permission: 'EDIT_ITEM_PRODUCT_RESTRICTED_CREATOR',\n route: '/shop/products',\n },\n {\n permission: 'EDIT_ITEM_PRODUCT_RESTRICTED_MODIFICATOR',\n route: '/shop/products',\n },\n {\n permission: 'EDIT_ITEM_PRODUCT_RESTRICTED_EDITOR',\n route: '/shop/products',\n },\n {\n permission: 'MANAGE_STORE',\n route: '/shop/products',\n },\n ];\n\n constructor(private router: Router, private authService: AuthService) {}\n\n canActivate(): boolean {\n const path = this.homes.find((home) => this.authService.permissions.includes(home.permission));\n\n if (path) {\n this.router\n .navigate([path.route], { queryParamsHandling: 'merge' })\n .catch((e) => console.error(e));\n } else {\n this.authService.logout();\n this.router.navigate(['/login']);\n }\n\n return false;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivate, Router, RouterStateSnapshot } from '@angular/router';\nimport { AuthService } from '../services/auth.service';\n\n@Injectable()\nexport class OneNetworkGuard implements CanActivate {\n constructor(private router: Router, private authService: AuthService) {}\n\n canActivate(route: ActivatedRouteSnapshot, state: RouterStateSnapshot): boolean {\n if (!this.authService.networks.length) return false;\n\n if (this.authService.networks.length > 1) return true;\n else this.router.navigate([state.url, this.authService.networks[0].id]);\n }\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRouteSnapshot, CanActivate, CanActivateChild, Router } from '@angular/router';\nimport { AuthService } from '../services/auth.service';\n\n@Injectable()\nexport class PermissionGuard implements CanActivate, CanActivateChild {\n constructor(private router: Router, private authService: AuthService) {}\n\n canActivate() {\n return true;\n }\n\n canActivateChild(route: ActivatedRouteSnapshot) {\n let data = route.data.permissions as {\n One?: Array;\n All?: Array;\n superAdmin?: boolean;\n redirectTo: string;\n };\n if (typeof data === 'undefined') return true;\n\n if (typeof data.One !== 'undefined') {\n if (this.isOnePermissionAllowed(data.One)) return true;\n } else if (typeof data.All !== 'undefined') {\n if (this.isAllPermissionAllowed(data.All)) return true;\n } else if (data?.superAdmin === true) {\n if (this.authService.isSuperAdmin) return true;\n }\n\n if (data.redirectTo && data.redirectTo !== undefined) this.router.navigate([data.redirectTo]);\n\n return false;\n }\n\n isOnePermissionAllowed(permissions: Array): boolean {\n if (!Array.isArray(permissions)) throw 'permissions must be an array !';\n if (permissions) {\n return permissions.some((p) => this.authService.permissions.indexOf(p) >= 0);\n }\n return false;\n }\n\n isAllPermissionAllowed(permissions: Array): boolean {\n if (!Array.isArray(permissions)) throw 'permissions must be an array !';\n if (permissions) {\n const isValid = permissions.every((p) => this.authService.permissions.indexOf(p) >= 0);\n\n return isValid;\n }\n return false;\n }\n}\n","export interface Network {\n id: string;\n name: string;\n lat: number;\n lng: number;\n universal: boolean;\n od: boolean;\n agency_key: string;\n payment_company: string;\n currency?: string;\n defaultLocale?: string;\n /**\n * @deprecated\n * This field doesn't exist in the two API calls we use\n * to fill the `auth.networks` array. It was only existing in\n * the `/login` which is is called in another authentication service now.\n *\n * To bypass this limitation we added the information in the `data` of networks.\n */\n tap_channel: null | string | number;\n locales?: string[];\n offlineEnabled: boolean;\n hasTransferableTickets: boolean;\n hasPaymentPreauthorization: boolean;\n hasMediaSignin: boolean;\n timezone?: string;\n data?: any;\n hasReservation?: boolean;\n attestationProviderId?: number;\n features: NetworkFeature[];\n}\n\nexport interface NetworkFeature {\n id: number;\n handle: NetworkFeatureHandle;\n createdAt: string;\n}\n\nexport enum NetworkFeatureHandle {\n PROFILING = 'PROFILING',\n PHONE_MANDATORY = 'PHONE_MANDATORY',\n MULTIPLE_PAYMENT = 'MULTIPLE_PAYMENT',\n SHIPPING = 'SHIPPING',\n SAVE_CARDS = 'SAVE_CARDS',\n QR_CODE_AVAILABLE = 'QR_CODE_AVAILABLE',\n QR_CODE_REQUIRED = 'QR_CODE_REQUIRED',\n OD = 'OD',\n USE_PREAUTHORIZATION = 'USE_PREAUTHORIZATION',\n PAYMENT_PREAUTHORIZATION = 'PAYMENT_PREAUTHORIZATION',\n USE_MAILING_PLATFORM = 'USE_MAILING_PLATFORM',\n SUBSCRIPTIONS = 'SUBSCRIPTIONS',\n OFFLINE = 'OFFLINE',\n ORDER_CUSTOMER_FAMILY_ONLY = 'ORDER_CUSTOMER_FAMILY_ONLY',\n GUEST_ORDERS = 'GUEST_ORDERS',\n TRANSFERABLE_TICKETS = 'TRANSFERABLE_TICKETS',\n MEDIA_SIGNIN = 'MEDIA_SIGNIN',\n INSPECTION_SESSIONS = 'INSPECTION_SESSIONS',\n MEDIA_TUNNEL = 'MEDIA_TUNNEL',\n MEDIA_TUNNEL_PAIRING = 'MEDIA_TUNNEL_PAIRING',\n RESERVATION = 'RESERVATION',\n VIGNETTE = 'VIGNETTE',\n}","export type TicketStatus = 'EXPIRED' | 'PENDING' | 'CANCELED' | 'REMAINING' | 'ACTIVE';\nexport type TicketPurpose = 'DEVICE_CHANGE' | 'COMMERCIAL_GESTURE' | 'COMMERCIAL_OFFER' | 'TEST';\nexport const TicketPurposeLabel = {\n DEVICE_CHANGE: `otherslabels.purpose_device_change`,\n COMMERCIAL_GESTURE: `otherslabels.purpose_commercial_gesture`,\n COMMERCIAL_OFFER: `otherslabels.purpose_commercial_offer`,\n CUSTOMER_SUPPORT: `otherslabels.support`,\n TEST: `otherslabels.purpose_test`,\n};\n\nexport interface Ticket {\n id: string;\n ticket_id: string;\n order_identifier?: string;\n order_id?: string;\n user_photo?: string;\n customer_id: string;\n date: string;\n expire_at: string;\n user_id: string;\n lastname: string;\n firstname: string;\n email: string;\n label: string;\n start_date: Date;\n end_date: Date;\n status: TicketStatus;\n punch_number?: null | string;\n network_id: string;\n photo_mandatory?: string;\n dematerialized: '0' | '1';\n universal?: boolean;\n ticket_order_issuer: 'airweb' | 'sncf';\n client: {\n id: string;\n name: string;\n slug: string;\n } | null;\n stop: { id: string; code: string; name: string } | null;\n stop_code: string | null;\n line: { id: string; code: string; name: string } | null;\n line_code: string | null;\n generationReason?: TicketPurpose | null;\n product_id?: string | null;\n cms_user_id?: string;\n operator?: {\n id: string;\n login: string;\n firstName: string;\n lastName: string;\n };\n}\n","import { Pipe, PipeTransform } from '@angular/core';\n\n@Pipe({ name: 'key' })\nexport class KeyPipe implements PipeTransform {\n transform(content: T[], key: string, value: any): T[] {\n if (!content || typeof value === 'undefined' || value === null) return content;\n if (!content.length || !(key in content[0])) return content; // if key doesn't appear in content\n\n return content.filter((data) => {\n const contentVal = key.split('.').reduce((a, c) => a[c], data);\n if (Array.isArray(contentVal)) return contentVal.some((v) => v == value);\n else return value == contentVal;\n });\n }\n}\n","import { Pipe, PipeTransform } from '@angular/core';\nimport { escapeRegExp } from '@app/utils/escape-regexp';\n\n@Pipe({\n name: 'search',\n pure: true,\n})\nexport class SearchPipe implements PipeTransform {\n transform(value: Object[], filter_str: string): any {\n if (value && filter_str) {\n return value.filter((val) => this.contains(val, filter_str));\n } else {\n return value;\n }\n }\n\n private contains(val: string | number | Object, str: string): boolean {\n let res = false;\n if (typeof val === 'object') {\n for (let v in val) {\n if (!val.hasOwnProperty(v)) {\n continue;\n }\n res = this.contains(val[v], str) || res;\n }\n } else if (typeof val === 'string') {\n let re = new RegExp(escapeRegExp(str), 'gi');\n res = re.test(val);\n } else if (typeof val === 'number') {\n let re = new RegExp(str, 'gi');\n res = re.test(val.toString());\n }\n return res;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Resolve } from '@angular/router';\nimport { NetworkService } from '@app/modules/network/services/network.service';\nimport { Observable } from 'rxjs';\n\n@Injectable()\nexport class EmailsNetworkResolver implements Resolve {\n constructor(private networkService: NetworkService) {}\n\n resolve(): Observable {\n return this.networkService.getMails();\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Resolve } from '@angular/router';\nimport { Observable } from 'rxjs';\nimport { LinesService } from '../services/lines.service';\n\n@Injectable()\nexport class LinesResolver implements Resolve {\n constructor(private lines: LinesService) {}\n\n resolve(): Observable {\n return this.lines.getLines();\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Resolve } from '@angular/router';\nimport { Observable } from 'rxjs';\nimport { PermissionsService } from '../services/permissions.service';\n\n@Injectable()\nexport class RolesResolver implements Resolve {\n constructor(private permissions: PermissionsService) {}\n\n resolve(): Observable {\n return this.permissions.getRoles();\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Resolve } from '@angular/router';\nimport { NetworkService } from '@app/modules/network/services/network.service';\nimport { Observable } from 'rxjs';\n\n@Injectable()\nexport class TagsMailNetworkResolver implements Resolve {\n constructor(private networkService: NetworkService) {}\n\n resolve(): Observable {\n return this.networkService.getTags();\n }\n}\n","import { map } from 'rxjs/operators';\nimport { Router } from '@angular/router';\nimport { Injectable } from '@angular/core';\nimport { firstValueFrom, Observable } from 'rxjs';\nimport { HttpClient } from '@angular/common/http';\n\nimport { Network, NetworkFeatureHandle } from '../models/network';\nimport { Session } from '../models/session';\nimport { ModalService } from './modal.service';\nimport { FilterService } from './filter.service';\nimport { environment as env } from '@env/environment';\nimport { GenericResponse } from '../models/generic-response';\n\n@Injectable()\nexport class AuthService {\n public redirectUrl: string;\n public tokenLoader: boolean = false;\n private _networks: Network[] = JSON.parse(localStorage.getItem('networks'));\n\n constructor(\n private router: Router,\n private http: HttpClient,\n private modal: ModalService,\n private filterService: FilterService\n ) {}\n\n set username(username: string) {\n if (username) localStorage.setItem('username', username);\n else localStorage.removeItem('username');\n }\n get username(): string {\n return localStorage.getItem('username');\n }\n get loggedIn(): boolean {\n return Boolean(this.username);\n }\n\n set token(token: string) {\n if (token) {\n localStorage.setItem('token', token);\n //Auth cookie\n let cookieContent = `AW_AUTH_SESSION=${token}; SameSite=None; Secure;`;\n const [extension, domain] = window.location.hostname.split('.').reverse();\n if (domain && extension) {\n const DOMAIN = `.${domain}.${extension}`;\n cookieContent += `Domain=${DOMAIN}; Path:/`;\n }\n document.cookie = cookieContent;\n } else {\n localStorage.removeItem('token');\n this.clearAuthCookie();\n }\n }\n get token(): string {\n return localStorage.getItem('token');\n }\n\n set user_id(userId: string) {\n if (userId) localStorage.setItem('user_id', userId);\n else localStorage.removeItem('user_id');\n }\n get user_id(): string {\n return localStorage.getItem('user_id');\n }\n\n set networks(networks: Network[]) {\n // HACK: reset cache on networks\n delete this._networks;\n this._networks = networks;\n\n if (networks?.length) {\n // We copy the networks into it's own array, because the `sort` function mutates the original array...\n localStorage.removeItem('networks');\n const networksForLocalStorage = [...networks].sort((a, b) => a.name.localeCompare(b.name));\n localStorage.setItem('networks', JSON.stringify(networksForLocalStorage));\n } else {\n localStorage.removeItem('networks');\n }\n }\n\n get networks(): Network[] {\n try {\n if (this._networks?.length) {\n return this._networks;\n }\n\n const networks = JSON.parse(localStorage.getItem('networks'));\n\n if (Array.isArray(networks)) {\n return networks;\n } else {\n return [];\n }\n } catch {\n return [];\n }\n }\n\n set currencies(currencies: string[]) {\n if (currencies) localStorage.setItem('currencies', JSON.stringify(currencies));\n else localStorage.removeItem('currencies');\n }\n get currencies(): string[] {\n return JSON.parse(localStorage.getItem('currencies'));\n }\n\n private normalizePermissions(permissions: string[] | Record) {\n if (!permissions) return [];\n\n return Array.isArray(permissions) ? permissions : Object.values(permissions);\n }\n\n set permissions(permissions: Array) {\n localStorage.setItem('permissions', JSON.stringify(this.normalizePermissions(permissions)));\n }\n get permissions(): Array {\n let tmp = localStorage.getItem('permissions');\n return tmp ? this.normalizePermissions(JSON.parse(tmp)) : [];\n }\n\n set roleNames(permissions: Array) {\n localStorage.setItem('roles', JSON.stringify(this.normalizePermissions(permissions)));\n }\n get roleNames(): Array {\n let tmp = localStorage.getItem('roles');\n return tmp ? this.normalizePermissions(JSON.parse(tmp)) : [];\n }\n\n get isSuperAdmin(): boolean {\n return this.roleNames.some((role) => role === 'SUPER_ADMIN');\n }\n\n public hasPermission(permission: string) {\n return this.permissions.includes(permission);\n }\n\n public login(user: string, password: string): Observable> {\n return this.http\n .post(env.config.feedRoot + `Authenticate/login`, { user, password })\n .pipe(\n map(async ({ response }) => {\n if (!response.success) return false;\n\n this.user_id = response.id;\n this.networks = response.networks;\n this.currencies = response.networks.reduce((currencies, network) => {\n if (currencies.includes(network.currency)) return currencies;\n\n return [...currencies, network.currency];\n }, []);\n this.permissions = response.permissions;\n this.username = user;\n this.token = response.token;\n const session = await this.session();\n this.roleNames = session.user?.roles?.map(({ name }) => name);\n return response.success;\n })\n );\n }\n\n public async logout() {\n try {\n await firstValueFrom(\n this.http.post>(env.config.feedRoot + `Authenticate/logout`, {})\n );\n } catch {\n } finally {\n this.clearStorage();\n this.filterService.network = null;\n this.networks = [];\n }\n }\n\n public session() {\n return firstValueFrom(\n this.http\n .get>(env.config.feedRoot + `Authenticate/session`, {})\n .pipe(map(({ response }) => response))\n );\n }\n\n private clearAuthCookie() {\n document.cookie = document.cookie\n .split(';')\n .map((cookie) =>\n cookie.includes('AW_AUTH_SESSION')\n ? `AW_AUTH_SESSION=;expires=Thu, 01 Jan 1970 00:00:00 GMT`\n : cookie\n )\n .join(';');\n }\n\n public clearStorage() {\n localStorage.removeItem('username');\n localStorage.removeItem('networks');\n localStorage.removeItem('currencies');\n localStorage.removeItem('user_id');\n localStorage.removeItem('permissions');\n localStorage.removeItem('token');\n localStorage.removeItem('roles');\n this.clearAuthCookie();\n }\n\n public checkLoggedIn(response: any) {\n if (response?.errorMessage) {\n this.clearStorage();\n }\n }\n\n public checkLogin(): boolean {\n // Store the attempted URL for redirecting\n this.redirectUrl = location.pathname;\n\n // Navigate to the login page with extras\n if (!this.loggedIn) {\n this.router.navigate(['/login'], {\n queryParams: { returnUrl: location.pathname },\n queryParamsHandling: 'merge',\n });\n }\n\n return this.loggedIn;\n }\n\n public getNetwork(networkId: string): Network {\n return this.networks.find((network) => +network.id === +networkId);\n }\n\n public getUserNetworks() {\n return this.http\n .get(`${env.config.feedRoot}/Network/updateLocalNetworks`)\n .pipe(map(({ response }) => response?.networks));\n }\n\n public getNetworks() {\n const antiCache = Math.random().toString(36).slice(2, 5);\n\n return this.http\n .get(`${env.config.feedRoot}/Network/getNetworks`, {\n params: {\n withId: true,\n antiCache,\n },\n })\n .pipe(map(({ response }) => response?.networks));\n }\n\n public sessionTimeoutModal() {\n return this.modal.open('Votre session a expiré', 'Veuillez vous reconnecter', [\n { label: 'Se reconnecter', value: 'REDIRECT' },\n ]);\n }\n\n public async loginWithToken(token: string): Promise {\n this.token = token;\n\n const [session, networks] = await Promise.all([\n this.session(),\n firstValueFrom(this.getNetworks()),\n ]);\n\n // HACK: Get locales & configuration of network\n const userNetworks = session.user.networks.reduce((networkList, network) => {\n const networkWithLocale = networks.find(({ id }) => network.id === id);\n\n if (networkWithLocale) {\n const localeList = networkWithLocale.languages.map(({ locale }) => locale);\n\n networkList.push({\n locales: localeList,\n features: networkWithLocale.features,\n ...networkWithLocale.configuration,\n ...network,\n });\n }\n\n return networkList;\n }, []);\n\n this.user_id = session.user.id.toString();\n this.networks = userNetworks;\n this.currencies = userNetworks.reduce((currencies, network) => {\n if (currencies.includes(network.currency)) return currencies;\n\n return [...currencies, network.currency];\n }, []);\n\n const permissionsFromRole = session.user.roles.reduce((permissionList, role) => {\n const rolesPermissionLabelList = role.permissions.map(({ label }) => label);\n return [...permissionList, ...rolesPermissionLabelList];\n }, []);\n\n const standAlonePermissions = session.user.permissions.map(({ label }) => label);\n\n this.permissions = Array.from(new Set([...permissionsFromRole, ...standAlonePermissions]));\n this.username = session.user.login;\n this.roleNames = session.user?.roles?.map(({ name }) => name);\n }\n\n public hasFeature(handle: NetworkFeatureHandle): boolean {\n return this.networks.some((network) => {\n // If user is already login before this new feature, the network hasn't got the `features` attribute\n return (network.features ?? []).some((feature) => {\n return feature.handle === handle;\n });\n });\n }\n\n public static networkHasFeature(network: Network, handle: NetworkFeatureHandle): boolean {\n return network.features.some((feature) => {\n return feature.handle === handle;\n });\n }\n\n public getNetworkWithFeature(handle: NetworkFeatureHandle): Network[] {\n return this.networks.filter((network) => {\n return network.features.some((feature) => {\n return feature.handle === handle;\n });\n });\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { first } from 'rxjs/operators';\n\nconst DEFAULT_HASH_PLACEHOLDER = '{{ POST_BUILD_ENTERS_HASH_HERE }}';\n\n@Injectable()\nexport class CacheBustingService {\n // this will be replaced by actual hash post-build.js\n private currentHash: string = DEFAULT_HASH_PLACEHOLDER;\n private interval: NodeJS.Timeout;\n private retry = 0;\n\n constructor(private http: HttpClient) {}\n\n /**\n * Checks in every set frequency the version of frontend application\n * @param {number} frequency - in milliseconds, defaults to 30 minutes\n */\n public initVersionCheck(frequency: number = 1000 * 60 * 30): void {\n this.interval = setInterval(() => this.checkVersion(), frequency);\n }\n\n public clearVersionCheck(): void {\n if (typeof this.interval !== 'number') return;\n clearInterval(this.interval);\n }\n\n /**\n * Will do the call and check if the hash has changed or not\n */\n private checkVersion(): void {\n const timestamp = Date.now().toString();\n\n this.http\n .get<{ version: string; hash: string }>('/version.json', { params: { timestamp } })\n .pipe(first())\n .subscribe({\n next: (response) => {\n const hash = response.hash;\n const hashChanged = this.hasHashChanged(hash);\n\n // If new version, do something\n if (hashChanged) {\n const url = new URL(location.href);\n url.searchParams.append('cache', timestamp);\n location.href = url.toString();\n }\n\n // store the new hash so we wouldn't trigger versionChange again\n // only necessary in case you did not force refresh\n this.currentHash = hash;\n },\n error: (err) => {\n this.retry++;\n console.error(err, 'Could not get version');\n\n if (this.retry > 10) this.clearVersionCheck();\n },\n });\n }\n\n /**\n * Checks if hash has changed.\n * This file has the JS hash, if it is a different one than in the version.json\n * we are dealing with version change\n * @param {string} currentHash\n * @param {string} newHash\n * @returns {boolean}\n */\n private hasHashChanged(newHash: string): boolean {\n return this.currentHash === DEFAULT_HASH_PLACEHOLDER ? false : this.currentHash !== newHash;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { AuthService } from './auth.service';\n\nconst AVAILABLE_COLORS = [\n '#162636',\n '#222222',\n '#e5e5e5',\n '#0b6484',\n '#46403b',\n '#413256',\n '#b43d38',\n '#69a8bb',\n];\nconst LIGHT = '#ffffff';\nconst DARK = '#000000';\nconst USER_PREFERENCES = 'userPreferences';\n\n@Injectable()\nexport class ColorService {\n constructor(private auth: AuthService) {\n const userPreferencesJson = localStorage.getItem(USER_PREFERENCES);\n const userPreferences = JSON.parse(userPreferencesJson);\n\n if (userPreferences?.[this.user_id]) {\n this._sidebarColor = userPreferences[this.user_id].sidebarColor;\n }\n }\n\n public availableColors = AVAILABLE_COLORS;\n\n private _sidebarColor: string = AVAILABLE_COLORS[0];\n\n private user_id = this.auth.user_id;\n\n public get sidebarColor() {\n const text = this.isColorLight(this._sidebarColor) ? DARK : LIGHT;\n const selected = this.shadeColor(this._sidebarColor, 20);\n const hover = this.shadeColor(this._sidebarColor, 30);\n const dropdown = this.shadeColor(this._sidebarColor, 10);\n\n return { primary: this._sidebarColor, selected, hover, dropdown, text };\n }\n\n public setSidebarColor(color: string) {\n this._sidebarColor = color;\n\n const userPreferencesJson = localStorage.getItem(USER_PREFERENCES);\n const userPreferences = JSON.parse(userPreferencesJson);\n\n if (!userPreferences) {\n const updatedUserPreferences = { [this.user_id]: { sidebarColor: color } };\n\n return localStorage.setItem(USER_PREFERENCES, JSON.stringify(updatedUserPreferences));\n }\n\n const selectedUserPreferences = userPreferences[this.user_id];\n const updatedUserPreferences = {\n ...userPreferences,\n [this.user_id]: { ...selectedUserPreferences, sidebarColor: color },\n };\n\n localStorage.setItem(USER_PREFERENCES, JSON.stringify(updatedUserPreferences));\n }\n //https://stackoverflow.com/questions/12043187/how-to-check-if-hex-color-is-too-black\n private isColorLight(color: string) {\n const c = color.substring(1); // strip #\n const rgb = parseInt(c, 16); // convert rrggbb to decimal\n\n const r = (rgb >> 16) & 0xff; // extract red\n const g = (rgb >> 8) & 0xff; // extract green\n const b = (rgb >> 0) & 0xff; // extract blue\n\n const luma = 0.2126 * r + 0.7152 * g + 0.0722 * b; // per ITU-R BT.709\n\n //https://www.w3.org/TR/WCAG21/\n const contrast = (0.8 * (luma / 255 + 0.05)) / 0.05;\n\n //WCAG20 1.4.6\n return contrast > 7;\n }\n\n //https://stackoverflow.com/questions/5560248/programmatically-lighten-or-darken-a-hex-color-or-rgb-and-blend-colors\n private shadeColor(color: string, percent: number) {\n let R = parseInt(color.substring(1, 3), 16);\n let G = parseInt(color.substring(3, 5), 16);\n let B = parseInt(color.substring(5, 7), 16);\n\n const ratio = (percent * 255) / 100;\n\n R = Math.floor(this.isColorLight(color) ? R - ratio : R + ratio);\n G = Math.floor(this.isColorLight(color) ? G - ratio : G + ratio);\n B = Math.floor(this.isColorLight(color) ? B - ratio : B + ratio);\n\n R = R < 255 ? R : 255;\n G = G < 255 ? G : 255;\n B = B < 255 ? B : 255;\n\n R = R > 0 ? R : 0;\n G = G > 0 ? G : 0;\n B = B > 0 ? B : 0;\n\n const RR = R.toString(16).length == 1 ? '0' + R.toString(16) : R.toString(16);\n const GG = G.toString(16).length == 1 ? '0' + G.toString(16) : G.toString(16);\n const BB = B.toString(16).length == 1 ? '0' + B.toString(16) : B.toString(16);\n\n return '#' + RR + GG + BB;\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Subscription } from 'rxjs';\n\n@Injectable()\nexport class ExportService {\n private stream: Subscription;\n private data: any[];\n private fields: { label: string; key: string; modifier?: Function }[];\n private name: string;\n\n constructor() {}\n\n public export(\n data: any[],\n fields: { label: string; key: string; modifier?: Function }[],\n filename: string\n ): void {\n const file = filename || 'export';\n\n const csvData = '\\ufeff' + this.generateCsv(data, fields);\n const blob = new Blob([csvData], { type: 'text/csv' });\n const fileName = `${file}.csv`;\n\n const a = document.createElement('a'),\n url = window.URL.createObjectURL(blob);\n\n a.setAttribute('style', 'display: none;');\n document.body.appendChild(a);\n a.href = url;\n a.download = fileName;\n a.click();\n }\n\n public generateCsv(\n datas: any[],\n fields: { label: string; key: string; modifier?: Function }[]\n ): string {\n const rows = [];\n let cols = [];\n\n fields.forEach((field) => {\n cols.push(`\"${field.label}\"`);\n });\n rows.push(cols.join(';'));\n\n datas.forEach((data) => {\n cols = [];\n fields.forEach((field) => {\n let val = '';\n\n try {\n val = field.key.split('.').reduce((a, c) => a[c], data);\n if (val === null) val = '';\n } catch (e) {\n if (!(e instanceof TypeError)) throw e;\n }\n\n if (field.modifier) val = field.modifier(val);\n cols.push(`\"${val}\"`);\n });\n rows.push(cols.join(';'));\n });\n\n return rows.join('\\r\\n');\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { environment as env } from '@env/environment';\nimport { of } from 'rxjs';\nimport { map, switchMap, tap } from 'rxjs/operators';\nimport { AuthService } from './auth.service';\n\n@Injectable()\nexport class FeedService {\n private endpoint = env.config.feedRoot;\n\n constructor(private http: HttpClient, private authService: AuthService, private router: Router) {}\n\n public get(url: string) {\n return of([null]).pipe(\n switchMap(() => this.http.get(this.endpoint + url)),\n\n tap((r) => this.checkAuthorized(r)),\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public post(url: string, data: any) {\n return this.http.post(this.endpoint + url, data).pipe(\n tap((r) => this.checkAuthorized(r)),\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public delete(url: string) {\n return this.http.delete(this.endpoint + url).pipe(tap((r) => this.checkAuthorized(r)));\n }\n\n public checkAuthorized(r) {\n if (r.errorCode) {\n this.logoutUser();\n throw new Error(r.errorCode);\n }\n if (r.errorMessage) {\n console.error(r.errorMessage);\n this.logoutUser();\n }\n }\n\n public logoutUser() {\n this.authService.logout();\n this.router.navigate(['/login']);\n }\n}\n","import { Injectable } from '@angular/core';\nimport { DateFormatPipe } from 'ngx-moment';\nimport { QueryParamsService } from './queryParams.service';\nimport { NavigationStart, Router } from '@angular/router';\nimport { getDefaultFrom, getDefaultTo } from '../utils/filter';\n\n@Injectable()\nexport class FilterService {\n private _filters: {\n network: string;\n networkName: string;\n from: Date;\n to: Date;\n query: string;\n onlyActive: boolean;\n };\n\n public set filters(updatedFilters) {\n let from = null;\n let to = null;\n\n if (updatedFilters.from) {\n from = new Date(updatedFilters.from);\n from.setHours(0);\n from.setMinutes(0);\n from.setSeconds(0);\n }\n if (updatedFilters.to) {\n to = new Date(updatedFilters.to);\n to.setHours(23);\n to.setMinutes(59);\n to.setSeconds(59);\n }\n\n this._filters = { ...updatedFilters, from, to };\n // TODO : remove legacy filters\n this.query = updatedFilters.query;\n this.networkName = updatedFilters.networkName;\n\n this.queryParamsService.globalFilterParams = this.filters;\n }\n\n public get filters() {\n let from: string = null;\n let to: string = null;\n\n if (this._filters.from) from = this._filters.from.toISOString();\n if (this._filters.to) to = this._filters.to.toISOString();\n\n return { ...this._filters, from, to };\n }\n\n public get filtersWithID() {\n const filters = this.filters;\n\n return {\n from: filters.from,\n to: filters.to,\n query: filters.query,\n network_id: filters.network,\n };\n }\n\n public set network(value: string) {\n this._filters.network = value;\n\n this.queryParamsService.globalFilterParams = this.filters;\n }\n\n public get network() {\n return this._filters.network;\n }\n public networkName: string = null;\n\n public query: string = null;\n\n set from(date: Date) {\n date.setHours(0);\n date.setMinutes(0);\n date.setSeconds(0);\n\n this._filters.from = date;\n this.queryParamsService.globalFilterParams = this.filters;\n }\n get from(): Date {\n return this._filters.from;\n }\n\n set to(date: Date) {\n date.setHours(23);\n date.setMinutes(59);\n date.setSeconds(59);\n\n this._filters.to = date;\n this.queryParamsService.globalFilterParams = this.filters;\n }\n\n get to(): Date {\n return this._filters.to;\n }\n\n set from_string(s: string) {\n this._filters.from = new Date(s);\n this.queryParamsService.globalFilterParams = this.filters;\n }\n get from_string(): string {\n return this._filters.from.toISOString();\n }\n\n set to_string(s: string) {\n this._filters.to = new Date(s);\n this.queryParamsService.globalFilterParams = this.filters;\n }\n get to_string(): string {\n return this._filters.to.toISOString();\n }\n\n set onlyActive(active: boolean) {\n this._filters.onlyActive = active || null;\n this.queryParamsService.globalFilterParams = this.filters;\n }\n get onlyActive(): boolean {\n return this._filters.onlyActive;\n }\n\n reset() {\n this.filters = {\n from: new Date().toISOString(),\n to: new Date().toISOString(),\n network: null,\n networkName: null,\n query: null,\n onlyActive: true,\n };\n }\n\n private initFilters() {\n const params = this.queryParamsService.globalFilterParams;\n\n const from = params.from || getDefaultFrom().toISOString();\n const to = params.to || getDefaultTo().toISOString();\n const network = params.network || null;\n const networkName = params.networkName || null;\n const query = params.query || null;\n const onlyActive = params.onlyActive;\n\n this.filters = { from, to, network, networkName, query, onlyActive };\n }\n\n constructor(\n public dateFormatPipe: DateFormatPipe,\n private queryParamsService: QueryParamsService,\n router: Router\n ) {\n router.events.subscribe((event) => {\n if (event instanceof NavigationStart && !event.url.includes('/dashboard-ng')) {\n this.initFilters();\n }\n });\n\n this.initFilters();\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { environment as env } from '@env/environment';\nimport { Observable } from 'rxjs';\nimport { map } from 'rxjs/operators';\nimport { AuthService } from './auth.service';\n\n@Injectable()\nexport class LinesService {\n constructor(private http: HttpClient, private authService: AuthService, private router: Router) {}\n\n public getLines(): Observable {\n return this.http.get(env.config.feedRoot + 'Lines/getLines.json').pipe(\n map(({ response: r }) => {\n return r.lines as any[];\n })\n );\n }\n\n public getRoutes(nid: string, agency_key: string): Observable {\n return this.http\n .get(\n env.config.feedRoot + `Lines/getRoutes.json?agency_key=${agency_key}&network_id=${nid}`\n )\n .pipe(\n map(({ response: r }) => {\n return r.routes as any[];\n })\n );\n }\n}\n","import { Injectable } from '@angular/core';\nimport { InputOptions, ModalComponent, Option } from '../components/modal/modal.component';\n\n@Injectable()\nexport class ModalService {\n private modalComponent: ModalComponent;\n\n public registerComponent(modalComponent: ModalComponent) {\n this.modalComponent = modalComponent;\n }\n\n public open(title: string, content: string, options: Option[]): Promise {\n return this.modalComponent.open(title, content, options);\n }\n\n public prompt(\n title: string,\n content: string,\n input: InputOptions,\n options: Option[]\n ): Promise<{ choice: string; value: string }> {\n return this.modalComponent.prompt(title, content, input, options);\n }\n}\n","import { Injectable } from '@angular/core';\n\n@Injectable()\nexport class PaginationService {\n private _selectedPage: { [route: string]: number } = {};\n\n setSelectedPage(selectedPage: number, id = location.pathname) {\n this._selectedPage[id] = selectedPage;\n }\n getSelectedPage(id = location.pathname): number {\n return this._selectedPage[id] || 1;\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { environment as env } from '@env/environment';\nimport { map } from 'rxjs/operators';\nimport { AuthService } from './auth.service';\n\n@Injectable()\nexport class PermissionsService {\n constructor(private http: HttpClient, private authService: AuthService, private router: Router) {}\n\n public getRoles() {\n return this.http.get(env.config.feedRoot + 'Permissions/getRoles.json').pipe(\n map(({ response: r }) => {\n return r.roles;\n })\n );\n }\n}\n","import { Injectable } from '@angular/core';\nimport { ActivatedRoute } from '@angular/router';\nimport { PaginationConfig } from '../models/server-pagination';\n\ninterface GlobalFilterParams {\n from: string;\n to: string;\n network: string;\n networkName: string;\n query: string;\n onlyActive: boolean;\n}\ninterface QueryParams extends GlobalFilterParams, PaginationConfig, Record {}\n\n@Injectable()\nexport class QueryParamsService {\n public queryParams: Partial;\n\n public set globalFilterParams(params: Partial) {\n this.queryParams = { ...this.queryParams, ...params };\n\n this.setQueryParams(params);\n }\n\n public get globalFilterParams() {\n const { from, to, network, networkName, query } = this.queryParams;\n\n return { from, to, network, networkName, query };\n }\n\n public set localFilterParams(params: Record) {\n this.queryParams = { ...this.globalFilterParams, ...this.paginationParams, ...params };\n\n this.setQueryParams(params);\n }\n\n public get localFilterParams() {\n //TODO : specify on local filter\n return this.queryParams;\n }\n\n public set paginationParams(params: Partial) {\n this.queryParams = { ...this.queryParams, ...params };\n\n this.setQueryParams(params);\n }\n\n public get paginationParams() {\n const { offset, limit } = this.queryParams;\n return { offset, limit };\n }\n\n constructor(private route: ActivatedRoute) {\n const url = new URL(location.href);\n this.queryParams = Object.fromEntries(url.searchParams.entries());\n }\n\n public getInitialQueryParams = this.route.queryParams;\n\n public setQueryParams(params: Record) {\n const url = new URL(location.href);\n const acceptedValues = [0, false];\n\n for (const [key, value] of Object.entries(params)) {\n if (value || acceptedValues.includes(value)) {\n url.searchParams.set(key, value);\n } else {\n url.searchParams.delete(key);\n }\n }\n\n history.replaceState(null, '', url.toString());\n }\n}\n","import { Injectable } from '@angular/core';\nimport { environment as env } from '@env/environment';\nimport { ServerConfig } from '../models/server-pagination';\nimport { AuthService } from './auth.service';\n\n@Injectable()\nexport class ServerPaginationService {\n constructor(private authService: AuthService) {}\n\n public get fetchConfig(): RequestInit {\n return {\n method: 'GET',\n headers: {\n Authorization: `Bearer ${this.authService.token}`,\n },\n };\n }\n\n public serverConfig(path: string, data?: string) {\n const config: ServerConfig = {\n url: env.config.feedRoot + path,\n data: (res) => (data ? res.response?.[data] ?? [] : res.response),\n sortableKeys: (res) => res.headers.get('X-Airweb-Sortable-Keys')?.split(',') ?? [],\n limit: (res) => +res.headers.get('X-Airweb-Pagination-Limit'),\n count: (res) => +res.headers.get('X-Airweb-Pagination-Count'),\n offset: (res) => +res.headers.get('X-Airweb-Pagination-Offset'),\n pages: (res) => +res.headers.get('X-Airweb-Pagination-Pages'),\n };\n return config;\n }\n}\n","import { Injectable } from '@angular/core';\n\n@Injectable()\nexport class ToDateStringService {\n constructor() {}\n\n public toDateString(date: Date): string {\n let day = `0${date.getDate()}`.slice(-2),\n month = `0${date.getMonth() + 1}`.slice(-2),\n year = date.getFullYear().toString();\n\n return `${year}-${month}-${day}`;\n }\n}\n","import { map } from 'rxjs/operators';\nimport { Injectable } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { TranslateService } from '@ngx-translate/core';\n\nimport { environment as env } from '@env/environment';\nimport { GenericResponse } from '../models/generic-response';\n\nexport interface ILanguage {\n id: string;\n locale: string;\n name: string;\n flag: string;\n}\n\n@Injectable()\nexport class TranslationService {\n public translations: ILanguage[] = [];\n public currentLang: string = this.translateService.getDefaultLang();\n\n constructor(private http: HttpClient, private translateService: TranslateService) {}\n\n public getTranslations() {\n this.translateService.onLangChange.subscribe((currentLanguage) => {\n this.currentLang = currentLanguage.lang;\n });\n\n return this.http\n .get>(env.config.feedRoot + `Translations/getLanguages`, {\n params: { lang: this.currentLang },\n })\n .pipe(\n map(({ response }) => {\n this.translations = response;\n return response;\n })\n );\n }\n}\n","export interface Option {\n timeout?: number;\n punishTime?: number;\n}\n\nexport interface Filter {\n channel: string;\n displayed: boolean;\n}\n\nexport interface Source {\n url: string;\n index: string;\n punishedUntil: Date | null;\n cachedUntil: Date | null;\n}\n\nexport interface FetchResponse {\n isCached: boolean;\n time: number;\n payload: MaintenanceStatus[];\n source: Source;\n}\n\nexport interface MaintenanceStatus {\n level: Level;\n channel: string;\n displayAfter?: string; // Min Date\n displayBefore?: string; // Max Date\n flags?: string[];\n}\n\nexport enum Level {\n CRITICAL = 'CRITICAL',\n WARNING = 'WARNING',\n NOTICE = 'NOTICE',\n}\n","import { Injectable } from '@angular/core';\nimport { MaintenanceStatus, Source, Option, FetchResponse, Filter, Level } from '../models/upKeep';\nimport { environment as env } from '@env/environment';\n\n@Injectable({\n providedIn: 'root',\n})\nexport class UpKeepService {\n private _sources: Source[];\n\n public options: Option = {\n timeout: 1000,\n punishTime: 60000,\n };\n\n constructor() {\n this._sources = UpKeepService.prepareSources(env.config.maintenanceUrlList);\n }\n\n public async getMaintenanceAlert(): Promise {\n const response = await this.fetch({\n channel: 'bo/global',\n displayed: true,\n });\n\n if (response) {\n const alert = response.payload\n .filter((maintenanceAlert) => {\n // Fitre only on alerts whose date range is correct\n\n let hasDisplayRangeTime = false;\n const now = new Date();\n\n if (!maintenanceAlert.displayBefore && !maintenanceAlert.displayAfter) {\n hasDisplayRangeTime = true;\n } else if (maintenanceAlert.displayAfter && !maintenanceAlert.displayBefore) {\n const displayAfterDate = new Date(maintenanceAlert.displayAfter);\n\n hasDisplayRangeTime = displayAfterDate >= now;\n } else if (maintenanceAlert.displayBefore && maintenanceAlert.displayAfter) {\n const displayAfterDate = new Date(maintenanceAlert.displayAfter);\n const displayBeforeDate = new Date(maintenanceAlert.displayBefore);\n\n hasDisplayRangeTime = now >= displayAfterDate && now <= displayBeforeDate;\n }\n\n return maintenanceAlert.level === Level.CRITICAL && hasDisplayRangeTime;\n })\n .sort((prevAlert, nextAlert) => {\n // Sort of the end of maintenance on the nearest date\n\n if (!nextAlert.displayBefore) {\n return 1;\n }\n\n if (!prevAlert.displayBefore) {\n return -1;\n }\n\n if (prevAlert.displayBefore && nextAlert.displayBefore) {\n const prevDisplayBefore = new Date(prevAlert.displayBefore);\n const nextDisplayBefore = new Date(nextAlert.displayBefore);\n\n if (nextDisplayBefore < prevDisplayBefore) {\n return -1;\n }\n }\n\n return 0;\n })\n .shift();\n\n return alert ?? null;\n }\n\n return null;\n }\n\n private async fetch(filters: Filter): Promise {\n const startDate = new Date();\n const startTime = startDate.getTime();\n\n for (const source of this._sources) {\n if (source.punishedUntil && source.punishedUntil > startDate) {\n continue;\n }\n\n try {\n const sourceRequestUrl = new URL(source.url);\n\n for (const [key, filter] of Object.entries(filters)) {\n sourceRequestUrl.searchParams.append(key, filter);\n }\n\n const sourceRequestUrlString = sourceRequestUrl.toString();\n\n const response = await UpKeepService.manageFetch(sourceRequestUrlString, {\n timeout: this.options.timeout ?? 1000,\n });\n\n const isCached = Boolean(source.cachedUntil && source.cachedUntil > startDate);\n\n if (!isCached) {\n const headerCacheControlMaxAge =\n UpKeepService.maybeExtractCacheControlMaxAgeFrom(response);\n\n if (headerCacheControlMaxAge !== null) {\n const cachedUntil = UpKeepService.makeDatePlusMilliseconds(\n headerCacheControlMaxAge * 1000\n );\n source.cachedUntil = cachedUntil;\n }\n }\n\n const responseBody: MaintenanceStatus[] = await response.json();\n\n const successTime = Date.now();\n\n return {\n isCached,\n time: successTime - startTime,\n payload: responseBody,\n source,\n };\n } catch (e) {\n const punishTime = this.options.punishTime ?? 10000;\n const punishedUntil = UpKeepService.makeDatePlusMilliseconds(punishTime);\n\n const sourceIndex = this.getSourceArrayIndex(source.index);\n\n if (sourceIndex > -1) {\n this._sources[sourceIndex].punishedUntil = punishedUntil;\n }\n\n continue;\n }\n }\n\n return null;\n }\n\n private getSourceArrayIndex(index: string): number {\n return this._sources.findIndex((source) => source.index === index);\n }\n\n private static maybeExtractCacheControlMaxAgeFrom(response: Response): number | null {\n const headerCacheControl = response.headers.get('cache-control');\n const headerCacheControlMaxAgeMatch = /max-age=(?\\d+)/gi.exec(\n headerCacheControl ?? ''\n );\n\n if (!headerCacheControlMaxAgeMatch) return null;\n\n return headerCacheControlMaxAgeMatch.groups?.seconds\n ? +headerCacheControlMaxAgeMatch.groups?.seconds\n : null;\n }\n\n private static prepareSources(urls: string[]): Source[] {\n const sources: Source[] = [];\n\n for (const index in urls) {\n const url = urls[index];\n sources.push({\n url,\n index,\n punishedUntil: null,\n cachedUntil: null,\n });\n }\n\n return sources;\n }\n\n private static async manageFetch(resource: string, options: Option = {}): Promise {\n return UpKeepService.fetchWithTimeout(resource, options);\n }\n\n private static async fetchWithTimeout(resource: string, options: Option = {}): Promise {\n const { timeout = 8000 } = options;\n\n const controller = new AbortController();\n const id = setTimeout(() => controller.abort(), timeout);\n\n const response = await fetch(resource, {\n ...options,\n signal: controller.signal,\n });\n clearTimeout(id);\n\n return response;\n }\n\n private static fetchWithoutTimeout(resource: string): Promise {\n return fetch(resource);\n }\n\n private static makeDatePlusMilliseconds(milliseconds = 0, date = new Date()) {\n date.setMilliseconds(date.getMilliseconds() + milliseconds);\n\n return date;\n }\n}\n","import { UppyAngularDashboardModule } from '@uppy/angular';\nimport { CommonModule } from '@angular/common';\nimport { ModuleWithProviders, NgModule } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatAutocompleteModule } from '@angular/material/autocomplete';\nimport { MatDialogModule } from '@angular/material/dialog';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { RouterModule } from '@angular/router';\nimport { NgbModule } from '@ng-bootstrap/ng-bootstrap';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { SimpleNotificationsModule } from 'angular2-notifications';\nimport { DateFormatPipe, MomentModule } from 'ngx-moment';\nimport { NgxPaginationModule } from 'ngx-pagination';\nimport { CustomersService } from '../customers/services/customers.service';\nimport { CustomersFiltersService } from '../customers/services/filters.service';\nimport { BreadcrumbComponent } from './components/breadcrumb/breadcrumb.component';\nimport { CardComponent } from './components/card/card.component';\nimport { CommentsComponent } from './components/comments/comments.component';\nimport { ClosableCardComponent } from './components/card/closable-card.component';\nimport { FileInputComponent } from './components/file-input/file-input.component';\n// Components\nimport { FilterComponent } from './components/filter/filter.component';\nimport { HeroIconComponent } from './components/hero-icon/hero-icon.component';\nimport { ImageCropperComponent } from './components/image-cropper/image-cropper.component';\nimport { LoaderComponent } from './components/loader/loader.component';\nimport { ModalComponent } from './components/modal/modal.component';\nimport { PaginationComponent } from './components/pagination/pagination.component';\nimport { ScheduleComponent } from './components/schedule/schedule.component';\nimport { Schedulev2Component } from './components/schedulev2/schedulev2.component';\nimport { SearchCustomerModalComponent } from './components/search-customer-modal/search-customer-modal.component';\nimport { TableServerComponent } from './components/table-server/table-server.component';\nimport { TableComponent } from './components/table/table.component';\nimport { TabComponent } from './components/tabs/tab.component';\nimport { TabsComponent } from './components/tabs/tabs.component';\nimport { TooltipComponent } from './components/tooltip/tooltip.component';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MultiSelectComponent } from './components/multi-select/multi-select.component';\nimport { DurationComponent } from './components/duration/duration.component';\nimport { JsonEditModalComponent } from './components/json-edit-modal/json-edit-modal.component';\n// Directives\nimport { DefaultImageDirective } from './directives/default-img.directive';\nimport { HasPermissionDirective } from './directives/has-one-permission.directive';\nimport { NAV_DROPDOWN_DIRECTIVES } from './directives/nav-dropdown.directive';\nimport { ShowPasswordDirective } from './directives/show-password.directive';\nimport { HasFeatureDirective } from './directives/has-feature.directive';\n// Guards\nimport { AuthGuard } from './guards/auth.guard';\nimport { CanDeactivateGuard } from './guards/can-deactivate.guard';\nimport { HomeGuard } from './guards/home.guard';\nimport { OneNetworkGuard } from './guards/one-network.guard';\nimport { PermissionGuard } from './guards/permission.guard';\nimport { FeatureGuard } from './guards/feature.guard';\n// Pipes\nimport { HighlightPipe } from './pipes/highlight.pipe';\nimport { IntervalPipe } from './pipes/interval.pipe';\nimport { KeyPipe } from './pipes/key.pipe';\nimport { OrderByPipe } from './pipes/order-by.pipe';\nimport { SearchPipe } from './pipes/search.pipe';\nimport { ValuesPipe } from './pipes/values.pipe';\nimport { CurrencyPipe } from './pipes/currency.pipe';\nimport { SafeHtmlPipe } from './pipes/safeHtml.pipe';\n// Resolvers\nimport { EmailsNetworkResolver } from './resolvers/emails.resolvers';\nimport { LinesResolver } from './resolvers/lines.resolver';\nimport { RolesResolver } from './resolvers/roles.resolver';\nimport { TagsMailNetworkResolver } from './resolvers/tags-mail.resolver';\n// Services\nimport { CacheBustingService } from './services/cache-busting.service';\nimport { ColorService } from './services/color.service';\nimport { ExportService } from './services/export.service';\nimport { FeedService } from './services/feed.service';\nimport { FilterService } from './services/filter.service';\nimport { LinesService } from './services/lines.service';\nimport { ModalService } from './services/modal.service';\nimport { PaginationService } from './services/pagination.service';\nimport { PermissionsService } from './services/permissions.service';\nimport { QueryParamsService } from './services/queryParams.service';\nimport { ServerPaginationService } from './services/server-pagination.service';\nimport { ToDateStringService } from './services/to-date-string.service';\nimport { TranslationService } from './services/translation.service';\nimport { UpKeepService } from './services/up-keep.service';\nimport { UpdateDocumentModalComponent } from './components/update-document-modal/update-document-modal.component';\nimport { ConfigGuard } from './guards/config.guard';\nimport { CommentsNgComponent } from './components/comments-ng/comments-ng.component';\nimport { FormatComponent } from './components/comments-ng/format/format.component';\nimport { DeleteModalComponent } from './components/comments-ng/delete-modal/delete-modal.component';\nimport { NgJsonEditorModule } from 'ang-jsoneditor';\n\n@NgModule({\n declarations: [\n BreadcrumbComponent,\n FileInputComponent,\n FilterComponent,\n ModalComponent,\n LoaderComponent,\n PaginationComponent,\n ScheduleComponent,\n Schedulev2Component,\n TableComponent,\n TableServerComponent,\n TabComponent,\n TabsComponent,\n TooltipComponent,\n CardComponent,\n ClosableCardComponent,\n HeroIconComponent,\n ImageCropperComponent,\n SearchCustomerModalComponent,\n CommentsComponent,\n UpdateDocumentModalComponent,\n MultiSelectComponent,\n CommentsNgComponent,\n DurationComponent,\n JsonEditModalComponent,\n\n DefaultImageDirective,\n HasPermissionDirective,\n NAV_DROPDOWN_DIRECTIVES,\n ShowPasswordDirective,\n HasFeatureDirective,\n\n HighlightPipe,\n IntervalPipe,\n KeyPipe,\n OrderByPipe,\n SearchPipe,\n ValuesPipe,\n CurrencyPipe,\n SafeHtmlPipe,\n FormatComponent,\n DeleteModalComponent,\n ],\n imports: [\n CommonModule,\n\n FormsModule,\n ReactiveFormsModule,\n RouterModule,\n MatDialogModule,\n NgxPaginationModule,\n MomentModule,\n NgbModule,\n MatAutocompleteModule,\n MatSelectModule,\n MatMenuModule,\n SimpleNotificationsModule.forRoot(),\n TranslateModule.forChild(),\n UppyAngularDashboardModule,\n NgJsonEditorModule,\n ],\n providers: [\n PaginationService,\n ToDateStringService,\n PermissionsService,\n LinesService,\n FeedService,\n ExportService,\n TranslationService,\n MomentModule,\n ServerPaginationService,\n UpKeepService,\n MatSelectModule,\n\n AuthGuard,\n ConfigGuard,\n CanDeactivateGuard,\n HomeGuard,\n OneNetworkGuard,\n PermissionGuard,\n FeatureGuard,\n\n LinesResolver,\n RolesResolver,\n TagsMailNetworkResolver,\n EmailsNetworkResolver,\n\n DateFormatPipe,\n CustomersService,\n CustomersFiltersService,\n ],\n exports: [\n BreadcrumbComponent,\n FileInputComponent,\n FilterComponent,\n ModalComponent,\n LoaderComponent,\n PaginationComponent,\n ScheduleComponent,\n Schedulev2Component,\n TableComponent,\n TableServerComponent,\n TabComponent,\n TabsComponent,\n TooltipComponent,\n CardComponent,\n ClosableCardComponent,\n HeroIconComponent,\n ImageCropperComponent,\n SearchCustomerModalComponent,\n CommentsComponent,\n UpdateDocumentModalComponent,\n MultiSelectComponent,\n CommentsNgComponent,\n DurationComponent,\n JsonEditModalComponent,\n\n DefaultImageDirective,\n HasPermissionDirective,\n NAV_DROPDOWN_DIRECTIVES,\n ShowPasswordDirective,\n HasFeatureDirective,\n\n HighlightPipe,\n IntervalPipe,\n KeyPipe,\n OrderByPipe,\n SearchPipe,\n ValuesPipe,\n CurrencyPipe,\n\n TranslateModule,\n ],\n})\nexport class SharedModule {\n static forRoot(): ModuleWithProviders {\n return {\n ngModule: SharedModule,\n providers: [\n ModalService,\n FilterService,\n CacheBustingService,\n ColorService,\n QueryParamsService,\n MatDialogModule,\n ],\n };\n }\n}\n","export function getDefaultFrom(): Date {\n const beforeYesterday = new Date();\n beforeYesterday.setHours(0);\n beforeYesterday.setMinutes(0);\n beforeYesterday.setSeconds(0);\n beforeYesterday.setHours(-24);\n\n return beforeYesterday;\n}\n\nexport function getDefaultTo(): Date {\n const to = new Date();\n to.setHours(23);\n to.setMinutes(59);\n to.setSeconds(59);\n\n return to;\n}\n","import { BehaviorSubject, Observable } from 'rxjs';\n\n/**\n * Creates an Observable for a specific property of an object.\n * https://craftsmen.nl/angular-lifehack-reactive-component-input-properties/\n *\n * @template T The type of the target object.\n * @template K The type of the key of the target object.\n *\n * @param {T} target The target object.\n * @param {K} key The key of the property to observe.\n *\n * @returns {Observable} An Observable that emits the new value whenever the property changes.\n */\nexport function observeProperty(target: T, key: K): Observable {\n // Create a new BehaviorSubject that will emit the current value of the property\n const subject = new BehaviorSubject(target[key]);\n\n // Redefine the property with a getter and setter\n Object.defineProperty(target, key, {\n // The getter simply returns the current value from the BehaviorSubject\n get(): T[K] {\n return subject.getValue();\n },\n // The setter checks if the new value is different from the current value\n // If it is, it updates the BehaviorSubject with the new value\n set(newValue: T[K]): void {\n if (newValue !== subject.getValue()) {\n subject.next(newValue);\n }\n },\n });\n\n // Return the BehaviorSubject as an Observable\n return subject.asObservable();\n}\n","import { Filters } from '@app/modules/shared/models/filters';\n\nexport const renewalIntervalOptions = [\n { id: 'P7D', label: 'pages.product_edit.renewal_p7d' }, //\n { id: 'P14D', label: 'pages.product_edit.renewal_p14d' },\n { id: 'P1M', label: 'pages.product_edit.renewal_p1m' },\n { id: 'P6M', label: 'pages.product_edit.renewal_p6m' },\n { id: 'P1Y', label: 'pages.product_edit.renewal_p1y' },\n];\n\nexport const saleChannelTypes = [\n { id: 'AIRWEB_TICKET_APP', label: 'pages.product_edit.sale_channel_airweb_app' },\n { id: 'AIRWEB_TICKET_ESHOP', label: 'pages.product_edit.sale_channel_airweb_eshop' },\n { id: 'AIRWEB_TIXIPASS', label: 'Tixipass' },\n { id: 'EXTERNAL', label: 'pages.product_edit.sale_channel_external' },\n];\n\nexport interface ProductItem {\n id: string;\n external_code?: string;\n categorie_id: string;\n renewal_fallback_product_id?: string | number;\n renewal_interval?: string | number;\n network_id: string;\n clients: any[];\n name: string;\n kind: ProductKind | null;\n description: string;\n data: string;\n active: number | number | boolean;\n image: string;\n price_vat: string;\n vat: string;\n start_date?: string;\n end_date?: string;\n period_type?: string;\n period_value?: number;\n pivot_type: string;\n pivot_value: string;\n start_proposals: number;\n sale_start_date?: string;\n sale_start_time?: string;\n sale_end_date?: string;\n sale_end_time?: string;\n supports: string[];\n type: string | number;\n sale_channels: string[];\n qty: string | number;\n product_id: string;\n universal: number | string | boolean;\n is_od: number | string | boolean;\n journey_matrix_id?: number;\n journey_matrix_layer_key?: string;\n to_process: boolean;\n notify_network: boolean;\n payment_preauthorization: boolean;\n photo_mandatory: boolean;\n birthday_mandatory: boolean;\n address_mandatory: boolean;\n documents: any[];\n timeslots?: any;\n tags?: any;\n global_tags: string[];\n info_text?: string;\n translations: any[] | Record;\n installments: number[];\n payment_options: PaymentOption[];\n max_buy_at_once?: any;\n is_renewable: boolean;\n renewal_notice_period?: any;\n provider_id?: any;\n provider_mode?: any;\n provider_entity?: any;\n provider_quantity?: any;\n provider_supports?: any;\n is_guest_orderable: boolean | string;\n is_device_assignable: boolean;\n quantity_min?: number | null;\n quantity_max?: number | null;\n provider_entity_parameter_mapping?: ProviderEntityParameterMappingItem[] | any;\n}\n\nexport interface PaymentOption {\n installments: number;\n payment_methods: string[];\n allow_subscription: boolean;\n customize?: boolean;\n distribution?: number[] | null;\n}\n\nexport interface ProviderEntityParameterMappingItem {\n key: string;\n value: string[] | string;\n}\n\nexport const permissionLevel = {\n EDIT: 1,\n READONLY: 0,\n HIDDEN: -1,\n};\n\n// Permissions by fields for STORE_EDITORIAL / STORE_EDIT / STORE_MANAGE roles\nexport const fieldsPermissions = {\n // TEMP: Field mentionned in Jira\n network_id: permissionLevel.EDIT, //Réseau\n name: permissionLevel.EDIT, //Titre\n description: permissionLevel.EDIT, //Description\n image: permissionLevel.EDIT, //Image\n translations: permissionLevel.EDIT, //Traduction\n active: permissionLevel.EDIT, //Produit actif\n category_id: permissionLevel.EDIT, //Catégorie\n product_id: permissionLevel.EDIT, //Type de produit\n qty: permissionLevel.EDIT, //Quantité\n price_vat: permissionLevel.EDIT, //Prix & TVA\n installments: permissionLevel.EDIT, //Nombre de paiement\n sale_interval: permissionLevel.EDIT, //Interval de la vente (date et time start/end)\n documents: permissionLevel.EDIT, //Pièce justificatives\n to_process: permissionLevel.EDIT, //A traiter\n notify_network: permissionLevel.EDIT, //Notification email\n tags: permissionLevel.EDIT, //Tags\n clients: permissionLevel.EDIT, // Référencer dans les applications régions => add HIDDEN hasRestrictedAccess\n provider_id: permissionLevel.EDIT, //Produit interfacé/Prestataire externe lié au produit\n product_tags_title: permissionLevel.EDIT, //Surcouche titre\n info_text: permissionLevel.EDIT, //Texte informatif web\n timeslots: permissionLevel.EDIT, //Plage horaire de vente\n type: permissionLevel.EDIT, //Produit web et/ou mobile\n universal: permissionLevel.EDIT, //TixiPass\n is_od: permissionLevel.EDIT, //Origine / Destination\n photo_mandatory: permissionLevel.EDIT, // Photo oligatoire\n birthday_mandatory: permissionLevel.EDIT, // Date de naissance obligatoire\n address_mandatory: permissionLevel.EDIT, // Adresse obligatoire\n supports: permissionLevel.EDIT, // Types de support\n tacit: permissionLevel.EDIT, //Tacite reconduction\n //TEMP: Field not mentionned in Jira => fixed to HIDDEN by default\n max_buy_at_once: permissionLevel.EDIT, //Nombre maximal par commande\n validity_interval: permissionLevel.EDIT, //Interval de la validatité (date et time start/end)\n period_type: permissionLevel.EDIT, //Période de validité\n pivot_value: permissionLevel.EDIT, //Date pivot\n pivot_type: permissionLevel.EDIT, //Type de date pivot\n renewal_notice_period: permissionLevel.EDIT, //Délai de résiliation\n renewal_fallback_product_id: permissionLevel.EDIT, //Produit de repli\n renewal_interval: permissionLevel.EDIT, //Interval de renouvellement\n fields: permissionLevel.EDIT, //Champs complémentaire\n data: permissionLevel.EDIT, //Données complémentaires\n is_guest_orderable: permissionLevel.EDIT, //Données complémentaires\n is_device_assignable: permissionLevel.EDIT, //Données complémentaires\n external_code: permissionLevel.EDIT, //Code externe\n};\n\nexport interface ProductItemFilters extends Filters {\n id?: number;\n onlyActive?: boolean;\n tacit?: boolean;\n sale_channels?: string;\n supports?: string;\n}\n\n//PIVOT DATE INFOS\nexport enum PivotType {\n MONTH_PIVOT = 'MONTH_PIVOT',\n WEEK_PIVOT = 'WEEK_PIVOT',\n DAY_PIVOT = 'DAY_PIVOT',\n}\n\nexport enum PeriodType {\n DAY = 'DAY',\n WEEK = 'WEEK',\n MONTH = 'MONTH',\n MONTH_ROLLING = 'MONTH_ROLLING',\n YEAR = 'YEAR',\n YEAR_ROLLING = 'YEAR_ROLLING',\n}\n\nexport const PivotDatePeriod = [\n { value: '', label: 'pages.product_edit.pivot_date_none' },\n { value: 'YEAR', label: 'pages.product_edit.pivot_date_year' },\n { value: 'YEAR_ROLLING', label: 'pages.product_edit.pivot_date_year_rolling' },\n { value: 'MONTH', label: 'pages.product_edit.pivot_date_month' },\n { value: 'MONTH_ROLLING', label: 'pages.product_edit.pivot_date_month_rolling' },\n { value: 'WEEK', label: 'pages.product_edit.pivot_date_week' },\n { value: 'DAY', label: 'pages.product_edit.pivot_date_day' },\n];\n\nexport const PivotDateType = [\n { value: '', label: 'pages.product_edit.pivot_type_none' }, //\n { value: PivotType.MONTH_PIVOT, label: 'pages.product_edit.pivot_type_month' },\n { value: PivotType.WEEK_PIVOT, label: 'pages.product_edit.pivot_type_week' },\n { value: PivotType.DAY_PIVOT, label: 'pages.product_edit.pivot_type_day' },\n];\n\nexport const PivotWeekDay = [\n { value: 1, label: 'otherslabels.days_num_2' },\n { value: 2, label: 'otherslabels.days_num_3' },\n { value: 3, label: 'otherslabels.days_num_4' },\n { value: 4, label: 'otherslabels.days_num_5' },\n { value: 5, label: 'otherslabels.days_num_6' },\n { value: 6, label: 'otherslabels.days_num_7' },\n { value: 0, label: 'otherslabels.days_num_1' },\n];\n\nexport const PivotInputsMode = {\n DISABLED: 0,\n};\n\n//SUPPORTS\nexport const ProductSupportsLabels = [\n { value: 'DEMATERIALIZED', label: 'pages.product_edit.product_support_dematerialized' },\n { value: 'PHYSICAL', label: 'pages.product_edit.product_support_physical' },\n { value: 'MEDIA', label: 'pages.product_edit.product_support_media' },\n { value: 'COLLECT', label: 'pages.product_edit.product_support_collect' },\n { value: 'SHIPPING', label: 'pages.product_edit.product_support_shipping' },\n];\nexport type ProductSupports = 'DEMATERIALIZED' | 'PHYSICAL' | 'MEDIA' | 'COLLECT' | 'SHIPPING';\n\nexport enum ProductKind {\n RESERVATION = 'RESERVATION',\n}\n\nexport enum ProductSupport {\n DEMATERIALIZED = 'DEMATERIALIZED',\n PHYSICAL = 'PHYSICAL',\n MEDIA = 'MEDIA',\n COLLECT = 'COLLECT',\n SHIPPING = 'SHIPPING',\n}","import { Injectable } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { catchError, map, switchMap, tap } from 'rxjs/operators';\nimport { Observable, firstValueFrom, of, throwError } from 'rxjs';\n\nimport { Categorie } from '../models/categorie';\nimport { ProductType } from '../models/product-type';\nimport { environment as env } from '@env/environment';\nimport { CacheManager } from '@app/utils/cache-manager';\nimport { Blueprint, BlueprintDetails } from '@app/modules/shop/models/blueprint/blueprint';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\n\nimport { BlueprintRules, BlueprintSettings } from '../models/blueprint/blueprint';\nimport { PeriodType, PivotType, ProductItem, ProductItemFilters } from '../models/product-item';\nimport { PreviewName } from '../models/preview-name';\nimport { ImageType } from '../models/image';\nimport { OverrideRule, RulePriority } from '../models/overridePrice';\n\ninterface FileUpload {\n name: string;\n path: string;\n url: string;\n}\n\n@Injectable()\nexport class ShopService {\n private cache = new CacheManager('shop');\n\n constructor(private http: HttpClient, private filters: FilterService) {}\n\n public getCategories(params: Partial<{ network_id: string; parent_id?: number }> = {}) {\n return this.http\n .get>(\n env.config.feedRoot + `Shop/getCategories`,\n {\n params: {\n ...params,\n v: this.cache.getVersion(),\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.categories;\n })\n );\n }\n\n public getCategorieDetails(id: number) {\n return this.http\n .get>(\n env.config.feedRoot + `Shop/getCategories`,\n {\n params: {\n id,\n v: this.cache.getVersion(),\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.categories[0];\n })\n );\n }\n\n public saveCategorie(categorie: Categorie) {\n const { cname, network_id, cid, active, translations, parent_id, index } = categorie;\n\n return this.http\n .post>(env.config.feedRoot + `Shop/saveCategorie`, {\n cname,\n network_id,\n cid,\n active,\n translations,\n parent_id,\n index,\n })\n .pipe(\n map(({ response }) => {\n this.cache.incrementVersion();\n return response.categorie as Categorie;\n })\n );\n }\n\n public saveProductType(productType: ProductType): any {\n return this.http.post(env.config.feedRoot + `Shop/saveProductType`, productType).pipe(\n map(({ response: r }) => {\n this.cache.incrementVersion();\n return r as { product_id: string };\n })\n );\n }\n\n public getProductsType(params: Partial<{ id: string; network_id: string }> = {}) {\n let version = this.cache.getVersion();\n const url = params.id ? `Shop/productTypeDetails` : 'Shop/getProductsType';\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + url, {\n params: {\n ...params,\n v: version,\n },\n })\n ),\n\n map(({ response }) => {\n return params.id ? (response.productType as ProductType) : response.productsType;\n })\n );\n }\n\n public getBlueprint(params: Partial<{ id: string }> = {}) {\n return firstValueFrom(\n this.http\n .get(env.config.feedRoot + `Wallet/blueprints`, {\n params: {\n id: `[\"${params.id}\"]`,\n },\n })\n .pipe(\n catchError(() => {\n console.error('Erreur sur le flux Wallet/blueprints');\n return of({\n status: 'ko',\n response: {\n errorMessage: 'Erreur sur le flux Wallet/blueprints',\n errorCode: 'BLUEPRINT_FLUX_ERROR',\n },\n });\n }),\n map(({ response }) => {\n if (!Array.isArray(response)) return response;\n if (response.length === 0) {\n return {\n errorMessage: `Aucun blueprint récupéré avec l'identifiant ${params.id}`,\n errorCode: 'BLUEPRINT_FLUX_EMPTY',\n };\n }\n return { blueprint: response[0] as Blueprint };\n })\n )\n );\n }\n\n public getBlueprints(params: Partial<{ ids: string[]; networks: number[] }> = {}) {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(`${env.config.feedRoot}Wallet/blueprints`, {\n params: {\n id: params?.ids ? `[${params.ids.join(',')}]` : null,\n network: params?.networks ? `[${params?.networks.join(',')}]` : null,\n },\n })\n ),\n map(({ response }) => {\n return response ? (response as Blueprint[]) : null;\n })\n );\n }\n\n public saveBlueprint(\n // TODO: Fix this type\n blueprint: any\n ): any {\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Wallet/createBlueprint`, {\n ...blueprint,\n })\n .pipe(\n catchError((error) => {\n return of(this._blueprintErrorsCatch(error));\n }),\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public updateBlueprintInfo(\n id: string,\n params: {\n networkId?: string;\n name?: string;\n description?: string;\n comment?: string;\n offlineEnabled?: boolean;\n transferEnabled?: boolean;\n attestationProviderId?: string;\n externalCode?: string;\n maxOwned?: number | string;\n }\n ) {\n return firstValueFrom(\n this.http\n .put(`${env.config.feedRoot}Wallet/updateBlueprintInfo`, {\n id,\n ...params,\n })\n .pipe(\n catchError((error) => {\n return of(this._blueprintErrorsCatch(error));\n }),\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public updateBlueprintConfig(\n id: string,\n config: {\n settings?: BlueprintSettings;\n rules?: BlueprintRules[];\n }\n ) {\n return firstValueFrom(\n this.http\n .put(`${env.config.feedRoot}Wallet/updateBlueprintConfig`, {\n id,\n config,\n })\n .pipe(\n catchError((error) => {\n return of(this._blueprintErrorsCatch(error));\n }),\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public getBlueprintDetails(\n blueprintId: string\n ): Promise<{ blueprintDetails: BlueprintDetails; errorCode?: string; errorMessage?: string }> {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Wallet/blueprintDetails`,\n {\n params: {\n blueprintId,\n },\n }\n )\n .pipe(\n catchError((error) => {\n return of({\n response: {\n blueprintDetails: null,\n errorCode: error.error.code,\n errorMessage: error.error.message,\n },\n });\n }),\n map(({ response }) => {\n return response || null;\n })\n )\n );\n }\n\n public deleteBlueprint(blueprintId: string): Observable<{ success: boolean }> {\n return this.http\n .delete(`${env.config.feedRoot}Wallet/deleteBlueprint`, {\n params: {\n blueprintId,\n },\n })\n .pipe(\n map(({ response: r }) => {\n return r.success as { success: boolean };\n })\n );\n }\n\n public getProductsItem(params?: ProductItemFilters): Observable {\n const version = this.cache.getVersion();\n\n return this.http\n .get(`${env.config.feedRoot}Shop/getProductsItem`, {\n params: {\n version,\n ...params,\n },\n })\n .pipe(\n map(({ response }) => {\n if (params?.id) {\n return response.productsItem[0];\n }\n\n return response.productsItem;\n }),\n catchError((err) => {\n console.error(err);\n return of(null);\n }),\n tap((res) => {\n if (!res) {\n console.warn('No response');\n }\n })\n );\n }\n\n public getNetworkProductsItem(nid?: string): any {\n let version = this.cache.getVersion();\n this.filters.network = nid;\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Shop/getProductsItem?v=` + version, {\n params: this.filters.filtersWithID,\n })\n ),\n\n map(({ response: r }) => {\n return r.productsItem as ProductItem[];\n }),\n catchError(() => throwError('Erreur serveur !'))\n );\n }\n\n public saveProduct(product: Partial) {\n return this.http\n .post>(env.config.feedRoot + `Shop/saveProduct`, {\n ...product,\n inst: product.installments,\n })\n .pipe(\n map(({ response }) => {\n this.cache.incrementVersion();\n return response;\n })\n );\n }\n\n public uploadImage(table: string, id: string, file: File, column: string) {\n let formData: FormData = new FormData();\n formData.append('table', table);\n formData.append('column', column);\n formData.append('media', file);\n formData.append('id', id);\n\n let headers = new Headers();\n headers.append('Content-Type', 'multipart/form-data');\n headers.append('Accept', 'application/json');\n\n return this.http.post(env.config.feedRoot + `Shop/UploadImage`, formData).pipe(\n map(({ response: r }) => {\n if (r.success) this.cache.incrementVersion();\n return r;\n })\n );\n }\n\n public getOrdering(nid: number) {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Shop/ordering?networkId=${nid}`, {})\n ),\n\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public saveOrdering(cats: any, citems: any): any {\n return this.http.post(env.config.feedRoot + `Shop/saveOrdering`, { cats, citems }).pipe(\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public getFamilyDiscount(nid: number) {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Shop/getFamilyDiscount?network_id=${nid}`, {})\n ),\n\n map(({ response: r }) => {\n return r.famillyDiscount;\n })\n );\n }\n\n public updateFamilyDiscount(network_id: number, promos: any): any {\n return this.http\n .post(env.config.feedRoot + `Shop/updateFamilyDiscount`, { network_id, promos })\n .pipe(\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public uploadAsset(file: File, path?: string): Observable {\n let formData: FormData = new FormData();\n formData.append('file', file);\n formData.append('path', path);\n\n return this.http\n .post>(env.config.feedRoot + `Assets/upload`, formData)\n .pipe(map(({ response }) => response));\n }\n\n public getProductStartDates(pivotType: PivotType, pivotValue: number, startProposals: number) {\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Product/getProductStartDates`,\n {\n params: {\n pivot_type: pivotType,\n pivot_value: pivotValue,\n start_proposals: startProposals,\n timezone: Intl.DateTimeFormat().resolvedOptions().timeZone,\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n )\n );\n }\n\n public getProductPreviewName(\n name: string,\n locale: string,\n data?: {\n startDate?: string;\n endDate?: string;\n pivotType?: PivotType;\n pivotValue?: number;\n startProposals?: number;\n periodType?: PeriodType;\n periodValue?: number;\n }\n ) {\n // remove empty params\n const filtered = Object.entries(data).filter(\n ([_, value]) => !['', 'null'].includes(`${value}`)\n );\n const dataFiltered = Object.fromEntries(filtered);\n\n // startProposals param is not compatible with startDate and endDate\n if (dataFiltered.startDate && dataFiltered.endDate) {\n delete dataFiltered.startProposals;\n }\n\n return firstValueFrom(\n this.http\n .get>(\n `${env.config.feedRoot}Product/previewNames`,\n {\n params: {\n name,\n locale,\n ...dataFiltered,\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.namePreviews;\n })\n )\n );\n }\n\n private _blueprintErrorsCatch(error: { error: { code?: string; message?: string; info?: any } }) {\n console.error('Erreur lors de la sauvegarde / mise à jour du blueprint', error);\n switch (error?.error?.code) {\n case 'UPDATE_BLUEPRINT_INFO_NETWORK_NOT_TRANSFERABLE':\n return {\n response: {\n error: {\n code: error.error.code,\n message: `La configuration du réseau n'autorise pas le transfert depuis le portefeuille`,\n issue:\n error.error?.info?.issues[0]?.message ||\n 'UPDATE_BLUEPRINT_INFO_NETWORK_NOT_TRANSFERABLE',\n },\n },\n };\n case 'WALLET_BLUEPRINT_UPDATE_ERROR':\n return {\n response: {\n error: {\n code: error.error.code,\n message: `Une des images ne corresponds pas aux limites`,\n issue: error.error.message || 'WALLET_BLUEPRINT_UPDATE_ERROR',\n },\n },\n };\n case 'WALLET_BLUEPRINT_CONFIG_INVALID':\n return {\n response: {\n error: {\n code: error.error.code,\n message: `La configuration du blueprint transmise ne correspond pas à l'algorithme du blueprint`,\n issue: error.error?.info?.issues[0]?.message || 'WALLET_BLUEPRINT_CONFIG_INVALID',\n },\n },\n };\n case 'WALLET_BLUEPRINTS_CREATE_ERROR':\n return {\n response: {\n error: {\n code: error.error.code,\n message: `Erreur lors de la création du blueprint`,\n issue: error.error?.message || 'WALLET_BLUEPRINTS_CREATE_ERROR',\n },\n },\n };\n default:\n return {\n response: {\n error: {\n code: 'BLUEPRINT_UNKNOW_ERROR',\n message: `Erreur non gérée, veuillez contacter le service technique`,\n issue: error?.error?.message ? error?.error?.message : 'BLUEPRINT_UNKNOW_ERROR',\n },\n },\n };\n }\n }\n\n public updateItemImage(\n itemId: string,\n imageType: ImageType,\n file: File\n ): Promise<{ success: true }> {\n let formData: FormData = new FormData();\n formData.append('itemId', itemId);\n formData.append('imageType', imageType);\n formData.append('file', file);\n\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `Shop/updateItemImage`,\n formData\n )\n .pipe(map(({ response }) => response))\n );\n }\n\n public deleteItemImage(itemId: string, imageType: ImageType): Promise<{ success: true }> {\n return firstValueFrom(\n this.http\n .delete>(env.config.feedRoot + `Shop/deleteItemImage`, {\n params: {\n itemId,\n imageType,\n },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n public updateBlueprintImage(\n blueprintId: string,\n imageType: ImageType,\n file: File\n ): Promise<{ success: true; path: string; url: string }> {\n let formData: FormData = new FormData();\n formData.append('blueprintId', blueprintId);\n formData.append('imageType', imageType);\n formData.append('file', file);\n\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `Wallet/updateBlueprintImage`,\n formData\n )\n .pipe(map(({ response }) => response))\n );\n }\n\n public deleteBlueprintImage(\n blueprintId: string,\n imageType: ImageType\n ): Promise<{ success: true }> {\n return firstValueFrom(\n this.http\n .delete(env.config.feedRoot + `Wallet/deleteBlueprintImage`, {\n params: {\n blueprintId,\n imageType,\n },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n /******************/\n /* PRICE OVERRIDE */\n /******************/\n\n public importPriceMatrix(\n productId: number,\n file: File,\n strategy: 'ADD' | 'REPLACE'\n ): Promise {\n const formData = new FormData();\n formData.append('productId', `${productId}`);\n formData.append('file', file);\n formData.append('strategy', strategy);\n\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `Shop/uploadProductOverrideMatrix`,\n formData\n )\n .pipe(map(({ response }) => response))\n );\n }\n\n public deleteProductOverride(id: number) {\n return firstValueFrom(\n this.http\n .delete>(env.config.feedRoot + `Shop/deleteProductOverride`, {\n params: {\n productOverrideId: id,\n },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n public addProductOverride(\n productId: number,\n label: string,\n priority: number,\n condition: any,\n effect: any\n ): Promise {\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `Shop/addProductOverride`,\n {\n productId,\n label,\n priority,\n condition,\n effect,\n }\n )\n .pipe(map(({ response }) => response.override))\n );\n }\n\n public editProductOverride(\n productOverrideId: number,\n label: string,\n priority: number,\n condition: any,\n effect: any\n ): Promise {\n return firstValueFrom(\n this.http\n .patch>(\n env.config.feedRoot + `Shop/editProductOverride`,\n {\n productOverrideId,\n label,\n priority,\n condition,\n effect,\n }\n )\n .pipe(map(({ response }) => response.override))\n );\n }\n\n public countProductOverride(productId: number): Promise {\n return firstValueFrom(\n this.http\n .get(env.config.feedRoot + `Shop/searchProductOverrides`, {\n params: {\n productId,\n pagination: JSON.stringify({\n offset: 0,\n limit: 1,\n }),\n },\n observe: 'response',\n })\n .pipe(\n map((response) => {\n return Number(response.headers.get('X-Airweb-Pagination-Count'));\n })\n )\n );\n }\n\n public clearProductOverrides(productId: number): Promise {\n return firstValueFrom(\n this.http\n .delete(env.config.feedRoot + `Shop/clearProductOverrides`, {\n params: {\n productId,\n },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n public downloadProductOverrideMatrix(productId: number): Observable {\n return this.http.get(env.config.feedRoot + 'Shop/downloadProductOverrideMatrix', {\n params: {\n productId,\n },\n responseType: 'blob' as 'json',\n });\n }\n\n public async updateOverridePriorities(rulesPriority: RulePriority[]): Promise {\n const request$: Promise[] = [];\n\n for (const rulePriority of rulesPriority) {\n request$.push(\n firstValueFrom(\n this.http\n .patch>(\n env.config.feedRoot + `Shop/editProductOverride`,\n {\n productOverrideId: rulePriority.id,\n priority: rulePriority.priority,\n }\n )\n .pipe(map(({ response }) => response.override))\n )\n );\n }\n\n await Promise.all(request$);\n }\n\n public async searchProductOverrides(\n productId: number,\n query?: string,\n sortBy?: string,\n sortIn?: string,\n pagination?: {\n offset: number;\n limit: number;\n }\n ): Promise {\n const params: {\n productId: number;\n query?: string;\n sortBy?: string;\n sortIn?: string;\n pagination?: string;\n } = {\n productId,\n };\n\n if (query) {\n params.query = query;\n }\n\n if (sortBy) {\n params.sortBy = sortBy;\n }\n\n if (sortIn) {\n params.sortIn = sortIn;\n }\n\n if (pagination) {\n params.pagination = JSON.stringify(pagination);\n }\n\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + `Shop/searchProductOverrides`,\n {\n params,\n }\n )\n .pipe(map(({ response }) => response.overrides))\n );\n }\n}\n","import { Component, EventEmitter, Inject, Output } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { SubmissionService } from '@app/modules/submissions/services/submission.service';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\n\nexport interface DialogData {\n documentId: string;\n documentTypeName: string;\n submissionId: number;\n customerId: string;\n}\n\n@Component({\n selector: 'tu-delete-document-modal',\n templateUrl: './delete-document-modal.component.html',\n providers: [SubmissionService],\n})\nexport class DeleteDocumentModalComponent {\n @Output() updateField = new EventEmitter();\n\n public loading = false;\n\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n constructor(\n public modalRef: MatDialogRef,\n private submissionService: SubmissionService,\n private customersService: CustomersService,\n private notification: NotificationsService,\n private translate: TranslateService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n public closeModal(): void {\n this.modalRef.close();\n }\n\n public async confirmDocumentDelete() {\n this.loading = true;\n\n try {\n await this.submissionService.deleteDocumentFile(this.data.documentId);\n\n if (this.data.submissionId) {\n this.submissionService\n .getSubmissionDetail(this.data.submissionId)\n .subscribe((submission) => this.updateField.emit(submission));\n } else {\n const documents = await this.customersService.$$getDocuments(this.data.customerId);\n this.updateField.emit(documents);\n }\n\n this.notification.success(this.translate.instant(`otherslabels.notif_delete_ok`));\n this.closeModal();\n } catch (deleteDocumentResponse) {\n this.notification.error(this.translate.instant(`otherslabels.notif_delete_ko`));\n return this.closeModal();\n } finally {\n this.loading = false;\n }\n }\n}\n","
\n {{ 'pages.submission_details.delete_document_validation' | translate }}\n
\n\n
\n

\n {{ 'pages.submission_details.modal_message_1' | translate }}\n

\n

\n {{ 'pages.submission_details.modal_message_2' | translate }} \"{{ data.documentTypeName }}\" ?\n

\n\n
\n
\n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n\n \n
\n
\n
\n\n\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, Observable, of } from 'rxjs';\nimport { map, switchMap } from 'rxjs/operators';\nimport { GenericResponse } from '../../shared/models/generic-response';\nimport { Document, Submission } from '../models/submission';\n\nexport interface $$Submission {\n id: number;\n date: string;\n product: string;\n status: string;\n supports: string[];\n docs: Doc[];\n user: User;\n}\n\ninterface User {\n customer_id: number;\n fullname: string;\n}\n\ninterface Doc {\n document_id: number;\n document_type_name: string;\n document_name: string;\n status: string;\n file_deleted_at: string;\n created_at: string;\n expires_at?: string;\n examined_at?: string;\n deleted_at?: string;\n comment?: string;\n url: string;\n file_id: string;\n file_label: string;\n type_id: number;\n}\n\n@Injectable({\n providedIn: 'root',\n})\nexport class SubmissionService {\n constructor(\n private http: HttpClient,\n private authService: AuthService,\n private filters: FilterService,\n private router: Router\n ) {}\n\n public getUnsubmitted(nid: any = null): Observable {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Submission/getUnsubmitted.json`, {\n params: { network_id: nid },\n })\n ),\n\n map(({ response: r }) => {\n if (r.errorMessage) return {} as any;\n return r.submissions as number;\n })\n );\n }\n\n public getSubmissions(): Observable {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Submission/list.json`, {\n params: this.filters.filtersWithID,\n })\n ),\n\n map(({ response: r }) => {\n return r.submissions as Submission[];\n })\n );\n }\n\n public $$getUserSubmissions(customerId: string): Promise {\n const http$ = this.http\n .get>(\n env.config.feedRoot + `Submission/getUserSubmissions.json`,\n {\n params: { user_id: customerId },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.submissions;\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public getUserSubmissions(user_id: any): Observable {\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `Submission/getUserSubmissions.json`, {\n params: { user_id },\n })\n ),\n\n map(({ response: r }) => {\n return r.submissions as Submission[];\n })\n );\n }\n\n public getSubmissionDetail(id: number) {\n return this.http\n .get>(\n env.config.feedRoot + `Submission/details.json`,\n { params: { id, antiCache: Date.now() } }\n )\n .pipe(\n map(({ response }) => {\n const documents = response.submission.docs as Doc[];\n /**\n * Essentially group every document by types:\n * ```ts\n * {\n * '1345': [{ ... }],\n * '1346': [{ ... }]\n * }\n * ```\n */\n const documentsGroupedByType = documents.reduce>(\n (acc: { [x: string]: any[] }, doc: { type_id: string | number }) => {\n if (!acc[doc.type_id]) {\n acc[doc.type_id] = [];\n }\n\n acc[doc.type_id].push(doc);\n\n return acc;\n },\n {}\n );\n\n /**\n * Then picking only the latest one (based on their ID)\n */\n const filteredDocuments = Object.values(documentsGroupedByType).reduce(\n (acc, docs) => {\n const [latestDocumentForThisType] = docs.sort(\n (a, b) => b.document_id - +a.document_id\n );\n acc.push(latestDocumentForThisType);\n\n return acc;\n },\n []\n );\n\n return { ...response.submission, docs: filteredDocuments };\n })\n );\n }\n\n public getDocumentDetail(document_id: number): Observable {\n return this.http\n .get>(\n env.config.feedRoot + `/Submission/documentFileDownloadUrl.json`,\n {\n params: { documentId: document_id },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.url;\n })\n );\n }\n\n public updateSubmission(\n id: string | number,\n submission_statut: string,\n documents: any[]\n ): Observable<{ id: number; documents: Array; submission_status: string }> {\n return this.http\n .post; submission_status: string }>>(\n env.config.feedRoot + `Submission/updateSubmission.json`,\n {\n id,\n submission_statut,\n documents,\n }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n );\n }\n\n public postDocumentComment(document_id: string, comment: string) {\n return this.http\n .post>(\n env.config.feedRoot + `Submission/postDocumentComment.json`,\n { document_id, comment }\n )\n .pipe(\n map(({ response }) => {\n return response;\n })\n );\n }\n\n public deleteDocument(documentId: string) {\n return firstValueFrom(\n this.http\n .delete(env.config.feedRoot + `Submission/deleteDocument.json`, {\n params: { documentId },\n })\n .pipe(map(({ response }) => response))\n );\n }\n\n public deleteDocumentFile(documentId: string) {\n return firstValueFrom(\n this.http\n .delete>(\n env.config.feedRoot + `Submission/deleteDocumentFile`,\n {\n params: { documentId },\n }\n )\n .pipe(map(({ response }) => response))\n );\n }\n\n public updateDocument(body: {\n documentTypeId: string;\n userId: string;\n file: File;\n submissionId?: string;\n }) {\n const formData = new FormData();\n formData.append('documentTypeId', body.documentTypeId);\n formData.append('userId', body.userId);\n formData.append('file', body.file);\n\n if (body.submissionId) {\n formData.append('submissionId', body.submissionId);\n }\n\n return firstValueFrom(\n this.http\n .post(env.config.feedRoot + `Submission/updateDocument`, formData)\n .pipe(map(({ response }) => response))\n );\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, throwError } from 'rxjs';\nimport { catchError, map } from 'rxjs/operators';\nimport { SubscriptionDetails, SubscriptionListItem } from '../models/subscription.interface';\n\n@Injectable()\nexport class SubscriptionsService {\n constructor(private http: HttpClient, private authService: AuthService, private router: Router) {}\n\n public list() {\n return this.http\n .get>(`${env.config.feedRoot}Subscriptions/list.json`)\n .pipe(\n map(({ response }) => {\n return response.sort((a, b) => b.id - a.id);\n }),\n catchError(() => throwError('Erreur serveur !'))\n );\n }\n\n public get(id: string, cache: boolean = true) {\n const params: { id: string; time?: number } = { id };\n\n if (!cache) {\n params.time = Date.now();\n }\n\n return this.http\n .get(`${env.config.feedRoot}Subscriptions/get.json`, {\n params,\n })\n .pipe(\n map(({ response }) => {\n return response as SubscriptionDetails;\n }),\n catchError(() => throwError('Erreur serveur !'))\n );\n }\n\n public enableSubscription(id: number) {\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Subscriptions/enable.json`, { id })\n .pipe(\n map(({ response }) => {\n return response as any;\n }),\n catchError(() => throwError('Erreur serveur !'))\n )\n );\n }\n\n public disableSubscription(id: number, handleError: boolean = true) {\n return firstValueFrom(\n this.http\n .post(`${env.config.feedRoot}Subscriptions/disable.json`, { id })\n .pipe(\n map(({ response }) => {\n return response;\n }),\n catchError((error) => {\n if (handleError) {\n return throwError('Erreur serveur !');\n }\n\n return throwError(error.error.code);\n })\n )\n );\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { CacheManager } from '@app/utils/cache-manager';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, Observable, of } from 'rxjs';\nimport { map, switchMap } from 'rxjs/operators';\nimport { DeleteUserResponse, User } from '../models/user';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\n\n@Injectable()\nexport class UsersService {\n private cache = new CacheManager('users');\n\n constructor(\n private http: HttpClient,\n private authService: AuthService,\n private router: Router,\n private filters: FilterService\n ) { }\n\n public getUsers(): Observable {\n let version = this.cache.getVersion();\n\n return of([null]).pipe(\n switchMap(() =>\n this.http.get(env.config.feedRoot + `User/list.json?v=${version}`, {\n params: this.filters.filtersWithID,\n })\n ),\n\n map(({ response: r }) => {\n return r.users as User[];\n })\n );\n }\n\n public getUserDetail(id: number): Observable {\n let version = this.cache.getVersion();\n\n return this.http.get(env.config.feedRoot + `User/details.json?id=${id}&v=${version}`).pipe(\n map(({ response: r }) => {\n return r.user as User;\n })\n );\n }\n\n public createUser(user: User): Observable<{ id: number }> {\n return this.http.post(env.config.feedRoot + `User/createUser.json`, user).pipe(\n map(({ response: r }) => {\n if (r.errorMessage) {\n if (r.errorCode !== null) throw new Error(r.errorCode);\n } else {\n this.cache.incrementVersion();\n return r.user as { id: number };\n }\n })\n );\n }\n\n public editUser(user: User): Observable<{ id: number }> {\n return this.http.post(env.config.feedRoot + `User/editUser.json`, user).pipe(\n map(({ response: r }) => {\n if (r.errorMessage) {\n if (r.errorCode !== null) throw new Error(r.errorCode);\n } else {\n this.cache.incrementVersion();\n return r as { id: number };\n }\n })\n );\n }\n\n public deleteUser(id: string) {\n return this.http\n .delete(`${env.config.feedRoot}User/deleteUser.json`, {\n params: { id },\n })\n .pipe(\n map(({ response: r }) => {\n return r;\n })\n );\n }\n\n public forceAuthSecondFactor(userIds: number[]): Promise<{ success: boolean }> {\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `User/forceAuthSecondFactor.json`,\n {\n userIds: JSON.stringify(userIds),\n }\n )\n .pipe(map(({ response }) => response))\n );\n }\n}\n","type VersionNumbers = { [serviceName: string]: number };\n\nexport class CacheManager {\n private lsKey = 'cacheVersions';\n private serviceName;\n\n private set versions(v: VersionNumbers) {\n if (v) localStorage.setItem(this.lsKey, JSON.stringify(v));\n else localStorage.removeItem(this.lsKey);\n }\n private get versions(): VersionNumbers {\n return JSON.parse(localStorage.getItem(this.lsKey));\n }\n\n constructor(serviceName) {\n this.serviceName = serviceName;\n }\n\n public incrementVersion() {\n let v = this.getVersion() + 1;\n this.versions = { ...this.versions, [this.serviceName]: v };\n }\n\n public getVersion(): number {\n return this.versions?.[this.serviceName] ?? 0;\n }\n}\n","export function escapeRegExp(s) {\n if (!s) return;\n\n return s.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&');\n}\n","import { outline } from './outline';\nimport { solid } from './solid';\n\nexport const heroicons = {\n outline,\n solid,\n};\n","export const outline = {\n academicCap: {\n path: ``,\n outline: true,\n },\n adjustments: {\n path: ``,\n outline: true,\n },\n annotation: {\n path: ``,\n outline: true,\n },\n archive: {\n path: ``,\n outline: true,\n },\n arrowCircleDown: {\n path: ``,\n outline: true,\n },\n arrowCircleLeft: {\n path: ``,\n outline: true,\n },\n arrowCircleRight: {\n path: ``,\n outline: true,\n },\n arrowCircleUp: {\n path: ``,\n outline: true,\n },\n arrowDown: {\n path: ``,\n outline: true,\n },\n arrowLeft: {\n path: ``,\n outline: true,\n },\n arrowNarrowDown: {\n path: ``,\n outline: true,\n },\n arrowNarrowLeft: {\n path: ``,\n outline: true,\n },\n arrowNarrowRight: {\n path: ``,\n outline: true,\n },\n arrowNarrowUp: {\n path: ``,\n outline: true,\n },\n arrowRight: {\n path: ``,\n outline: true,\n },\n arrowSmDown: {\n path: ``,\n outline: true,\n },\n arrowSmLeft: {\n path: ``,\n outline: true,\n },\n arrowSmRight: {\n path: ``,\n outline: true,\n },\n arrowSmUp: {\n path: ``,\n outline: true,\n },\n arrowUp: {\n path: ``,\n outline: true,\n },\n arrowsExpand: {\n path: ``,\n outline: true,\n },\n atSymbol: {\n path: ``,\n outline: true,\n },\n backspace: {\n path: ``,\n outline: true,\n },\n badgeCheck: {\n path: ``,\n outline: true,\n },\n ban: {\n path: ``,\n outline: true,\n },\n beaker: {\n path: ``,\n outline: true,\n },\n bell: {\n path: ``,\n outline: true,\n },\n bookOpen: {\n path: ``,\n outline: true,\n },\n bookmarkAlt: {\n path: ``,\n outline: true,\n },\n bookmark: {\n path: ``,\n outline: true,\n },\n briefcase: {\n path: ``,\n outline: true,\n },\n cake: {\n path: ``,\n outline: true,\n },\n calculator: {\n path: ``,\n outline: true,\n },\n calendar: {\n path: ``,\n outline: true,\n },\n camera: {\n path: ``,\n outline: true,\n },\n cash: {\n path: ``,\n outline: true,\n },\n chartBar: {\n path: ``,\n outline: true,\n },\n chartPie: {\n path: ``,\n outline: true,\n },\n chartSquareBar: {\n path: ``,\n outline: true,\n },\n chatAlt_2: {\n path: ``,\n outline: true,\n },\n chatAlt: {\n path: ``,\n outline: true,\n },\n chat: {\n path: ``,\n outline: true,\n },\n checkCircle: {\n path: ``,\n outline: true,\n },\n check: {\n path: ``,\n outline: true,\n },\n chevronDoubleDown: {\n path: ``,\n outline: true,\n },\n chevronDoubleLeft: {\n path: ``,\n outline: true,\n },\n chevronDoubleRight: {\n path: ``,\n outline: true,\n },\n chevronDoubleUp: {\n path: ``,\n outline: true,\n },\n chevronDown: {\n path: ``,\n outline: true,\n },\n chevronLeft: {\n path: ``,\n outline: true,\n },\n chevronRight: {\n path: ``,\n outline: true,\n },\n chevronUp: {\n path: ``,\n outline: true,\n },\n chip: {\n path: ``,\n outline: true,\n },\n clipboardCheck: {\n path: ``,\n outline: true,\n },\n clipboardCopy: {\n path: ``,\n outline: true,\n },\n clipboardList: {\n path: ``,\n outline: true,\n },\n clipboard: {\n path: ``,\n outline: true,\n },\n clock: {\n path: ``,\n outline: true,\n },\n cloudDownload: {\n path: ``,\n outline: true,\n },\n cloudUpload: {\n path: ``,\n outline: true,\n },\n cloud: {\n path: ``,\n outline: true,\n },\n code: {\n path: ``,\n outline: true,\n },\n cog: {\n path: ``,\n outline: true,\n },\n collection: {\n path: ``,\n outline: true,\n },\n colorSwatch: {\n path: ``,\n outline: true,\n },\n creditCard: {\n path: ``,\n outline: true,\n },\n cubeTransparent: {\n path: ``,\n outline: true,\n },\n cube: {\n path: ``,\n outline: true,\n },\n currencyBangladeshi: {\n path: ``,\n outline: true,\n },\n currencyDollar: {\n path: ``,\n outline: true,\n },\n currencyEuro: {\n path: ``,\n outline: true,\n },\n currencyPound: {\n path: ``,\n outline: true,\n },\n currencyRupee: {\n path: ``,\n outline: true,\n },\n currencyYen: {\n path: ``,\n outline: true,\n },\n cursorClick: {\n path: ``,\n outline: true,\n },\n database: {\n path: ``,\n outline: true,\n },\n desktopComputer: {\n path: ``,\n outline: true,\n },\n deviceMobile: {\n path: ``,\n outline: true,\n },\n deviceTablet: {\n path: ``,\n outline: true,\n },\n documentAdd: {\n path: ``,\n outline: true,\n },\n documentDownload: {\n path: ``,\n outline: true,\n },\n documentDuplicate: {\n path: ``,\n outline: true,\n },\n documentRemove: {\n path: ``,\n outline: true,\n },\n documentReport: {\n path: ``,\n outline: true,\n },\n documentSearch: {\n path: ``,\n outline: true,\n },\n documentText: {\n path: ``,\n outline: true,\n },\n document: {\n path: ``,\n outline: true,\n },\n dotsCircleHorizontal: {\n path: ``,\n outline: true,\n },\n dotsHorizontal: {\n path: ``,\n outline: true,\n },\n dotsVertical: {\n path: ``,\n outline: true,\n },\n download: {\n path: ``,\n outline: true,\n },\n duplicate: {\n path: ``,\n outline: true,\n },\n emojiHappy: {\n path: ``,\n outline: true,\n },\n emojiSad: {\n path: ``,\n outline: true,\n },\n exclamationCircle: {\n path: ``,\n outline: true,\n },\n exclamation: {\n path: ``,\n outline: true,\n },\n externalLink: {\n path: ``,\n outline: true,\n },\n eyeOff: {\n path: ``,\n outline: true,\n },\n eye: {\n path: ``,\n outline: true,\n },\n fastForward: {\n path: ``,\n outline: true,\n },\n film: {\n path: ``,\n outline: true,\n },\n filter: {\n path: ``,\n outline: true,\n },\n fingerPrint: {\n path: ``,\n outline: true,\n },\n fire: {\n path: ``,\n outline: true,\n },\n flag: {\n path: ``,\n outline: true,\n },\n folderAdd: {\n path: ``,\n outline: true,\n },\n folderDownload: {\n path: ``,\n outline: true,\n },\n folderOpen: {\n path: ``,\n outline: true,\n },\n folderRemove: {\n path: ``,\n outline: true,\n },\n folder: {\n path: ``,\n outline: true,\n },\n gift: {\n path: ``,\n outline: true,\n },\n globeAlt: {\n path: ``,\n outline: true,\n },\n globe: {\n path: ``,\n outline: true,\n },\n hand: {\n path: ``,\n outline: true,\n },\n hashtag: {\n path: ``,\n outline: true,\n },\n heart: {\n path: ``,\n outline: true,\n },\n home: {\n path: ``,\n outline: true,\n },\n identification: {\n path: ``,\n outline: true,\n },\n inboxIn: {\n path: ``,\n outline: true,\n },\n inbox: {\n path: ``,\n outline: true,\n },\n informationCircle: {\n path: ``,\n outline: true,\n },\n key: {\n path: ``,\n outline: true,\n },\n library: {\n path: ``,\n outline: true,\n },\n lightBulb: {\n path: ``,\n outline: true,\n },\n lightningBolt: {\n path: ``,\n outline: true,\n },\n link: {\n path: ``,\n outline: true,\n },\n locationMarker: {\n path: ``,\n outline: true,\n },\n lockClosed: {\n path: ``,\n outline: true,\n },\n lockOpen: {\n path: ``,\n outline: true,\n },\n login: {\n path: ``,\n outline: true,\n },\n logout: {\n path: ``,\n outline: true,\n },\n mailOpen: {\n path: ``,\n outline: true,\n },\n mail: {\n path: ``,\n outline: true,\n },\n map: {\n path: ``,\n outline: true,\n },\n menuAlt_1: {\n path: ``,\n outline: true,\n },\n menuAlt_2: {\n path: ``,\n outline: true,\n },\n menuAlt_3: {\n path: ``,\n outline: true,\n },\n menuAlt_4: {\n path: ``,\n outline: true,\n },\n menu: {\n path: ``,\n outline: true,\n },\n microphone: {\n path: ``,\n outline: true,\n },\n minusCircle: {\n path: ``,\n outline: true,\n },\n minusSm: {\n path: ``,\n outline: true,\n },\n minus: {\n path: ``,\n outline: true,\n },\n moon: {\n path: ``,\n outline: true,\n },\n musicNote: {\n path: ``,\n outline: true,\n },\n newspaper: {\n path: ``,\n outline: true,\n },\n officeBuilding: {\n path: ``,\n outline: true,\n },\n paperAirplane: {\n path: ``,\n outline: true,\n },\n paperClip: {\n path: ``,\n outline: true,\n },\n pause: {\n path: ``,\n outline: true,\n },\n pencilAlt: {\n path: ``,\n outline: true,\n },\n pencil: {\n path: ``,\n outline: true,\n },\n phoneIncoming: {\n path: ``,\n outline: true,\n },\n phoneMissedCall: {\n path: ``,\n outline: true,\n },\n phoneOutgoing: {\n path: ``,\n outline: true,\n },\n phone: {\n path: ``,\n outline: true,\n },\n photograph: {\n path: ``,\n outline: true,\n },\n play: {\n path: ``,\n outline: true,\n },\n plusCircle: {\n path: ``,\n outline: true,\n },\n plusSm: {\n path: ``,\n outline: true,\n },\n plus: {\n path: ``,\n outline: true,\n },\n presentationChartBar: {\n path: ``,\n outline: true,\n },\n presentationChartLine: {\n path: ``,\n outline: true,\n },\n printer: {\n path: ``,\n outline: true,\n },\n puzzle: {\n path: ``,\n outline: true,\n },\n qrcode: {\n path: ``,\n outline: true,\n },\n questionMarkCircle: {\n path: ``,\n outline: true,\n },\n receiptRefund: {\n path: ``,\n outline: true,\n },\n receiptTax: {\n path: ``,\n outline: true,\n },\n refresh: {\n path: ``,\n outline: true,\n },\n reply: {\n path: ``,\n outline: true,\n },\n rewind: {\n path: ``,\n outline: true,\n },\n rss: {\n path: ``,\n outline: true,\n },\n saveAs: {\n path: ``,\n outline: true,\n },\n save: {\n path: ``,\n outline: true,\n },\n scale: {\n path: ``,\n outline: true,\n },\n scissors: {\n path: ``,\n outline: true,\n },\n searchCircle: {\n path: ``,\n outline: true,\n },\n search: {\n path: ``,\n outline: true,\n },\n selector: {\n path: ``,\n outline: true,\n },\n server: {\n path: ``,\n outline: true,\n },\n share: {\n path: ``,\n outline: true,\n },\n shieldCheck: {\n path: ``,\n outline: true,\n },\n shieldExclamation: {\n path: ``,\n outline: true,\n },\n shoppingBag: {\n path: ``,\n outline: true,\n },\n shoppingCart: {\n path: ``,\n outline: true,\n },\n sortAscending: {\n path: ``,\n outline: true,\n },\n sortDescending: {\n path: ``,\n outline: true,\n },\n sparkles: {\n path: ``,\n outline: true,\n },\n speakerphone: {\n path: ``,\n outline: true,\n },\n star: {\n path: ``,\n outline: true,\n },\n statusOffline: {\n path: ``,\n outline: true,\n },\n statusOnline: {\n path: ``,\n outline: true,\n },\n stop: {\n path: ``,\n outline: true,\n },\n sun: {\n path: ``,\n outline: true,\n },\n support: {\n path: ``,\n outline: true,\n },\n switchHorizontal: {\n path: ``,\n outline: true,\n },\n switchVertical: {\n path: ``,\n outline: true,\n },\n table: {\n path: ``,\n outline: true,\n },\n tag: {\n path: ``,\n outline: true,\n },\n template: {\n path: ``,\n outline: true,\n },\n terminal: {\n path: ``,\n outline: true,\n },\n thumbDown: {\n path: ``,\n outline: true,\n },\n thumbUp: {\n path: ``,\n outline: true,\n },\n ticket: {\n path: ``,\n outline: true,\n },\n translate: {\n path: ``,\n outline: true,\n },\n trash: {\n path: ``,\n outline: true,\n },\n trendingDown: {\n path: ``,\n outline: true,\n },\n trendingUp: {\n path: ``,\n outline: true,\n },\n truck: {\n path: ``,\n outline: true,\n },\n upload: {\n path: ``,\n outline: true,\n },\n userAdd: {\n path: ``,\n outline: true,\n },\n userCircle: {\n path: ``,\n outline: true,\n },\n userGroup: {\n path: ``,\n outline: true,\n },\n userRemove: {\n path: ``,\n outline: true,\n },\n user: {\n path: ``,\n outline: true,\n },\n users: {\n path: ``,\n outline: true,\n },\n variable: {\n path: ``,\n outline: true,\n },\n videoCamera: {\n path: ``,\n outline: true,\n },\n viewBoards: {\n path: ``,\n outline: true,\n },\n viewGridAdd: {\n path: ``,\n outline: true,\n },\n viewGrid: {\n path: ``,\n outline: true,\n },\n viewList: {\n path: ``,\n outline: true,\n },\n volumeOff: {\n path: ``,\n outline: true,\n },\n volumeUp: {\n path: ``,\n outline: true,\n },\n wifi: {\n path: ``,\n outline: true,\n },\n xCircle: {\n path: ``,\n outline: true,\n },\n x: {\n path: ``,\n outline: true,\n },\n zoomIn: {\n path: ``,\n outline: true,\n },\n zoomOut: {\n path: ``,\n outline: true,\n },\n banknotes: {\n path: ``,\n outline: true,\n },\n square3Stack3D: {\n path: ``,\n outline: true,\n },\n};\n","export const solid = {\n academicCap: {\n path: ``,\n outline: false,\n },\n adjustments: {\n path: ``,\n outline: false,\n },\n annotation: {\n path: ``,\n outline: false,\n },\n archive: {\n path: ``,\n outline: false,\n },\n arrowCircleDown: {\n path: ``,\n outline: false,\n },\n arrowCircleLeft: {\n path: ``,\n outline: false,\n },\n arrowCircleRight: {\n path: ``,\n outline: false,\n },\n arrowCircleUp: {\n path: ``,\n outline: false,\n },\n arrowDown: {\n path: ``,\n outline: false,\n },\n arrowLeft: {\n path: ``,\n outline: false,\n },\n arrowNarrowDown: {\n path: ``,\n outline: false,\n },\n arrowNarrowLeft: {\n path: ``,\n outline: false,\n },\n arrowNarrowRight: {\n path: ``,\n outline: false,\n },\n arrowNarrowUp: {\n path: ``,\n outline: false,\n },\n arrowRight: {\n path: ``,\n outline: false,\n },\n arrowSmDown: {\n path: ``,\n outline: false,\n },\n arrowSmLeft: {\n path: ``,\n outline: false,\n },\n arrowSmRight: {\n path: ``,\n outline: false,\n },\n arrowSmUp: {\n path: ``,\n outline: false,\n },\n arrowUp: {\n path: ``,\n outline: false,\n },\n arrowsExpand: {\n path: ``,\n outline: false,\n },\n atSymbol: {\n path: ``,\n outline: false,\n },\n backspace: {\n path: ``,\n outline: false,\n },\n badgeCheck: {\n path: ``,\n outline: false,\n },\n ban: {\n path: ``,\n outline: false,\n },\n beaker: {\n path: ``,\n outline: false,\n },\n bell: {\n path: ``,\n outline: false,\n },\n bookOpen: {\n path: ``,\n outline: false,\n },\n bookmarkAlt: {\n path: ``,\n outline: false,\n },\n bookmark: {\n path: ``,\n outline: false,\n },\n briefcase: {\n path: ``,\n outline: false,\n },\n cake: {\n path: ``,\n outline: false,\n },\n calculator: {\n path: ``,\n outline: false,\n },\n calendar: {\n path: ``,\n outline: false,\n },\n camera: {\n path: ``,\n outline: false,\n },\n cash: {\n path: ``,\n outline: false,\n },\n chartBar: {\n path: ``,\n outline: false,\n },\n chartPie: {\n path: ``,\n outline: false,\n },\n chartSquareBar: {\n path: ``,\n outline: false,\n },\n chatAlt_2: {\n path: ``,\n outline: false,\n },\n chatAlt: {\n path: ``,\n outline: false,\n },\n chat: {\n path: ``,\n outline: false,\n },\n checkCircle: {\n path: ``,\n outline: false,\n },\n check: {\n path: ``,\n outline: false,\n },\n chevronDoubleDown: {\n path: ``,\n outline: false,\n },\n chevronDoubleLeft: {\n path: ``,\n outline: false,\n },\n chevronDoubleRight: {\n path: ``,\n outline: false,\n },\n chevronDoubleUp: {\n path: ``,\n outline: false,\n },\n chevronDown: {\n path: ``,\n outline: false,\n },\n chevronLeft: {\n path: ``,\n outline: false,\n },\n chevronRight: {\n path: ``,\n outline: false,\n },\n chevronUp: {\n path: ``,\n outline: false,\n },\n chip: {\n path: ``,\n outline: false,\n },\n clipboardCheck: {\n path: ``,\n outline: false,\n },\n clipboardCopy: {\n path: ``,\n outline: false,\n },\n clipboardList: {\n path: ``,\n outline: false,\n },\n clipboard: {\n path: ``,\n outline: false,\n },\n clock: {\n path: ``,\n outline: false,\n },\n cloudDownload: {\n path: ``,\n outline: false,\n },\n cloudUpload: {\n path: ``,\n outline: false,\n },\n cloud: {\n path: ``,\n outline: false,\n },\n code: {\n path: ``,\n outline: false,\n },\n cog: {\n path: ``,\n outline: false,\n },\n collection: {\n path: ``,\n outline: false,\n },\n colorSwatch: {\n path: ``,\n outline: false,\n },\n creditCard: {\n path: ``,\n outline: false,\n },\n cubeTransparent: {\n path: ``,\n outline: false,\n },\n cube: {\n path: ``,\n outline: false,\n },\n currencyBangladeshi: {\n path: ``,\n outline: false,\n },\n currencyDollar: {\n path: ``,\n outline: false,\n },\n currencyEuro: {\n path: ``,\n outline: false,\n },\n currencyPound: {\n path: ``,\n outline: false,\n },\n currencyRupee: {\n path: ``,\n outline: false,\n },\n currencyYen: {\n path: ``,\n outline: false,\n },\n cursorClick: {\n path: ``,\n outline: false,\n },\n database: {\n path: ``,\n outline: false,\n },\n desktopComputer: {\n path: ``,\n outline: false,\n },\n deviceMobile: {\n path: ``,\n outline: false,\n },\n deviceTablet: {\n path: ``,\n outline: false,\n },\n documentAdd: {\n path: ``,\n outline: false,\n },\n documentDownload: {\n path: ``,\n outline: false,\n },\n documentDuplicate: {\n path: ``,\n outline: false,\n },\n documentRemove: {\n path: ``,\n outline: false,\n },\n documentReport: {\n path: ``,\n outline: false,\n },\n documentSearch: {\n path: ``,\n outline: false,\n },\n documentText: {\n path: ``,\n outline: false,\n },\n document: {\n path: ``,\n outline: false,\n },\n dotsCircleHorizontal: {\n path: ``,\n outline: false,\n },\n dotsHorizontal: {\n path: ``,\n outline: false,\n },\n dotsVertical: {\n path: ``,\n outline: false,\n },\n download: {\n path: ``,\n outline: false,\n },\n duplicate: {\n path: ``,\n outline: false,\n },\n emojiHappy: {\n path: ``,\n outline: false,\n },\n emojiSad: {\n path: ``,\n outline: false,\n },\n exclamationCircle: {\n path: ``,\n outline: false,\n },\n exclamation: {\n path: ``,\n outline: false,\n },\n externalLink: {\n path: ``,\n outline: false,\n },\n eyeOff: {\n path: ``,\n outline: false,\n },\n eye: {\n path: ``,\n outline: false,\n },\n fastForward: {\n path: ``,\n outline: false,\n },\n film: {\n path: ``,\n outline: false,\n },\n filter: {\n path: ``,\n outline: false,\n },\n fingerPrint: {\n path: ``,\n outline: false,\n },\n fire: {\n path: ``,\n outline: false,\n },\n flag: {\n path: ``,\n outline: false,\n },\n folderAdd: {\n path: ``,\n outline: false,\n },\n folderDownload: {\n path: ``,\n outline: false,\n },\n folderOpen: {\n path: ``,\n outline: false,\n },\n folderRemove: {\n path: ``,\n outline: false,\n },\n folder: {\n path: ``,\n outline: false,\n },\n gift: {\n path: ``,\n outline: false,\n },\n globeAlt: {\n path: ``,\n outline: false,\n },\n globe: {\n path: ``,\n outline: false,\n },\n hand: {\n path: ``,\n outline: false,\n },\n hashtag: {\n path: ``,\n outline: false,\n },\n heart: {\n path: ``,\n outline: false,\n },\n home: {\n path: ``,\n outline: false,\n },\n identification: {\n path: ``,\n outline: false,\n },\n inboxIn: {\n path: ``,\n outline: false,\n },\n inbox: {\n path: ``,\n outline: false,\n },\n informationCircle: {\n path: ``,\n outline: false,\n },\n key: {\n path: ``,\n outline: false,\n },\n library: {\n path: ``,\n outline: false,\n },\n lightBulb: {\n path: ``,\n outline: false,\n },\n lightningBolt: {\n path: ``,\n outline: false,\n },\n link: {\n path: ``,\n outline: false,\n },\n locationMarker: {\n path: ``,\n outline: false,\n },\n lockClosed: {\n path: ``,\n outline: false,\n },\n lockOpen: {\n path: ``,\n outline: false,\n },\n login: {\n path: ``,\n outline: false,\n },\n logout: {\n path: ``,\n outline: false,\n },\n mailOpen: {\n path: ``,\n outline: false,\n },\n mail: {\n path: ``,\n outline: false,\n },\n map: {\n path: ``,\n outline: false,\n },\n menuAlt_1: {\n path: ``,\n outline: false,\n },\n menuAlt_2: {\n path: ``,\n outline: false,\n },\n menuAlt_3: {\n path: ``,\n outline: false,\n },\n menuAlt_4: {\n path: ``,\n outline: false,\n },\n menu: {\n path: ``,\n outline: false,\n },\n microphone: {\n path: ``,\n outline: false,\n },\n minusCircle: {\n path: ``,\n outline: false,\n },\n minusSm: {\n path: ``,\n outline: false,\n },\n minus: {\n path: ``,\n outline: false,\n },\n moon: {\n path: ``,\n outline: false,\n },\n musicNote: {\n path: ``,\n outline: false,\n },\n newspaper: {\n path: ``,\n outline: false,\n },\n officeBuilding: {\n path: ``,\n outline: false,\n },\n paperAirplane: {\n path: ``,\n outline: false,\n },\n paperClip: {\n path: ``,\n outline: false,\n },\n pause: {\n path: ``,\n outline: false,\n },\n pencilAlt: {\n path: ``,\n outline: false,\n },\n pencil: {\n path: ``,\n outline: false,\n },\n phoneIncoming: {\n path: ``,\n outline: false,\n },\n phoneMissedCall: {\n path: ``,\n outline: false,\n },\n phoneOutgoing: {\n path: ``,\n outline: false,\n },\n phone: {\n path: ``,\n outline: false,\n },\n photograph: {\n path: ``,\n outline: false,\n },\n play: {\n path: ``,\n outline: false,\n },\n plusCircle: {\n path: ``,\n outline: false,\n },\n plusSm: {\n path: ``,\n outline: false,\n },\n plus: {\n path: ``,\n outline: false,\n },\n presentationChartBar: {\n path: ``,\n outline: false,\n },\n presentationChartLine: {\n path: ``,\n outline: false,\n },\n printer: {\n path: ``,\n outline: false,\n },\n puzzle: {\n path: ``,\n outline: false,\n },\n qrcode: {\n path: ``,\n outline: false,\n },\n questionMarkCircle: {\n path: ``,\n outline: false,\n },\n receiptRefund: {\n path: ``,\n outline: false,\n },\n receiptTax: {\n path: ``,\n outline: false,\n },\n refresh: {\n path: ``,\n outline: false,\n },\n reply: {\n path: ``,\n outline: false,\n },\n rewind: {\n path: ``,\n outline: false,\n },\n rss: {\n path: ``,\n outline: false,\n },\n saveAs: {\n path: ``,\n outline: false,\n },\n save: {\n path: ``,\n outline: false,\n },\n scale: {\n path: ``,\n outline: false,\n },\n scissors: {\n path: ``,\n outline: false,\n },\n searchCircle: {\n path: ``,\n outline: false,\n },\n search: {\n path: ``,\n outline: false,\n },\n selector: {\n path: ``,\n outline: false,\n },\n server: {\n path: ``,\n outline: false,\n },\n share: {\n path: ``,\n outline: false,\n },\n shieldCheck: {\n path: ``,\n outline: false,\n },\n shieldExclamation: {\n path: ``,\n outline: false,\n },\n shoppingBag: {\n path: ``,\n outline: false,\n },\n shoppingCart: {\n path: ``,\n outline: false,\n },\n sortAscending: {\n path: ``,\n outline: false,\n },\n sortDescending: {\n path: ``,\n outline: false,\n },\n sparkles: {\n path: ``,\n outline: false,\n },\n speakerphone: {\n path: ``,\n outline: false,\n },\n star: {\n path: ``,\n outline: false,\n },\n statusOffline: {\n path: ``,\n outline: false,\n },\n statusOnline: {\n path: ``,\n outline: false,\n },\n stop: {\n path: ``,\n outline: false,\n },\n sun: {\n path: ``,\n outline: false,\n },\n support: {\n path: ``,\n outline: false,\n },\n switchHorizontal: {\n path: ``,\n outline: false,\n },\n switchVertical: {\n path: ``,\n outline: false,\n },\n table: {\n path: ``,\n outline: false,\n },\n tag: {\n path: ``,\n outline: false,\n },\n template: {\n path: ``,\n outline: false,\n },\n terminal: {\n path: ``,\n outline: false,\n },\n thumbDown: {\n path: ``,\n outline: false,\n },\n thumbUp: {\n path: ``,\n outline: false,\n },\n ticket: {\n path: ``,\n outline: false,\n },\n translate: {\n path: ``,\n outline: false,\n },\n trash: {\n path: ``,\n outline: false,\n },\n trendingDown: {\n path: ``,\n outline: false,\n },\n trendingUp: {\n path: ``,\n outline: false,\n },\n truck: {\n path: ``,\n outline: false,\n },\n upload: {\n path: ``,\n outline: false,\n },\n userAdd: {\n path: ``,\n outline: false,\n },\n userCircle: {\n path: ``,\n outline: false,\n },\n userGroup: {\n path: ``,\n outline: false,\n },\n userRemove: {\n path: ``,\n outline: false,\n },\n user: {\n path: ``,\n outline: false,\n },\n users: {\n path: ``,\n outline: false,\n },\n variable: {\n path: ``,\n outline: false,\n },\n videoCamera: {\n path: ``,\n outline: false,\n },\n viewBoards: {\n path: ``,\n outline: false,\n },\n viewGridAdd: {\n path: ``,\n outline: false,\n },\n viewGrid: {\n path: ``,\n outline: false,\n },\n viewList: {\n path: ``,\n outline: false,\n },\n volumeOff: {\n path: ``,\n outline: false,\n },\n volumeUp: {\n path: ``,\n outline: false,\n },\n wifi: {\n path: ``,\n outline: false,\n },\n xCircle: {\n path: ``,\n outline: false,\n },\n x: {\n path: ``,\n outline: false,\n },\n zoomIn: {\n path: ``,\n outline: false,\n },\n zoomOut: {\n path: ``,\n outline: false,\n },\n exclamationTriangle: {\n path: ``,\n outline: false,\n },\n dragHandle: {\n path: ``,\n outline: false,\n },\n};\n","export const environment = {\n production: true,\n config: {\n feedRoot: 'https://api.bo.ticket.run.airweb.fr/feed/',\n tapRoot: 'https://production-api-kcvqfuvlla-ew.a.run.app/',\n abtRoot: 'https://api.validators.ticket.airweb.fr',\n exportRoot: 'https://eu-export-ack5cv4rnq-ew.a.run.app/',\n mapApiKey: 'AIzaSyBBqcl8KrnY5PJetpLkmNT6SwxVdZwmDtQ',\n defaultCoords: '46.71109,1.7191036',\n clarityCode: 'd0s2llzfwm',\n statsRoot: 'https://analytics.airwebpass.com/',\n assetsRoot: 'https://eu.assets.ticket.airweb.fr/',\n configURL: 'https://storage.googleapis.com/airweb-ticket-public-europe/back-office/config.json',\n\n adminlabsURL: 'https://europe-west9-ticket-cms.cloudfunctions.net/eu-production-admin-labs',\n adminlabsBackOfficeComponentIds: [\n '960e8d2b-1a1f-11ed-ac0f-000c29d658bc',\n '46060d83-1a1f-11ed-ac0f-000c29d658bc',\n '8c6ae7a8-1a1f-11ed-ac0f-000c29d658bc',\n 'c9a3059d-1c71-11eb-ac0f-000c29d658bc',\n '58f70937-edf3-11ea-ac0f-000c29d658bc',\n '555bc964-2b1f-11eb-ac0f-000c29d658bc',\n '5c638f3c-edf3-11ea-ac0f-000c29d658bc',\n '9356ed0e-2b20-11eb-ac0f-000c29d658bc',\n 'ad9f396b-22ab-11eb-ac0f-000c29d658bc',\n '376398d1-22b0-11eb-ac0f-000c29d658bc',\n '543f8e0a-edf3-11ea-ac0f-000c29d658bc',\n '3d45399d-1a1f-11ed-ac0f-000c29d658bc',\n '79eb74b3-1a1f-11ed-ac0f-000c29d658bc',\n 'c253d2aa-1c71-11eb-ac0f-000c29d658bc',\n '482d9348-edf3-11ea-ac0f-000c29d658bc',\n '844eebf6-22ab-11eb-ac0f-000c29d658bc',\n '37314b6b-6f36-11ee-ac0f-000c29d658bc',\n '9105d282-1c4c-11eb-ac0f-000c29d658bc',\n '4ddd7a70-edf3-11ea-ac0f-000c29d658bc',\n '593e5428-ff10-11ea-ac0f-000c29d658bc',\n '3c2b825f-edf3-11ea-ac0f-000c29d658bc',\n 'b75e5fc8-1c71-11eb-ac0f-000c29d658bc',\n '64312ffe-2b20-11eb-ac0f-000c29d658bc',\n '82d49049-1a1f-11ed-ac0f-000c29d658bc',\n '538eba7c-ff10-11ea-ac0f-000c29d658bc',\n '8c44ded8-22ab-11eb-ac0f-000c29d658bc',\n ],\n airwebStatusUrl: 'https://status.airweb.fr/',\n newAuthenticationUrl: 'https://eu.connect.airwebpass.com/auth',\n authFactorTutoPath:\n 'https://assets.ticket.airweb.dev/administration/documentation/6e4fd5ae-3dd5-11ef-bcc2-2abc636a44d3.pdf',\n maintenanceUrlList: [\n 'https://primary.upkeep.run.airweb.fr/api/alerts',\n 'https://secondary.upkeep.airwebpass.com/api/alerts',\n ],\n\n airwebNewArticleUrl:\n 'https://production-news-module-6lp4usbfna-ew.a.run.app/api/exports/articles',\n matrixDocumentation:\n 'https://www.notion.so/airweb/Template-des-matrices-OD-130ae269a45d807da4ece1bb39d58280',\n },\n};\n","/** Internal global with common properties and Sentry extensions */\n\n// The code below for 'isGlobalObj' and 'GLOBAL_OBJ' was copied from core-js before modification\n// https://github.com/zloirock/core-js/blob/1b944df55282cdc99c90db5f49eb0b6eda2cc0a3/packages/core-js/internals/global.js\n// core-js has the following licence:\n//\n// Copyright (c) 2014-2022 Denis Pushkarev\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n// of this software and associated documentation files (the \"Software\"), to deal\n// in the Software without restriction, including without limitation the rights\n// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell\n// copies of the Software, and to permit persons to whom the Software is\n// furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n// all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n// THE SOFTWARE.\n\n/** Returns 'obj' if it's the global object, otherwise returns undefined */\nfunction isGlobalObj(obj) {\n return obj && obj.Math == Math ? obj : undefined;\n}\n\n/** Get's the global object for the current JavaScript runtime */\nconst GLOBAL_OBJ =\n (typeof globalThis == 'object' && isGlobalObj(globalThis)) ||\n // eslint-disable-next-line no-restricted-globals\n (typeof window == 'object' && isGlobalObj(window)) ||\n (typeof self == 'object' && isGlobalObj(self)) ||\n (typeof global == 'object' && isGlobalObj(global)) ||\n (function () {\n return this;\n })() ||\n {};\n\n/**\n * @deprecated Use GLOBAL_OBJ instead or WINDOW from @sentry/browser. This will be removed in v8\n */\nfunction getGlobalObject() {\n return GLOBAL_OBJ ;\n}\n\n/**\n * Returns a global singleton contained in the global `__SENTRY__` object.\n *\n * If the singleton doesn't already exist in `__SENTRY__`, it will be created using the given factory\n * function and added to the `__SENTRY__` object.\n *\n * @param name name of the global singleton on __SENTRY__\n * @param creator creator Factory function to create the singleton if it doesn't already exist on `__SENTRY__`\n * @param obj (Optional) The global object on which to look for `__SENTRY__`, if not `GLOBAL_OBJ`'s return value\n * @returns the singleton\n */\nfunction getGlobalSingleton(name, creator, obj) {\n const gbl = (obj || GLOBAL_OBJ) ;\n const __SENTRY__ = (gbl.__SENTRY__ = gbl.__SENTRY__ || {});\n const singleton = __SENTRY__[name] || (__SENTRY__[name] = creator());\n return singleton;\n}\n\nexport { GLOBAL_OBJ, getGlobalObject, getGlobalSingleton };\n","const DEFAULT_ENVIRONMENT = 'production';\n\nexport { DEFAULT_ENVIRONMENT };\n","// eslint-disable-next-line @typescript-eslint/unbound-method\nconst objectToString = Object.prototype.toString;\n\n/**\n * Checks whether given value's type is one of a few Error or Error-like\n * {@link isError}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isError(wat) {\n switch (objectToString.call(wat)) {\n case '[object Error]':\n case '[object Exception]':\n case '[object DOMException]':\n return true;\n default:\n return isInstanceOf(wat, Error);\n }\n}\n/**\n * Checks whether given value is an instance of the given built-in class.\n *\n * @param wat The value to be checked\n * @param className\n * @returns A boolean representing the result.\n */\nfunction isBuiltin(wat, className) {\n return objectToString.call(wat) === `[object ${className}]`;\n}\n\n/**\n * Checks whether given value's type is ErrorEvent\n * {@link isErrorEvent}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isErrorEvent(wat) {\n return isBuiltin(wat, 'ErrorEvent');\n}\n\n/**\n * Checks whether given value's type is DOMError\n * {@link isDOMError}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isDOMError(wat) {\n return isBuiltin(wat, 'DOMError');\n}\n\n/**\n * Checks whether given value's type is DOMException\n * {@link isDOMException}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isDOMException(wat) {\n return isBuiltin(wat, 'DOMException');\n}\n\n/**\n * Checks whether given value's type is a string\n * {@link isString}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isString(wat) {\n return isBuiltin(wat, 'String');\n}\n\n/**\n * Checks whether given string is parameterized\n * {@link isParameterizedString}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isParameterizedString(wat) {\n return (\n typeof wat === 'object' &&\n wat !== null &&\n '__sentry_template_string__' in wat &&\n '__sentry_template_values__' in wat\n );\n}\n\n/**\n * Checks whether given value is a primitive (undefined, null, number, boolean, string, bigint, symbol)\n * {@link isPrimitive}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isPrimitive(wat) {\n return wat === null || isParameterizedString(wat) || (typeof wat !== 'object' && typeof wat !== 'function');\n}\n\n/**\n * Checks whether given value's type is an object literal, or a class instance.\n * {@link isPlainObject}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isPlainObject(wat) {\n return isBuiltin(wat, 'Object');\n}\n\n/**\n * Checks whether given value's type is an Event instance\n * {@link isEvent}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isEvent(wat) {\n return typeof Event !== 'undefined' && isInstanceOf(wat, Event);\n}\n\n/**\n * Checks whether given value's type is an Element instance\n * {@link isElement}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isElement(wat) {\n return typeof Element !== 'undefined' && isInstanceOf(wat, Element);\n}\n\n/**\n * Checks whether given value's type is an regexp\n * {@link isRegExp}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isRegExp(wat) {\n return isBuiltin(wat, 'RegExp');\n}\n\n/**\n * Checks whether given value has a then function.\n * @param wat A value to be checked.\n */\nfunction isThenable(wat) {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return Boolean(wat && wat.then && typeof wat.then === 'function');\n}\n\n/**\n * Checks whether given value's type is a SyntheticEvent\n * {@link isSyntheticEvent}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isSyntheticEvent(wat) {\n return isPlainObject(wat) && 'nativeEvent' in wat && 'preventDefault' in wat && 'stopPropagation' in wat;\n}\n\n/**\n * Checks whether given value is NaN\n * {@link isNaN}.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isNaN(wat) {\n return typeof wat === 'number' && wat !== wat;\n}\n\n/**\n * Checks whether given value's type is an instance of provided constructor.\n * {@link isInstanceOf}.\n *\n * @param wat A value to be checked.\n * @param base A constructor to be used in a check.\n * @returns A boolean representing the result.\n */\nfunction isInstanceOf(wat, base) {\n try {\n return wat instanceof base;\n } catch (_e) {\n return false;\n }\n}\n\n/**\n * Checks whether given value's type is a Vue ViewModel.\n *\n * @param wat A value to be checked.\n * @returns A boolean representing the result.\n */\nfunction isVueViewModel(wat) {\n // Not using Object.prototype.toString because in Vue 3 it would read the instance's Symbol(Symbol.toStringTag) property.\n return !!(typeof wat === 'object' && wat !== null && ((wat ).__isVue || (wat )._isVue));\n}\n\nexport { isDOMError, isDOMException, isElement, isError, isErrorEvent, isEvent, isInstanceOf, isNaN, isParameterizedString, isPlainObject, isPrimitive, isRegExp, isString, isSyntheticEvent, isThenable, isVueViewModel };\n","import { isString } from './is.js';\nimport { getGlobalObject } from './worldwide.js';\n\n// eslint-disable-next-line deprecation/deprecation\nconst WINDOW = getGlobalObject();\n\nconst DEFAULT_MAX_STRING_LENGTH = 80;\n\n/**\n * Given a child DOM element, returns a query-selector statement describing that\n * and its ancestors\n * e.g. [HTMLElement] => body > div > input#foo.btn[name=baz]\n * @returns generated DOM path\n */\nfunction htmlTreeAsString(\n elem,\n options = {},\n) {\n if (!elem) {\n return '';\n }\n\n // try/catch both:\n // - accessing event.target (see getsentry/raven-js#838, #768)\n // - `htmlTreeAsString` because it's complex, and just accessing the DOM incorrectly\n // - can throw an exception in some circumstances.\n try {\n let currentElem = elem ;\n const MAX_TRAVERSE_HEIGHT = 5;\n const out = [];\n let height = 0;\n let len = 0;\n const separator = ' > ';\n const sepLength = separator.length;\n let nextStr;\n const keyAttrs = Array.isArray(options) ? options : options.keyAttrs;\n const maxStringLength = (!Array.isArray(options) && options.maxStringLength) || DEFAULT_MAX_STRING_LENGTH;\n\n while (currentElem && height++ < MAX_TRAVERSE_HEIGHT) {\n nextStr = _htmlElementAsString(currentElem, keyAttrs);\n // bail out if\n // - nextStr is the 'html' element\n // - the length of the string that would be created exceeds maxStringLength\n // (ignore this limit if we are on the first iteration)\n if (nextStr === 'html' || (height > 1 && len + out.length * sepLength + nextStr.length >= maxStringLength)) {\n break;\n }\n\n out.push(nextStr);\n\n len += nextStr.length;\n currentElem = currentElem.parentNode;\n }\n\n return out.reverse().join(separator);\n } catch (_oO) {\n return '';\n }\n}\n\n/**\n * Returns a simple, query-selector representation of a DOM element\n * e.g. [HTMLElement] => input#foo.btn[name=baz]\n * @returns generated DOM path\n */\nfunction _htmlElementAsString(el, keyAttrs) {\n const elem = el\n\n;\n\n const out = [];\n let className;\n let classes;\n let key;\n let attr;\n let i;\n\n if (!elem || !elem.tagName) {\n return '';\n }\n\n // @ts-expect-error WINDOW has HTMLElement\n if (WINDOW.HTMLElement) {\n // If using the component name annotation plugin, this value may be available on the DOM node\n if (elem instanceof HTMLElement && elem.dataset && elem.dataset['sentryComponent']) {\n return elem.dataset['sentryComponent'];\n }\n }\n\n out.push(elem.tagName.toLowerCase());\n\n // Pairs of attribute keys defined in `serializeAttribute` and their values on element.\n const keyAttrPairs =\n keyAttrs && keyAttrs.length\n ? keyAttrs.filter(keyAttr => elem.getAttribute(keyAttr)).map(keyAttr => [keyAttr, elem.getAttribute(keyAttr)])\n : null;\n\n if (keyAttrPairs && keyAttrPairs.length) {\n keyAttrPairs.forEach(keyAttrPair => {\n out.push(`[${keyAttrPair[0]}=\"${keyAttrPair[1]}\"]`);\n });\n } else {\n if (elem.id) {\n out.push(`#${elem.id}`);\n }\n\n // eslint-disable-next-line prefer-const\n className = elem.className;\n if (className && isString(className)) {\n classes = className.split(/\\s+/);\n for (i = 0; i < classes.length; i++) {\n out.push(`.${classes[i]}`);\n }\n }\n }\n const allowedAttrs = ['aria-label', 'type', 'name', 'title', 'alt'];\n for (i = 0; i < allowedAttrs.length; i++) {\n key = allowedAttrs[i];\n attr = elem.getAttribute(key);\n if (attr) {\n out.push(`[${key}=\"${attr}\"]`);\n }\n }\n return out.join('');\n}\n\n/**\n * A safe form of location.href\n */\nfunction getLocationHref() {\n try {\n return WINDOW.document.location.href;\n } catch (oO) {\n return '';\n }\n}\n\n/**\n * Gets a DOM element by using document.querySelector.\n *\n * This wrapper will first check for the existance of the function before\n * actually calling it so that we don't have to take care of this check,\n * every time we want to access the DOM.\n *\n * Reason: DOM/querySelector is not available in all environments.\n *\n * We have to cast to any because utils can be consumed by a variety of environments,\n * and we don't want to break TS users. If you know what element will be selected by\n * `document.querySelector`, specify it as part of the generic call. For example,\n * `const element = getDomElement('selector');`\n *\n * @param selector the selector string passed on to document.querySelector\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction getDomElement(selector) {\n if (WINDOW.document && WINDOW.document.querySelector) {\n return WINDOW.document.querySelector(selector) ;\n }\n return null;\n}\n\n/**\n * Given a DOM element, traverses up the tree until it finds the first ancestor node\n * that has the `data-sentry-component` attribute. This attribute is added at build-time\n * by projects that have the component name annotation plugin installed.\n *\n * @returns a string representation of the component for the provided DOM element, or `null` if not found\n */\nfunction getComponentName(elem) {\n // @ts-expect-error WINDOW has HTMLElement\n if (!WINDOW.HTMLElement) {\n return null;\n }\n\n let currentElem = elem ;\n const MAX_TRAVERSE_HEIGHT = 5;\n for (let i = 0; i < MAX_TRAVERSE_HEIGHT; i++) {\n if (!currentElem) {\n return null;\n }\n\n if (currentElem instanceof HTMLElement && currentElem.dataset['sentryComponent']) {\n return currentElem.dataset['sentryComponent'];\n }\n\n currentElem = currentElem.parentNode;\n }\n\n return null;\n}\n\nexport { getComponentName, getDomElement, getLocationHref, htmlTreeAsString };\n","/**\n * This serves as a build time flag that will be true by default, but false in non-debug builds or if users replace `__SENTRY_DEBUG__` in their generated code.\n *\n * ATTENTION: This constant must never cross package boundaries (i.e. be exported) to guarantee that it can be used for tree shaking.\n */\nconst DEBUG_BUILD = (typeof __SENTRY_DEBUG__ === 'undefined' || __SENTRY_DEBUG__);\n\nexport { DEBUG_BUILD };\n","import { DEBUG_BUILD } from './debug-build.js';\nimport { GLOBAL_OBJ } from './worldwide.js';\n\n/** Prefix for logging strings */\nconst PREFIX = 'Sentry Logger ';\n\nconst CONSOLE_LEVELS = [\n 'debug',\n 'info',\n 'warn',\n 'error',\n 'log',\n 'assert',\n 'trace',\n] ;\n\n/** This may be mutated by the console instrumentation. */\nconst originalConsoleMethods\n\n = {};\n\n/** JSDoc */\n\n/**\n * Temporarily disable sentry console instrumentations.\n *\n * @param callback The function to run against the original `console` messages\n * @returns The results of the callback\n */\nfunction consoleSandbox(callback) {\n if (!('console' in GLOBAL_OBJ)) {\n return callback();\n }\n\n const console = GLOBAL_OBJ.console ;\n const wrappedFuncs = {};\n\n const wrappedLevels = Object.keys(originalConsoleMethods) ;\n\n // Restore all wrapped console methods\n wrappedLevels.forEach(level => {\n const originalConsoleMethod = originalConsoleMethods[level] ;\n wrappedFuncs[level] = console[level] ;\n console[level] = originalConsoleMethod;\n });\n\n try {\n return callback();\n } finally {\n // Revert restoration to wrapped state\n wrappedLevels.forEach(level => {\n console[level] = wrappedFuncs[level] ;\n });\n }\n}\n\nfunction makeLogger() {\n let enabled = false;\n const logger = {\n enable: () => {\n enabled = true;\n },\n disable: () => {\n enabled = false;\n },\n isEnabled: () => enabled,\n };\n\n if (DEBUG_BUILD) {\n CONSOLE_LEVELS.forEach(name => {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n logger[name] = (...args) => {\n if (enabled) {\n consoleSandbox(() => {\n GLOBAL_OBJ.console[name](`${PREFIX}[${name}]:`, ...args);\n });\n }\n };\n });\n } else {\n CONSOLE_LEVELS.forEach(name => {\n logger[name] = () => undefined;\n });\n }\n\n return logger ;\n}\n\nconst logger = makeLogger();\n\nexport { CONSOLE_LEVELS, consoleSandbox, logger, originalConsoleMethods };\n","import { isVueViewModel, isString, isRegExp } from './is.js';\n\n/**\n * Truncates given string to the maximum characters count\n *\n * @param str An object that contains serializable values\n * @param max Maximum number of characters in truncated string (0 = unlimited)\n * @returns string Encoded\n */\nfunction truncate(str, max = 0) {\n if (typeof str !== 'string' || max === 0) {\n return str;\n }\n return str.length <= max ? str : `${str.slice(0, max)}...`;\n}\n\n/**\n * This is basically just `trim_line` from\n * https://github.com/getsentry/sentry/blob/master/src/sentry/lang/javascript/processor.py#L67\n *\n * @param str An object that contains serializable values\n * @param max Maximum number of characters in truncated string\n * @returns string Encoded\n */\nfunction snipLine(line, colno) {\n let newLine = line;\n const lineLength = newLine.length;\n if (lineLength <= 150) {\n return newLine;\n }\n if (colno > lineLength) {\n // eslint-disable-next-line no-param-reassign\n colno = lineLength;\n }\n\n let start = Math.max(colno - 60, 0);\n if (start < 5) {\n start = 0;\n }\n\n let end = Math.min(start + 140, lineLength);\n if (end > lineLength - 5) {\n end = lineLength;\n }\n if (end === lineLength) {\n start = Math.max(end - 140, 0);\n }\n\n newLine = newLine.slice(start, end);\n if (start > 0) {\n newLine = `'{snip} ${newLine}`;\n }\n if (end < lineLength) {\n newLine += ' {snip}';\n }\n\n return newLine;\n}\n\n/**\n * Join values in array\n * @param input array of values to be joined together\n * @param delimiter string to be placed in-between values\n * @returns Joined values\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction safeJoin(input, delimiter) {\n if (!Array.isArray(input)) {\n return '';\n }\n\n const output = [];\n // eslint-disable-next-line @typescript-eslint/prefer-for-of\n for (let i = 0; i < input.length; i++) {\n const value = input[i];\n try {\n // This is a hack to fix a Vue3-specific bug that causes an infinite loop of\n // console warnings. This happens when a Vue template is rendered with\n // an undeclared variable, which we try to stringify, ultimately causing\n // Vue to issue another warning which repeats indefinitely.\n // see: https://github.com/getsentry/sentry-javascript/pull/8981\n if (isVueViewModel(value)) {\n output.push('[VueViewModel]');\n } else {\n output.push(String(value));\n }\n } catch (e) {\n output.push('[value cannot be serialized]');\n }\n }\n\n return output.join(delimiter);\n}\n\n/**\n * Checks if the given value matches a regex or string\n *\n * @param value The string to test\n * @param pattern Either a regex or a string against which `value` will be matched\n * @param requireExactStringMatch If true, `value` must match `pattern` exactly. If false, `value` will match\n * `pattern` if it contains `pattern`. Only applies to string-type patterns.\n */\nfunction isMatchingPattern(\n value,\n pattern,\n requireExactStringMatch = false,\n) {\n if (!isString(value)) {\n return false;\n }\n\n if (isRegExp(pattern)) {\n return pattern.test(value);\n }\n if (isString(pattern)) {\n return requireExactStringMatch ? value === pattern : value.includes(pattern);\n }\n\n return false;\n}\n\n/**\n * Test the given string against an array of strings and regexes. By default, string matching is done on a\n * substring-inclusion basis rather than a strict equality basis\n *\n * @param testString The string to test\n * @param patterns The patterns against which to test the string\n * @param requireExactStringMatch If true, `testString` must match one of the given string patterns exactly in order to\n * count. If false, `testString` will match a string pattern if it contains that pattern.\n * @returns\n */\nfunction stringMatchesSomePattern(\n testString,\n patterns = [],\n requireExactStringMatch = false,\n) {\n return patterns.some(pattern => isMatchingPattern(testString, pattern, requireExactStringMatch));\n}\n\nexport { isMatchingPattern, safeJoin, snipLine, stringMatchesSomePattern, truncate };\n","import { htmlTreeAsString } from './browser.js';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { isError, isEvent, isInstanceOf, isElement, isPlainObject, isPrimitive } from './is.js';\nimport { logger } from './logger.js';\nimport { truncate } from './string.js';\n\n/**\n * Replace a method in an object with a wrapped version of itself.\n *\n * @param source An object that contains a method to be wrapped.\n * @param name The name of the method to be wrapped.\n * @param replacementFactory A higher-order function that takes the original version of the given method and returns a\n * wrapped version. Note: The function returned by `replacementFactory` needs to be a non-arrow function, in order to\n * preserve the correct value of `this`, and the original method must be called using `origMethod.call(this, )` or `origMethod.apply(this, [])` (rather than being called directly), again to preserve `this`.\n * @returns void\n */\nfunction fill(source, name, replacementFactory) {\n if (!(name in source)) {\n return;\n }\n\n const original = source[name] ;\n const wrapped = replacementFactory(original) ;\n\n // Make sure it's a function first, as we need to attach an empty prototype for `defineProperties` to work\n // otherwise it'll throw \"TypeError: Object.defineProperties called on non-object\"\n if (typeof wrapped === 'function') {\n markFunctionWrapped(wrapped, original);\n }\n\n source[name] = wrapped;\n}\n\n/**\n * Defines a non-enumerable property on the given object.\n *\n * @param obj The object on which to set the property\n * @param name The name of the property to be set\n * @param value The value to which to set the property\n */\nfunction addNonEnumerableProperty(obj, name, value) {\n try {\n Object.defineProperty(obj, name, {\n // enumerable: false, // the default, so we can save on bundle size by not explicitly setting it\n value: value,\n writable: true,\n configurable: true,\n });\n } catch (o_O) {\n DEBUG_BUILD && logger.log(`Failed to add non-enumerable property \"${name}\" to object`, obj);\n }\n}\n\n/**\n * Remembers the original function on the wrapped function and\n * patches up the prototype.\n *\n * @param wrapped the wrapper function\n * @param original the original function that gets wrapped\n */\nfunction markFunctionWrapped(wrapped, original) {\n try {\n const proto = original.prototype || {};\n wrapped.prototype = original.prototype = proto;\n addNonEnumerableProperty(wrapped, '__sentry_original__', original);\n } catch (o_O) {} // eslint-disable-line no-empty\n}\n\n/**\n * This extracts the original function if available. See\n * `markFunctionWrapped` for more information.\n *\n * @param func the function to unwrap\n * @returns the unwrapped version of the function if available.\n */\nfunction getOriginalFunction(func) {\n return func.__sentry_original__;\n}\n\n/**\n * Encodes given object into url-friendly format\n *\n * @param object An object that contains serializable values\n * @returns string Encoded\n */\nfunction urlEncode(object) {\n return Object.keys(object)\n .map(key => `${encodeURIComponent(key)}=${encodeURIComponent(object[key])}`)\n .join('&');\n}\n\n/**\n * Transforms any `Error` or `Event` into a plain object with all of their enumerable properties, and some of their\n * non-enumerable properties attached.\n *\n * @param value Initial source that we have to transform in order for it to be usable by the serializer\n * @returns An Event or Error turned into an object - or the value argurment itself, when value is neither an Event nor\n * an Error.\n */\nfunction convertToPlainObject(\n value,\n)\n\n {\n if (isError(value)) {\n return {\n message: value.message,\n name: value.name,\n stack: value.stack,\n ...getOwnProperties(value),\n };\n } else if (isEvent(value)) {\n const newObj\n\n = {\n type: value.type,\n target: serializeEventTarget(value.target),\n currentTarget: serializeEventTarget(value.currentTarget),\n ...getOwnProperties(value),\n };\n\n if (typeof CustomEvent !== 'undefined' && isInstanceOf(value, CustomEvent)) {\n newObj.detail = value.detail;\n }\n\n return newObj;\n } else {\n return value;\n }\n}\n\n/** Creates a string representation of the target of an `Event` object */\nfunction serializeEventTarget(target) {\n try {\n return isElement(target) ? htmlTreeAsString(target) : Object.prototype.toString.call(target);\n } catch (_oO) {\n return '';\n }\n}\n\n/** Filters out all but an object's own properties */\nfunction getOwnProperties(obj) {\n if (typeof obj === 'object' && obj !== null) {\n const extractedProps = {};\n for (const property in obj) {\n if (Object.prototype.hasOwnProperty.call(obj, property)) {\n extractedProps[property] = (obj )[property];\n }\n }\n return extractedProps;\n } else {\n return {};\n }\n}\n\n/**\n * Given any captured exception, extract its keys and create a sorted\n * and truncated list that will be used inside the event message.\n * eg. `Non-error exception captured with keys: foo, bar, baz`\n */\nfunction extractExceptionKeysForMessage(exception, maxLength = 40) {\n const keys = Object.keys(convertToPlainObject(exception));\n keys.sort();\n\n if (!keys.length) {\n return '[object has no keys]';\n }\n\n if (keys[0].length >= maxLength) {\n return truncate(keys[0], maxLength);\n }\n\n for (let includedKeys = keys.length; includedKeys > 0; includedKeys--) {\n const serialized = keys.slice(0, includedKeys).join(', ');\n if (serialized.length > maxLength) {\n continue;\n }\n if (includedKeys === keys.length) {\n return serialized;\n }\n return truncate(serialized, maxLength);\n }\n\n return '';\n}\n\n/**\n * Given any object, return a new object having removed all fields whose value was `undefined`.\n * Works recursively on objects and arrays.\n *\n * Attention: This function keeps circular references in the returned object.\n */\nfunction dropUndefinedKeys(inputValue) {\n // This map keeps track of what already visited nodes map to.\n // Our Set - based memoBuilder doesn't work here because we want to the output object to have the same circular\n // references as the input object.\n const memoizationMap = new Map();\n\n // This function just proxies `_dropUndefinedKeys` to keep the `memoBuilder` out of this function's API\n return _dropUndefinedKeys(inputValue, memoizationMap);\n}\n\nfunction _dropUndefinedKeys(inputValue, memoizationMap) {\n if (isPojo(inputValue)) {\n // If this node has already been visited due to a circular reference, return the object it was mapped to in the new object\n const memoVal = memoizationMap.get(inputValue);\n if (memoVal !== undefined) {\n return memoVal ;\n }\n\n const returnValue = {};\n // Store the mapping of this value in case we visit it again, in case of circular data\n memoizationMap.set(inputValue, returnValue);\n\n for (const key of Object.keys(inputValue)) {\n if (typeof inputValue[key] !== 'undefined') {\n returnValue[key] = _dropUndefinedKeys(inputValue[key], memoizationMap);\n }\n }\n\n return returnValue ;\n }\n\n if (Array.isArray(inputValue)) {\n // If this node has already been visited due to a circular reference, return the array it was mapped to in the new object\n const memoVal = memoizationMap.get(inputValue);\n if (memoVal !== undefined) {\n return memoVal ;\n }\n\n const returnValue = [];\n // Store the mapping of this value in case we visit it again, in case of circular data\n memoizationMap.set(inputValue, returnValue);\n\n inputValue.forEach((item) => {\n returnValue.push(_dropUndefinedKeys(item, memoizationMap));\n });\n\n return returnValue ;\n }\n\n return inputValue;\n}\n\nfunction isPojo(input) {\n if (!isPlainObject(input)) {\n return false;\n }\n\n try {\n const name = (Object.getPrototypeOf(input) ).constructor.name;\n return !name || name === 'Object';\n } catch (e) {\n return true;\n }\n}\n\n/**\n * Ensure that something is an object.\n *\n * Turns `undefined` and `null` into `String`s and all other primitives into instances of their respective wrapper\n * classes (String, Boolean, Number, etc.). Acts as the identity function on non-primitives.\n *\n * @param wat The subject of the objectification\n * @returns A version of `wat` which can safely be used with `Object` class methods\n */\nfunction objectify(wat) {\n let objectified;\n switch (true) {\n case wat === undefined || wat === null:\n objectified = new String(wat);\n break;\n\n // Though symbols and bigints do have wrapper classes (`Symbol` and `BigInt`, respectively), for whatever reason\n // those classes don't have constructors which can be used with the `new` keyword. We therefore need to cast each as\n // an object in order to wrap it.\n case typeof wat === 'symbol' || typeof wat === 'bigint':\n objectified = Object(wat);\n break;\n\n // this will catch the remaining primitives: `String`, `Number`, and `Boolean`\n case isPrimitive(wat):\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n objectified = new (wat ).constructor(wat);\n break;\n\n // by process of elimination, at this point we know that `wat` must already be an object\n default:\n objectified = wat;\n break;\n }\n return objectified;\n}\n\nexport { addNonEnumerableProperty, convertToPlainObject, dropUndefinedKeys, extractExceptionKeysForMessage, fill, getOriginalFunction, markFunctionWrapped, objectify, urlEncode };\n","import { addNonEnumerableProperty } from './object.js';\nimport { snipLine } from './string.js';\nimport { GLOBAL_OBJ } from './worldwide.js';\n\n/**\n * UUID4 generator\n *\n * @returns string Generated UUID4.\n */\nfunction uuid4() {\n const gbl = GLOBAL_OBJ ;\n const crypto = gbl.crypto || gbl.msCrypto;\n\n let getRandomByte = () => Math.random() * 16;\n try {\n if (crypto && crypto.randomUUID) {\n return crypto.randomUUID().replace(/-/g, '');\n }\n if (crypto && crypto.getRandomValues) {\n getRandomByte = () => {\n // crypto.getRandomValues might return undefined instead of the typed array\n // in old Chromium versions (e.g. 23.0.1235.0 (151422))\n // However, `typedArray` is still filled in-place.\n // @see https://developer.mozilla.org/en-US/docs/Web/API/Crypto/getRandomValues#typedarray\n const typedArray = new Uint8Array(1);\n crypto.getRandomValues(typedArray);\n return typedArray[0];\n };\n }\n } catch (_) {\n // some runtimes can crash invoking crypto\n // https://github.com/getsentry/sentry-javascript/issues/8935\n }\n\n // http://stackoverflow.com/questions/105034/how-to-create-a-guid-uuid-in-javascript/2117523#2117523\n // Concatenating the following numbers as strings results in '10000000100040008000100000000000'\n return (([1e7] ) + 1e3 + 4e3 + 8e3 + 1e11).replace(/[018]/g, c =>\n // eslint-disable-next-line no-bitwise\n ((c ) ^ ((getRandomByte() & 15) >> ((c ) / 4))).toString(16),\n );\n}\n\nfunction getFirstException(event) {\n return event.exception && event.exception.values ? event.exception.values[0] : undefined;\n}\n\n/**\n * Extracts either message or type+value from an event that can be used for user-facing logs\n * @returns event's description\n */\nfunction getEventDescription(event) {\n const { message, event_id: eventId } = event;\n if (message) {\n return message;\n }\n\n const firstException = getFirstException(event);\n if (firstException) {\n if (firstException.type && firstException.value) {\n return `${firstException.type}: ${firstException.value}`;\n }\n return firstException.type || firstException.value || eventId || '';\n }\n return eventId || '';\n}\n\n/**\n * Adds exception values, type and value to an synthetic Exception.\n * @param event The event to modify.\n * @param value Value of the exception.\n * @param type Type of the exception.\n * @hidden\n */\nfunction addExceptionTypeValue(event, value, type) {\n const exception = (event.exception = event.exception || {});\n const values = (exception.values = exception.values || []);\n const firstException = (values[0] = values[0] || {});\n if (!firstException.value) {\n firstException.value = value || '';\n }\n if (!firstException.type) {\n firstException.type = type || 'Error';\n }\n}\n\n/**\n * Adds exception mechanism data to a given event. Uses defaults if the second parameter is not passed.\n *\n * @param event The event to modify.\n * @param newMechanism Mechanism data to add to the event.\n * @hidden\n */\nfunction addExceptionMechanism(event, newMechanism) {\n const firstException = getFirstException(event);\n if (!firstException) {\n return;\n }\n\n const defaultMechanism = { type: 'generic', handled: true };\n const currentMechanism = firstException.mechanism;\n firstException.mechanism = { ...defaultMechanism, ...currentMechanism, ...newMechanism };\n\n if (newMechanism && 'data' in newMechanism) {\n const mergedData = { ...(currentMechanism && currentMechanism.data), ...newMechanism.data };\n firstException.mechanism.data = mergedData;\n }\n}\n\n// https://semver.org/#is-there-a-suggested-regular-expression-regex-to-check-a-semver-string\nconst SEMVER_REGEXP =\n /^(0|[1-9]\\d*)\\.(0|[1-9]\\d*)\\.(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$/;\n\n/**\n * Represents Semantic Versioning object\n */\n\n/**\n * Parses input into a SemVer interface\n * @param input string representation of a semver version\n */\nfunction parseSemver(input) {\n const match = input.match(SEMVER_REGEXP) || [];\n const major = parseInt(match[1], 10);\n const minor = parseInt(match[2], 10);\n const patch = parseInt(match[3], 10);\n return {\n buildmetadata: match[5],\n major: isNaN(major) ? undefined : major,\n minor: isNaN(minor) ? undefined : minor,\n patch: isNaN(patch) ? undefined : patch,\n prerelease: match[4],\n };\n}\n\n/**\n * This function adds context (pre/post/line) lines to the provided frame\n *\n * @param lines string[] containing all lines\n * @param frame StackFrame that will be mutated\n * @param linesOfContext number of context lines we want to add pre/post\n */\nfunction addContextToFrame(lines, frame, linesOfContext = 5) {\n // When there is no line number in the frame, attaching context is nonsensical and will even break grouping\n if (frame.lineno === undefined) {\n return;\n }\n\n const maxLines = lines.length;\n const sourceLine = Math.max(Math.min(maxLines - 1, frame.lineno - 1), 0);\n\n frame.pre_context = lines\n .slice(Math.max(0, sourceLine - linesOfContext), sourceLine)\n .map((line) => snipLine(line, 0));\n\n frame.context_line = snipLine(lines[Math.min(maxLines - 1, sourceLine)], frame.colno || 0);\n\n frame.post_context = lines\n .slice(Math.min(sourceLine + 1, maxLines), sourceLine + 1 + linesOfContext)\n .map((line) => snipLine(line, 0));\n}\n\n/**\n * Checks whether or not we've already captured the given exception (note: not an identical exception - the very object\n * in question), and marks it captured if not.\n *\n * This is useful because it's possible for an error to get captured by more than one mechanism. After we intercept and\n * record an error, we rethrow it (assuming we've intercepted it before it's reached the top-level global handlers), so\n * that we don't interfere with whatever effects the error might have had were the SDK not there. At that point, because\n * the error has been rethrown, it's possible for it to bubble up to some other code we've instrumented. If it's not\n * caught after that, it will bubble all the way up to the global handlers (which of course we also instrument). This\n * function helps us ensure that even if we encounter the same error more than once, we only record it the first time we\n * see it.\n *\n * Note: It will ignore primitives (always return `false` and not mark them as seen), as properties can't be set on\n * them. {@link: Object.objectify} can be used on exceptions to convert any that are primitives into their equivalent\n * object wrapper forms so that this check will always work. However, because we need to flag the exact object which\n * will get rethrown, and because that rethrowing happens outside of the event processing pipeline, the objectification\n * must be done before the exception captured.\n *\n * @param A thrown exception to check or flag as having been seen\n * @returns `true` if the exception has already been captured, `false` if not (with the side effect of marking it seen)\n */\nfunction checkOrSetAlreadyCaught(exception) {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n if (exception && (exception ).__sentry_captured__) {\n return true;\n }\n\n try {\n // set it this way rather than by assignment so that it's not ennumerable and therefore isn't recorded by the\n // `ExtraErrorData` integration\n addNonEnumerableProperty(exception , '__sentry_captured__', true);\n } catch (err) {\n // `exception` is a primitive, so we can't mark it seen\n }\n\n return false;\n}\n\n/**\n * Checks whether the given input is already an array, and if it isn't, wraps it in one.\n *\n * @param maybeArray Input to turn into an array, if necessary\n * @returns The input, if already an array, or an array with the input as the only element, if not\n */\nfunction arrayify(maybeArray) {\n return Array.isArray(maybeArray) ? maybeArray : [maybeArray];\n}\n\nexport { addContextToFrame, addExceptionMechanism, addExceptionTypeValue, arrayify, checkOrSetAlreadyCaught, getEventDescription, parseSemver, uuid4 };\n","import { GLOBAL_OBJ } from './worldwide.js';\n\nconst ONE_SECOND_IN_MS = 1000;\n\n/**\n * A partial definition of the [Performance Web API]{@link https://developer.mozilla.org/en-US/docs/Web/API/Performance}\n * for accessing a high-resolution monotonic clock.\n */\n\n/**\n * Returns a timestamp in seconds since the UNIX epoch using the Date API.\n *\n * TODO(v8): Return type should be rounded.\n */\nfunction dateTimestampInSeconds() {\n return Date.now() / ONE_SECOND_IN_MS;\n}\n\n/**\n * Returns a wrapper around the native Performance API browser implementation, or undefined for browsers that do not\n * support the API.\n *\n * Wrapping the native API works around differences in behavior from different browsers.\n */\nfunction createUnixTimestampInSecondsFunc() {\n const { performance } = GLOBAL_OBJ ;\n if (!performance || !performance.now) {\n return dateTimestampInSeconds;\n }\n\n // Some browser and environments don't have a timeOrigin, so we fallback to\n // using Date.now() to compute the starting time.\n const approxStartingTimeOrigin = Date.now() - performance.now();\n const timeOrigin = performance.timeOrigin == undefined ? approxStartingTimeOrigin : performance.timeOrigin;\n\n // performance.now() is a monotonic clock, which means it starts at 0 when the process begins. To get the current\n // wall clock time (actual UNIX timestamp), we need to add the starting time origin and the current time elapsed.\n //\n // TODO: This does not account for the case where the monotonic clock that powers performance.now() drifts from the\n // wall clock time, which causes the returned timestamp to be inaccurate. We should investigate how to detect and\n // correct for this.\n // See: https://github.com/getsentry/sentry-javascript/issues/2590\n // See: https://github.com/mdn/content/issues/4713\n // See: https://dev.to/noamr/when-a-millisecond-is-not-a-millisecond-3h6\n return () => {\n return (timeOrigin + performance.now()) / ONE_SECOND_IN_MS;\n };\n}\n\n/**\n * Returns a timestamp in seconds since the UNIX epoch using either the Performance or Date APIs, depending on the\n * availability of the Performance API.\n *\n * BUG: Note that because of how browsers implement the Performance API, the clock might stop when the computer is\n * asleep. This creates a skew between `dateTimestampInSeconds` and `timestampInSeconds`. The\n * skew can grow to arbitrary amounts like days, weeks or months.\n * See https://github.com/getsentry/sentry-javascript/issues/2590.\n */\nconst timestampInSeconds = createUnixTimestampInSecondsFunc();\n\n/**\n * Re-exported with an old name for backwards-compatibility.\n * TODO (v8): Remove this\n *\n * @deprecated Use `timestampInSeconds` instead.\n */\nconst timestampWithMs = timestampInSeconds;\n\n/**\n * Internal helper to store what is the source of browserPerformanceTimeOrigin below. For debugging only.\n */\nlet _browserPerformanceTimeOriginMode;\n\n/**\n * The number of milliseconds since the UNIX epoch. This value is only usable in a browser, and only when the\n * performance API is available.\n */\nconst browserPerformanceTimeOrigin = (() => {\n // Unfortunately browsers may report an inaccurate time origin data, through either performance.timeOrigin or\n // performance.timing.navigationStart, which results in poor results in performance data. We only treat time origin\n // data as reliable if they are within a reasonable threshold of the current time.\n\n const { performance } = GLOBAL_OBJ ;\n if (!performance || !performance.now) {\n _browserPerformanceTimeOriginMode = 'none';\n return undefined;\n }\n\n const threshold = 3600 * 1000;\n const performanceNow = performance.now();\n const dateNow = Date.now();\n\n // if timeOrigin isn't available set delta to threshold so it isn't used\n const timeOriginDelta = performance.timeOrigin\n ? Math.abs(performance.timeOrigin + performanceNow - dateNow)\n : threshold;\n const timeOriginIsReliable = timeOriginDelta < threshold;\n\n // While performance.timing.navigationStart is deprecated in favor of performance.timeOrigin, performance.timeOrigin\n // is not as widely supported. Namely, performance.timeOrigin is undefined in Safari as of writing.\n // Also as of writing, performance.timing is not available in Web Workers in mainstream browsers, so it is not always\n // a valid fallback. In the absence of an initial time provided by the browser, fallback to the current time from the\n // Date API.\n // eslint-disable-next-line deprecation/deprecation\n const navigationStart = performance.timing && performance.timing.navigationStart;\n const hasNavigationStart = typeof navigationStart === 'number';\n // if navigationStart isn't available set delta to threshold so it isn't used\n const navigationStartDelta = hasNavigationStart ? Math.abs(navigationStart + performanceNow - dateNow) : threshold;\n const navigationStartIsReliable = navigationStartDelta < threshold;\n\n if (timeOriginIsReliable || navigationStartIsReliable) {\n // Use the more reliable time origin\n if (timeOriginDelta <= navigationStartDelta) {\n _browserPerformanceTimeOriginMode = 'timeOrigin';\n return performance.timeOrigin;\n } else {\n _browserPerformanceTimeOriginMode = 'navigationStart';\n return navigationStart;\n }\n }\n\n // Either both timeOrigin and navigationStart are skewed or neither is available, fallback to Date.\n _browserPerformanceTimeOriginMode = 'dateNow';\n return dateNow;\n})();\n\nexport { _browserPerformanceTimeOriginMode, browserPerformanceTimeOrigin, dateTimestampInSeconds, timestampInSeconds, timestampWithMs };\n","/**\n * This serves as a build time flag that will be true by default, but false in non-debug builds or if users replace `__SENTRY_DEBUG__` in their generated code.\n *\n * ATTENTION: This constant must never cross package boundaries (i.e. be exported) to guarantee that it can be used for tree shaking.\n */\nconst DEBUG_BUILD = (typeof __SENTRY_DEBUG__ === 'undefined' || __SENTRY_DEBUG__);\n\nexport { DEBUG_BUILD };\n","import { isThenable } from './is.js';\n\n/* eslint-disable @typescript-eslint/explicit-function-return-type */\n\n/** SyncPromise internal states */\nvar States; (function (States) {\n /** Pending */\n const PENDING = 0; States[States[\"PENDING\"] = PENDING] = \"PENDING\";\n /** Resolved / OK */\n const RESOLVED = 1; States[States[\"RESOLVED\"] = RESOLVED] = \"RESOLVED\";\n /** Rejected / Error */\n const REJECTED = 2; States[States[\"REJECTED\"] = REJECTED] = \"REJECTED\";\n})(States || (States = {}));\n\n// Overloads so we can call resolvedSyncPromise without arguments and generic argument\n\n/**\n * Creates a resolved sync promise.\n *\n * @param value the value to resolve the promise with\n * @returns the resolved sync promise\n */\nfunction resolvedSyncPromise(value) {\n return new SyncPromise(resolve => {\n resolve(value);\n });\n}\n\n/**\n * Creates a rejected sync promise.\n *\n * @param value the value to reject the promise with\n * @returns the rejected sync promise\n */\nfunction rejectedSyncPromise(reason) {\n return new SyncPromise((_, reject) => {\n reject(reason);\n });\n}\n\n/**\n * Thenable class that behaves like a Promise and follows it's interface\n * but is not async internally\n */\nclass SyncPromise {\n\n constructor(\n executor,\n ) {SyncPromise.prototype.__init.call(this);SyncPromise.prototype.__init2.call(this);SyncPromise.prototype.__init3.call(this);SyncPromise.prototype.__init4.call(this);\n this._state = States.PENDING;\n this._handlers = [];\n\n try {\n executor(this._resolve, this._reject);\n } catch (e) {\n this._reject(e);\n }\n }\n\n /** JSDoc */\n then(\n onfulfilled,\n onrejected,\n ) {\n return new SyncPromise((resolve, reject) => {\n this._handlers.push([\n false,\n result => {\n if (!onfulfilled) {\n // TODO: ¯\\_(ツ)_/¯\n // TODO: FIXME\n resolve(result );\n } else {\n try {\n resolve(onfulfilled(result));\n } catch (e) {\n reject(e);\n }\n }\n },\n reason => {\n if (!onrejected) {\n reject(reason);\n } else {\n try {\n resolve(onrejected(reason));\n } catch (e) {\n reject(e);\n }\n }\n },\n ]);\n this._executeHandlers();\n });\n }\n\n /** JSDoc */\n catch(\n onrejected,\n ) {\n return this.then(val => val, onrejected);\n }\n\n /** JSDoc */\n finally(onfinally) {\n return new SyncPromise((resolve, reject) => {\n let val;\n let isRejected;\n\n return this.then(\n value => {\n isRejected = false;\n val = value;\n if (onfinally) {\n onfinally();\n }\n },\n reason => {\n isRejected = true;\n val = reason;\n if (onfinally) {\n onfinally();\n }\n },\n ).then(() => {\n if (isRejected) {\n reject(val);\n return;\n }\n\n resolve(val );\n });\n });\n }\n\n /** JSDoc */\n __init() {this._resolve = (value) => {\n this._setResult(States.RESOLVED, value);\n };}\n\n /** JSDoc */\n __init2() {this._reject = (reason) => {\n this._setResult(States.REJECTED, reason);\n };}\n\n /** JSDoc */\n __init3() {this._setResult = (state, value) => {\n if (this._state !== States.PENDING) {\n return;\n }\n\n if (isThenable(value)) {\n void (value ).then(this._resolve, this._reject);\n return;\n }\n\n this._state = state;\n this._value = value;\n\n this._executeHandlers();\n };}\n\n /** JSDoc */\n __init4() {this._executeHandlers = () => {\n if (this._state === States.PENDING) {\n return;\n }\n\n const cachedHandlers = this._handlers.slice();\n this._handlers = [];\n\n cachedHandlers.forEach(handler => {\n if (handler[0]) {\n return;\n }\n\n if (this._state === States.RESOLVED) {\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n handler[1](this._value );\n }\n\n if (this._state === States.REJECTED) {\n handler[2](this._value);\n }\n\n handler[0] = true;\n });\n };}\n}\n\nexport { SyncPromise, rejectedSyncPromise, resolvedSyncPromise };\n","import { SyncPromise, logger, isThenable, getGlobalSingleton } from '@sentry/utils';\nimport { DEBUG_BUILD } from './debug-build.js';\n\n/**\n * Returns the global event processors.\n * @deprecated Global event processors will be removed in v8.\n */\nfunction getGlobalEventProcessors() {\n return getGlobalSingleton('globalEventProcessors', () => []);\n}\n\n/**\n * Add a EventProcessor to be kept globally.\n * @deprecated Use `addEventProcessor` instead. Global event processors will be removed in v8.\n */\nfunction addGlobalEventProcessor(callback) {\n // eslint-disable-next-line deprecation/deprecation\n getGlobalEventProcessors().push(callback);\n}\n\n/**\n * Process an array of event processors, returning the processed event (or `null` if the event was dropped).\n */\nfunction notifyEventProcessors(\n processors,\n event,\n hint,\n index = 0,\n) {\n return new SyncPromise((resolve, reject) => {\n const processor = processors[index];\n if (event === null || typeof processor !== 'function') {\n resolve(event);\n } else {\n const result = processor({ ...event }, hint) ;\n\n DEBUG_BUILD && processor.id && result === null && logger.log(`Event processor \"${processor.id}\" dropped event`);\n\n if (isThenable(result)) {\n void result\n .then(final => notifyEventProcessors(processors, final, hint, index + 1).then(resolve))\n .then(null, reject);\n } else {\n void notifyEventProcessors(processors, result, hint, index + 1)\n .then(resolve)\n .then(null, reject);\n }\n }\n });\n}\n\nexport { addGlobalEventProcessor, getGlobalEventProcessors, notifyEventProcessors };\n","import { timestampInSeconds, uuid4, dropUndefinedKeys } from '@sentry/utils';\n\n/**\n * Creates a new `Session` object by setting certain default parameters. If optional @param context\n * is passed, the passed properties are applied to the session object.\n *\n * @param context (optional) additional properties to be applied to the returned session object\n *\n * @returns a new `Session` object\n */\nfunction makeSession(context) {\n // Both timestamp and started are in seconds since the UNIX epoch.\n const startingTime = timestampInSeconds();\n\n const session = {\n sid: uuid4(),\n init: true,\n timestamp: startingTime,\n started: startingTime,\n duration: 0,\n status: 'ok',\n errors: 0,\n ignoreDuration: false,\n toJSON: () => sessionToJSON(session),\n };\n\n if (context) {\n updateSession(session, context);\n }\n\n return session;\n}\n\n/**\n * Updates a session object with the properties passed in the context.\n *\n * Note that this function mutates the passed object and returns void.\n * (Had to do this instead of returning a new and updated session because closing and sending a session\n * makes an update to the session after it was passed to the sending logic.\n * @see BaseClient.captureSession )\n *\n * @param session the `Session` to update\n * @param context the `SessionContext` holding the properties that should be updated in @param session\n */\n// eslint-disable-next-line complexity\nfunction updateSession(session, context = {}) {\n if (context.user) {\n if (!session.ipAddress && context.user.ip_address) {\n session.ipAddress = context.user.ip_address;\n }\n\n if (!session.did && !context.did) {\n session.did = context.user.id || context.user.email || context.user.username;\n }\n }\n\n session.timestamp = context.timestamp || timestampInSeconds();\n\n if (context.abnormal_mechanism) {\n session.abnormal_mechanism = context.abnormal_mechanism;\n }\n\n if (context.ignoreDuration) {\n session.ignoreDuration = context.ignoreDuration;\n }\n if (context.sid) {\n // Good enough uuid validation. — Kamil\n session.sid = context.sid.length === 32 ? context.sid : uuid4();\n }\n if (context.init !== undefined) {\n session.init = context.init;\n }\n if (!session.did && context.did) {\n session.did = `${context.did}`;\n }\n if (typeof context.started === 'number') {\n session.started = context.started;\n }\n if (session.ignoreDuration) {\n session.duration = undefined;\n } else if (typeof context.duration === 'number') {\n session.duration = context.duration;\n } else {\n const duration = session.timestamp - session.started;\n session.duration = duration >= 0 ? duration : 0;\n }\n if (context.release) {\n session.release = context.release;\n }\n if (context.environment) {\n session.environment = context.environment;\n }\n if (!session.ipAddress && context.ipAddress) {\n session.ipAddress = context.ipAddress;\n }\n if (!session.userAgent && context.userAgent) {\n session.userAgent = context.userAgent;\n }\n if (typeof context.errors === 'number') {\n session.errors = context.errors;\n }\n if (context.status) {\n session.status = context.status;\n }\n}\n\n/**\n * Closes a session by setting its status and updating the session object with it.\n * Internally calls `updateSession` to update the passed session object.\n *\n * Note that this function mutates the passed session (@see updateSession for explanation).\n *\n * @param session the `Session` object to be closed\n * @param status the `SessionStatus` with which the session was closed. If you don't pass a status,\n * this function will keep the previously set status, unless it was `'ok'` in which case\n * it is changed to `'exited'`.\n */\nfunction closeSession(session, status) {\n let context = {};\n if (status) {\n context = { status };\n } else if (session.status === 'ok') {\n context = { status: 'exited' };\n }\n\n updateSession(session, context);\n}\n\n/**\n * Serializes a passed session object to a JSON object with a slightly different structure.\n * This is necessary because the Sentry backend requires a slightly different schema of a session\n * than the one the JS SDKs use internally.\n *\n * @param session the session to be converted\n *\n * @returns a JSON object of the passed session\n */\nfunction sessionToJSON(session) {\n return dropUndefinedKeys({\n sid: `${session.sid}`,\n init: session.init,\n // Make sure that sec is converted to ms for date constructor\n started: new Date(session.started * 1000).toISOString(),\n timestamp: new Date(session.timestamp * 1000).toISOString(),\n status: session.status,\n errors: session.errors,\n did: typeof session.did === 'number' || typeof session.did === 'string' ? `${session.did}` : undefined,\n duration: session.duration,\n abnormal_mechanism: session.abnormal_mechanism,\n attrs: {\n release: session.release,\n environment: session.environment,\n ip_address: session.ipAddress,\n user_agent: session.userAgent,\n },\n });\n}\n\nexport { closeSession, makeSession, updateSession };\n","/**\n * Returns the root span of a given span.\n *\n * As long as we use `Transaction`s internally, the returned root span\n * will be a `Transaction` but be aware that this might change in the future.\n *\n * If the given span has no root span or transaction, `undefined` is returned.\n */\nfunction getRootSpan(span) {\n // TODO (v8): Remove this check and just return span\n // eslint-disable-next-line deprecation/deprecation\n return span.transaction;\n}\n\nexport { getRootSpan };\n","import { DEBUG_BUILD } from './debug-build.js';\nimport { isString } from './is.js';\nimport { logger } from './logger.js';\n\nconst BAGGAGE_HEADER_NAME = 'baggage';\n\nconst SENTRY_BAGGAGE_KEY_PREFIX = 'sentry-';\n\nconst SENTRY_BAGGAGE_KEY_PREFIX_REGEX = /^sentry-/;\n\n/**\n * Max length of a serialized baggage string\n *\n * https://www.w3.org/TR/baggage/#limits\n */\nconst MAX_BAGGAGE_STRING_LENGTH = 8192;\n\n/**\n * Takes a baggage header and turns it into Dynamic Sampling Context, by extracting all the \"sentry-\" prefixed values\n * from it.\n *\n * @param baggageHeader A very bread definition of a baggage header as it might appear in various frameworks.\n * @returns The Dynamic Sampling Context that was found on `baggageHeader`, if there was any, `undefined` otherwise.\n */\nfunction baggageHeaderToDynamicSamplingContext(\n // Very liberal definition of what any incoming header might look like\n baggageHeader,\n) {\n if (!isString(baggageHeader) && !Array.isArray(baggageHeader)) {\n return undefined;\n }\n\n // Intermediary object to store baggage key value pairs of incoming baggage headers on.\n // It is later used to read Sentry-DSC-values from.\n let baggageObject = {};\n\n if (Array.isArray(baggageHeader)) {\n // Combine all baggage headers into one object containing the baggage values so we can later read the Sentry-DSC-values from it\n baggageObject = baggageHeader.reduce((acc, curr) => {\n const currBaggageObject = baggageHeaderToObject(curr);\n for (const key of Object.keys(currBaggageObject)) {\n acc[key] = currBaggageObject[key];\n }\n return acc;\n }, {});\n } else {\n // Return undefined if baggage header is an empty string (technically an empty baggage header is not spec conform but\n // this is how we choose to handle it)\n if (!baggageHeader) {\n return undefined;\n }\n\n baggageObject = baggageHeaderToObject(baggageHeader);\n }\n\n // Read all \"sentry-\" prefixed values out of the baggage object and put it onto a dynamic sampling context object.\n const dynamicSamplingContext = Object.entries(baggageObject).reduce((acc, [key, value]) => {\n if (key.match(SENTRY_BAGGAGE_KEY_PREFIX_REGEX)) {\n const nonPrefixedKey = key.slice(SENTRY_BAGGAGE_KEY_PREFIX.length);\n acc[nonPrefixedKey] = value;\n }\n return acc;\n }, {});\n\n // Only return a dynamic sampling context object if there are keys in it.\n // A keyless object means there were no sentry values on the header, which means that there is no DSC.\n if (Object.keys(dynamicSamplingContext).length > 0) {\n return dynamicSamplingContext ;\n } else {\n return undefined;\n }\n}\n\n/**\n * Turns a Dynamic Sampling Object into a baggage header by prefixing all the keys on the object with \"sentry-\".\n *\n * @param dynamicSamplingContext The Dynamic Sampling Context to turn into a header. For convenience and compatibility\n * with the `getDynamicSamplingContext` method on the Transaction class ,this argument can also be `undefined`. If it is\n * `undefined` the function will return `undefined`.\n * @returns a baggage header, created from `dynamicSamplingContext`, or `undefined` either if `dynamicSamplingContext`\n * was `undefined`, or if `dynamicSamplingContext` didn't contain any values.\n */\nfunction dynamicSamplingContextToSentryBaggageHeader(\n // this also takes undefined for convenience and bundle size in other places\n dynamicSamplingContext,\n) {\n if (!dynamicSamplingContext) {\n return undefined;\n }\n\n // Prefix all DSC keys with \"sentry-\" and put them into a new object\n const sentryPrefixedDSC = Object.entries(dynamicSamplingContext).reduce(\n (acc, [dscKey, dscValue]) => {\n if (dscValue) {\n acc[`${SENTRY_BAGGAGE_KEY_PREFIX}${dscKey}`] = dscValue;\n }\n return acc;\n },\n {},\n );\n\n return objectToBaggageHeader(sentryPrefixedDSC);\n}\n\n/**\n * Will parse a baggage header, which is a simple key-value map, into a flat object.\n *\n * @param baggageHeader The baggage header to parse.\n * @returns a flat object containing all the key-value pairs from `baggageHeader`.\n */\nfunction baggageHeaderToObject(baggageHeader) {\n return baggageHeader\n .split(',')\n .map(baggageEntry => baggageEntry.split('=').map(keyOrValue => decodeURIComponent(keyOrValue.trim())))\n .reduce((acc, [key, value]) => {\n acc[key] = value;\n return acc;\n }, {});\n}\n\n/**\n * Turns a flat object (key-value pairs) into a baggage header, which is also just key-value pairs.\n *\n * @param object The object to turn into a baggage header.\n * @returns a baggage header string, or `undefined` if the object didn't have any values, since an empty baggage header\n * is not spec compliant.\n */\nfunction objectToBaggageHeader(object) {\n if (Object.keys(object).length === 0) {\n // An empty baggage header is not spec compliant: We return undefined.\n return undefined;\n }\n\n return Object.entries(object).reduce((baggageHeader, [objectKey, objectValue], currentIndex) => {\n const baggageEntry = `${encodeURIComponent(objectKey)}=${encodeURIComponent(objectValue)}`;\n const newBaggageHeader = currentIndex === 0 ? baggageEntry : `${baggageHeader},${baggageEntry}`;\n if (newBaggageHeader.length > MAX_BAGGAGE_STRING_LENGTH) {\n DEBUG_BUILD &&\n logger.warn(\n `Not adding key: ${objectKey} with val: ${objectValue} to baggage header due to exceeding baggage size limits.`,\n );\n return baggageHeader;\n } else {\n return newBaggageHeader;\n }\n }, '');\n}\n\nexport { BAGGAGE_HEADER_NAME, MAX_BAGGAGE_STRING_LENGTH, SENTRY_BAGGAGE_KEY_PREFIX, SENTRY_BAGGAGE_KEY_PREFIX_REGEX, baggageHeaderToDynamicSamplingContext, dynamicSamplingContextToSentryBaggageHeader };\n","import { baggageHeaderToDynamicSamplingContext } from './baggage.js';\nimport { uuid4 } from './misc.js';\n\n// eslint-disable-next-line @sentry-internal/sdk/no-regexp-constructor -- RegExp is used for readability here\nconst TRACEPARENT_REGEXP = new RegExp(\n '^[ \\\\t]*' + // whitespace\n '([0-9a-f]{32})?' + // trace_id\n '-?([0-9a-f]{16})?' + // span_id\n '-?([01])?' + // sampled\n '[ \\\\t]*$', // whitespace\n);\n\n/**\n * Extract transaction context data from a `sentry-trace` header.\n *\n * @param traceparent Traceparent string\n *\n * @returns Object containing data from the header, or undefined if traceparent string is malformed\n */\nfunction extractTraceparentData(traceparent) {\n if (!traceparent) {\n return undefined;\n }\n\n const matches = traceparent.match(TRACEPARENT_REGEXP);\n if (!matches) {\n return undefined;\n }\n\n let parentSampled;\n if (matches[3] === '1') {\n parentSampled = true;\n } else if (matches[3] === '0') {\n parentSampled = false;\n }\n\n return {\n traceId: matches[1],\n parentSampled,\n parentSpanId: matches[2],\n };\n}\n\n/**\n * Create tracing context from incoming headers.\n *\n * @deprecated Use `propagationContextFromHeaders` instead.\n */\n// TODO(v8): Remove this function\nfunction tracingContextFromHeaders(\n sentryTrace,\n baggage,\n)\n\n {\n const traceparentData = extractTraceparentData(sentryTrace);\n const dynamicSamplingContext = baggageHeaderToDynamicSamplingContext(baggage);\n\n const { traceId, parentSpanId, parentSampled } = traceparentData || {};\n\n if (!traceparentData) {\n return {\n traceparentData,\n dynamicSamplingContext: undefined,\n propagationContext: {\n traceId: traceId || uuid4(),\n spanId: uuid4().substring(16),\n },\n };\n } else {\n return {\n traceparentData,\n dynamicSamplingContext: dynamicSamplingContext || {}, // If we have traceparent data but no DSC it means we are not head of trace and we must freeze it\n propagationContext: {\n traceId: traceId || uuid4(),\n parentSpanId: parentSpanId || uuid4().substring(16),\n spanId: uuid4().substring(16),\n sampled: parentSampled,\n dsc: dynamicSamplingContext || {}, // If we have traceparent data but no DSC it means we are not head of trace and we must freeze it\n },\n };\n }\n}\n\n/**\n * Create a propagation context from incoming headers.\n */\nfunction propagationContextFromHeaders(\n sentryTrace,\n baggage,\n) {\n const traceparentData = extractTraceparentData(sentryTrace);\n const dynamicSamplingContext = baggageHeaderToDynamicSamplingContext(baggage);\n\n const { traceId, parentSpanId, parentSampled } = traceparentData || {};\n\n if (!traceparentData) {\n return {\n traceId: traceId || uuid4(),\n spanId: uuid4().substring(16),\n };\n } else {\n return {\n traceId: traceId || uuid4(),\n parentSpanId: parentSpanId || uuid4().substring(16),\n spanId: uuid4().substring(16),\n sampled: parentSampled,\n dsc: dynamicSamplingContext || {}, // If we have traceparent data but no DSC it means we are not head of trace and we must freeze it\n };\n }\n}\n\n/**\n * Create sentry-trace header from span context values.\n */\nfunction generateSentryTraceHeader(\n traceId = uuid4(),\n spanId = uuid4().substring(16),\n sampled,\n) {\n let sampledString = '';\n if (sampled !== undefined) {\n sampledString = sampled ? '-1' : '-0';\n }\n return `${traceId}-${spanId}${sampledString}`;\n}\n\nexport { TRACEPARENT_REGEXP, extractTraceparentData, generateSentryTraceHeader, propagationContextFromHeaders, tracingContextFromHeaders };\n","import { dropUndefinedKeys, generateSentryTraceHeader, timestampInSeconds } from '@sentry/utils';\n\n// These are aligned with OpenTelemetry trace flags\nconst TRACE_FLAG_NONE = 0x0;\nconst TRACE_FLAG_SAMPLED = 0x1;\n\n/**\n * Convert a span to a trace context, which can be sent as the `trace` context in an event.\n */\nfunction spanToTraceContext(span) {\n const { spanId: span_id, traceId: trace_id } = span.spanContext();\n const { data, op, parent_span_id, status, tags, origin } = spanToJSON(span);\n\n return dropUndefinedKeys({\n data,\n op,\n parent_span_id,\n span_id,\n status,\n tags,\n trace_id,\n origin,\n });\n}\n\n/**\n * Convert a Span to a Sentry trace header.\n */\nfunction spanToTraceHeader(span) {\n const { traceId, spanId } = span.spanContext();\n const sampled = spanIsSampled(span);\n return generateSentryTraceHeader(traceId, spanId, sampled);\n}\n\n/**\n * Convert a span time input intp a timestamp in seconds.\n */\nfunction spanTimeInputToSeconds(input) {\n if (typeof input === 'number') {\n return ensureTimestampInSeconds(input);\n }\n\n if (Array.isArray(input)) {\n // See {@link HrTime} for the array-based time format\n return input[0] + input[1] / 1e9;\n }\n\n if (input instanceof Date) {\n return ensureTimestampInSeconds(input.getTime());\n }\n\n return timestampInSeconds();\n}\n\n/**\n * Converts a timestamp to second, if it was in milliseconds, or keeps it as second.\n */\nfunction ensureTimestampInSeconds(timestamp) {\n const isMs = timestamp > 9999999999;\n return isMs ? timestamp / 1000 : timestamp;\n}\n\n/**\n * Convert a span to a JSON representation.\n * Note that all fields returned here are optional and need to be guarded against.\n *\n * Note: Because of this, we currently have a circular type dependency (which we opted out of in package.json).\n * This is not avoidable as we need `spanToJSON` in `spanUtils.ts`, which in turn is needed by `span.ts` for backwards compatibility.\n * And `spanToJSON` needs the Span class from `span.ts` to check here.\n * TODO v8: When we remove the deprecated stuff from `span.ts`, we can remove the circular dependency again.\n */\nfunction spanToJSON(span) {\n if (spanIsSpanClass(span)) {\n return span.getSpanJSON();\n }\n\n // Fallback: We also check for `.toJSON()` here...\n // eslint-disable-next-line deprecation/deprecation\n if (typeof span.toJSON === 'function') {\n // eslint-disable-next-line deprecation/deprecation\n return span.toJSON();\n }\n\n return {};\n}\n\n/**\n * Sadly, due to circular dependency checks we cannot actually import the Span class here and check for instanceof.\n * :( So instead we approximate this by checking if it has the `getSpanJSON` method.\n */\nfunction spanIsSpanClass(span) {\n return typeof (span ).getSpanJSON === 'function';\n}\n\n/**\n * Returns true if a span is sampled.\n * In most cases, you should just use `span.isRecording()` instead.\n * However, this has a slightly different semantic, as it also returns false if the span is finished.\n * So in the case where this distinction is important, use this method.\n */\nfunction spanIsSampled(span) {\n // We align our trace flags with the ones OpenTelemetry use\n // So we also check for sampled the same way they do.\n const { traceFlags } = span.spanContext();\n // eslint-disable-next-line no-bitwise\n return Boolean(traceFlags & TRACE_FLAG_SAMPLED);\n}\n\nexport { TRACE_FLAG_NONE, TRACE_FLAG_SAMPLED, spanIsSampled, spanTimeInputToSeconds, spanToJSON, spanToTraceContext, spanToTraceHeader };\n","import { dropUndefinedKeys } from '@sentry/utils';\nimport { DEFAULT_ENVIRONMENT } from '../constants.js';\nimport { getClient, getCurrentScope } from '../exports.js';\nimport { getRootSpan } from '../utils/getRootSpan.js';\nimport { spanToJSON, spanIsSampled } from '../utils/spanUtils.js';\n\n/**\n * Creates a dynamic sampling context from a client.\n *\n * Dispatches the `createDsc` lifecycle hook as a side effect.\n */\nfunction getDynamicSamplingContextFromClient(\n trace_id,\n client,\n scope,\n) {\n const options = client.getOptions();\n\n const { publicKey: public_key } = client.getDsn() || {};\n // TODO(v8): Remove segment from User\n // eslint-disable-next-line deprecation/deprecation\n const { segment: user_segment } = (scope && scope.getUser()) || {};\n\n const dsc = dropUndefinedKeys({\n environment: options.environment || DEFAULT_ENVIRONMENT,\n release: options.release,\n user_segment,\n public_key,\n trace_id,\n }) ;\n\n client.emit && client.emit('createDsc', dsc);\n\n return dsc;\n}\n\n/**\n * A Span with a frozen dynamic sampling context.\n */\n\n/**\n * Creates a dynamic sampling context from a span (and client and scope)\n *\n * @param span the span from which a few values like the root span name and sample rate are extracted.\n *\n * @returns a dynamic sampling context\n */\nfunction getDynamicSamplingContextFromSpan(span) {\n const client = getClient();\n if (!client) {\n return {};\n }\n\n // passing emit=false here to only emit later once the DSC is actually populated\n const dsc = getDynamicSamplingContextFromClient(spanToJSON(span).trace_id || '', client, getCurrentScope());\n\n // TODO (v8): Remove v7FrozenDsc as a Transaction will no longer have _frozenDynamicSamplingContext\n const txn = getRootSpan(span) ;\n if (!txn) {\n return dsc;\n }\n\n // TODO (v8): Remove v7FrozenDsc as a Transaction will no longer have _frozenDynamicSamplingContext\n // For now we need to avoid breaking users who directly created a txn with a DSC, where this field is still set.\n // @see Transaction class constructor\n const v7FrozenDsc = txn && txn._frozenDynamicSamplingContext;\n if (v7FrozenDsc) {\n return v7FrozenDsc;\n }\n\n // TODO (v8): Replace txn.metadata with txn.attributes[]\n // We can't do this yet because attributes aren't always set yet.\n // eslint-disable-next-line deprecation/deprecation\n const { sampleRate: maybeSampleRate, source } = txn.metadata;\n if (maybeSampleRate != null) {\n dsc.sample_rate = `${maybeSampleRate}`;\n }\n\n // We don't want to have a transaction name in the DSC if the source is \"url\" because URLs might contain PII\n const jsonSpan = spanToJSON(txn);\n\n // after JSON conversion, txn.name becomes jsonSpan.description\n if (source && source !== 'url') {\n dsc.transaction = jsonSpan.description;\n }\n\n dsc.sampled = String(spanIsSampled(txn));\n\n client.emit && client.emit('createDsc', dsc);\n\n return dsc;\n}\n\nexport { getDynamicSamplingContextFromClient, getDynamicSamplingContextFromSpan };\n","import { dropUndefinedKeys, arrayify } from '@sentry/utils';\nimport { getDynamicSamplingContextFromSpan } from '../tracing/dynamicSamplingContext.js';\nimport { getRootSpan } from './getRootSpan.js';\nimport { spanToTraceContext, spanToJSON } from './spanUtils.js';\n\n/**\n * Applies data from the scope to the event and runs all event processors on it.\n */\nfunction applyScopeDataToEvent(event, data) {\n const { fingerprint, span, breadcrumbs, sdkProcessingMetadata } = data;\n\n // Apply general data\n applyDataToEvent(event, data);\n\n // We want to set the trace context for normal events only if there isn't already\n // a trace context on the event. There is a product feature in place where we link\n // errors with transaction and it relies on that.\n if (span) {\n applySpanToEvent(event, span);\n }\n\n applyFingerprintToEvent(event, fingerprint);\n applyBreadcrumbsToEvent(event, breadcrumbs);\n applySdkMetadataToEvent(event, sdkProcessingMetadata);\n}\n\n/** Merge data of two scopes together. */\nfunction mergeScopeData(data, mergeData) {\n const {\n extra,\n tags,\n user,\n contexts,\n level,\n sdkProcessingMetadata,\n breadcrumbs,\n fingerprint,\n eventProcessors,\n attachments,\n propagationContext,\n // eslint-disable-next-line deprecation/deprecation\n transactionName,\n span,\n } = mergeData;\n\n mergeAndOverwriteScopeData(data, 'extra', extra);\n mergeAndOverwriteScopeData(data, 'tags', tags);\n mergeAndOverwriteScopeData(data, 'user', user);\n mergeAndOverwriteScopeData(data, 'contexts', contexts);\n mergeAndOverwriteScopeData(data, 'sdkProcessingMetadata', sdkProcessingMetadata);\n\n if (level) {\n data.level = level;\n }\n\n if (transactionName) {\n // eslint-disable-next-line deprecation/deprecation\n data.transactionName = transactionName;\n }\n\n if (span) {\n data.span = span;\n }\n\n if (breadcrumbs.length) {\n data.breadcrumbs = [...data.breadcrumbs, ...breadcrumbs];\n }\n\n if (fingerprint.length) {\n data.fingerprint = [...data.fingerprint, ...fingerprint];\n }\n\n if (eventProcessors.length) {\n data.eventProcessors = [...data.eventProcessors, ...eventProcessors];\n }\n\n if (attachments.length) {\n data.attachments = [...data.attachments, ...attachments];\n }\n\n data.propagationContext = { ...data.propagationContext, ...propagationContext };\n}\n\n/**\n * Merges certain scope data. Undefined values will overwrite any existing values.\n * Exported only for tests.\n */\nfunction mergeAndOverwriteScopeData\n\n(data, prop, mergeVal) {\n if (mergeVal && Object.keys(mergeVal).length) {\n // Clone object\n data[prop] = { ...data[prop] };\n for (const key in mergeVal) {\n if (Object.prototype.hasOwnProperty.call(mergeVal, key)) {\n data[prop][key] = mergeVal[key];\n }\n }\n }\n}\n\nfunction applyDataToEvent(event, data) {\n const {\n extra,\n tags,\n user,\n contexts,\n level,\n // eslint-disable-next-line deprecation/deprecation\n transactionName,\n } = data;\n\n const cleanedExtra = dropUndefinedKeys(extra);\n if (cleanedExtra && Object.keys(cleanedExtra).length) {\n event.extra = { ...cleanedExtra, ...event.extra };\n }\n\n const cleanedTags = dropUndefinedKeys(tags);\n if (cleanedTags && Object.keys(cleanedTags).length) {\n event.tags = { ...cleanedTags, ...event.tags };\n }\n\n const cleanedUser = dropUndefinedKeys(user);\n if (cleanedUser && Object.keys(cleanedUser).length) {\n event.user = { ...cleanedUser, ...event.user };\n }\n\n const cleanedContexts = dropUndefinedKeys(contexts);\n if (cleanedContexts && Object.keys(cleanedContexts).length) {\n event.contexts = { ...cleanedContexts, ...event.contexts };\n }\n\n if (level) {\n event.level = level;\n }\n\n if (transactionName) {\n event.transaction = transactionName;\n }\n}\n\nfunction applyBreadcrumbsToEvent(event, breadcrumbs) {\n const mergedBreadcrumbs = [...(event.breadcrumbs || []), ...breadcrumbs];\n event.breadcrumbs = mergedBreadcrumbs.length ? mergedBreadcrumbs : undefined;\n}\n\nfunction applySdkMetadataToEvent(event, sdkProcessingMetadata) {\n event.sdkProcessingMetadata = {\n ...event.sdkProcessingMetadata,\n ...sdkProcessingMetadata,\n };\n}\n\nfunction applySpanToEvent(event, span) {\n event.contexts = { trace: spanToTraceContext(span), ...event.contexts };\n const rootSpan = getRootSpan(span);\n if (rootSpan) {\n event.sdkProcessingMetadata = {\n dynamicSamplingContext: getDynamicSamplingContextFromSpan(span),\n ...event.sdkProcessingMetadata,\n };\n const transactionName = spanToJSON(rootSpan).description;\n if (transactionName) {\n event.tags = { transaction: transactionName, ...event.tags };\n }\n }\n}\n\n/**\n * Applies fingerprint from the scope to the event if there's one,\n * uses message if there's one instead or get rid of empty fingerprint\n */\nfunction applyFingerprintToEvent(event, fingerprint) {\n // Make sure it's an array first and we actually have something in place\n event.fingerprint = event.fingerprint ? arrayify(event.fingerprint) : [];\n\n // If we have something on the scope, then merge it with event\n if (fingerprint) {\n event.fingerprint = event.fingerprint.concat(fingerprint);\n }\n\n // If we have no data at all, remove empty array default\n if (event.fingerprint && !event.fingerprint.length) {\n delete event.fingerprint;\n }\n}\n\nexport { applyScopeDataToEvent, mergeAndOverwriteScopeData, mergeScopeData };\n","import { isPlainObject, dateTimestampInSeconds, uuid4, logger } from '@sentry/utils';\nimport { getGlobalEventProcessors, notifyEventProcessors } from './eventProcessors.js';\nimport { updateSession } from './session.js';\nimport { applyScopeDataToEvent } from './utils/applyScopeDataToEvent.js';\n\n/**\n * Default value for maximum number of breadcrumbs added to an event.\n */\nconst DEFAULT_MAX_BREADCRUMBS = 100;\n\n/**\n * The global scope is kept in this module.\n * When accessing this via `getGlobalScope()` we'll make sure to set one if none is currently present.\n */\nlet globalScope;\n\n/**\n * Holds additional event information. {@link Scope.applyToEvent} will be\n * called by the client before an event will be sent.\n */\nclass Scope {\n /** Flag if notifying is happening. */\n\n /** Callback for client to receive scope changes. */\n\n /** Callback list that will be called after {@link applyToEvent}. */\n\n /** Array of breadcrumbs. */\n\n /** User */\n\n /** Tags */\n\n /** Extra */\n\n /** Contexts */\n\n /** Attachments */\n\n /** Propagation Context for distributed tracing */\n\n /**\n * A place to stash data which is needed at some point in the SDK's event processing pipeline but which shouldn't get\n * sent to Sentry\n */\n\n /** Fingerprint */\n\n /** Severity */\n // eslint-disable-next-line deprecation/deprecation\n\n /**\n * Transaction Name\n */\n\n /** Span */\n\n /** Session */\n\n /** Request Mode Session Status */\n\n /** The client on this scope */\n\n // NOTE: Any field which gets added here should get added not only to the constructor but also to the `clone` method.\n\n constructor() {\n this._notifyingListeners = false;\n this._scopeListeners = [];\n this._eventProcessors = [];\n this._breadcrumbs = [];\n this._attachments = [];\n this._user = {};\n this._tags = {};\n this._extra = {};\n this._contexts = {};\n this._sdkProcessingMetadata = {};\n this._propagationContext = generatePropagationContext();\n }\n\n /**\n * Inherit values from the parent scope.\n * @deprecated Use `scope.clone()` and `new Scope()` instead.\n */\n static clone(scope) {\n return scope ? scope.clone() : new Scope();\n }\n\n /**\n * Clone this scope instance.\n */\n clone() {\n const newScope = new Scope();\n newScope._breadcrumbs = [...this._breadcrumbs];\n newScope._tags = { ...this._tags };\n newScope._extra = { ...this._extra };\n newScope._contexts = { ...this._contexts };\n newScope._user = this._user;\n newScope._level = this._level;\n newScope._span = this._span;\n newScope._session = this._session;\n newScope._transactionName = this._transactionName;\n newScope._fingerprint = this._fingerprint;\n newScope._eventProcessors = [...this._eventProcessors];\n newScope._requestSession = this._requestSession;\n newScope._attachments = [...this._attachments];\n newScope._sdkProcessingMetadata = { ...this._sdkProcessingMetadata };\n newScope._propagationContext = { ...this._propagationContext };\n newScope._client = this._client;\n\n return newScope;\n }\n\n /** Update the client on the scope. */\n setClient(client) {\n this._client = client;\n }\n\n /**\n * Get the client assigned to this scope.\n *\n * It is generally recommended to use the global function `Sentry.getClient()` instead, unless you know what you are doing.\n */\n getClient() {\n return this._client;\n }\n\n /**\n * Add internal on change listener. Used for sub SDKs that need to store the scope.\n * @hidden\n */\n addScopeListener(callback) {\n this._scopeListeners.push(callback);\n }\n\n /**\n * @inheritDoc\n */\n addEventProcessor(callback) {\n this._eventProcessors.push(callback);\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setUser(user) {\n // If null is passed we want to unset everything, but still define keys,\n // so that later down in the pipeline any existing values are cleared.\n this._user = user || {\n email: undefined,\n id: undefined,\n ip_address: undefined,\n segment: undefined,\n username: undefined,\n };\n\n if (this._session) {\n updateSession(this._session, { user });\n }\n\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n getUser() {\n return this._user;\n }\n\n /**\n * @inheritDoc\n */\n getRequestSession() {\n return this._requestSession;\n }\n\n /**\n * @inheritDoc\n */\n setRequestSession(requestSession) {\n this._requestSession = requestSession;\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setTags(tags) {\n this._tags = {\n ...this._tags,\n ...tags,\n };\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setTag(key, value) {\n this._tags = { ...this._tags, [key]: value };\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setExtras(extras) {\n this._extra = {\n ...this._extra,\n ...extras,\n };\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setExtra(key, extra) {\n this._extra = { ...this._extra, [key]: extra };\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setFingerprint(fingerprint) {\n this._fingerprint = fingerprint;\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setLevel(\n // eslint-disable-next-line deprecation/deprecation\n level,\n ) {\n this._level = level;\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * Sets the transaction name on the scope for future events.\n */\n setTransactionName(name) {\n this._transactionName = name;\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setContext(key, context) {\n if (context === null) {\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete this._contexts[key];\n } else {\n this._contexts[key] = context;\n }\n\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * Sets the Span on the scope.\n * @param span Span\n * @deprecated Instead of setting a span on a scope, use `startSpan()`/`startSpanManual()` instead.\n */\n setSpan(span) {\n this._span = span;\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * Returns the `Span` if there is one.\n * @deprecated Use `getActiveSpan()` instead.\n */\n getSpan() {\n return this._span;\n }\n\n /**\n * Returns the `Transaction` attached to the scope (if there is one).\n * @deprecated You should not rely on the transaction, but just use `startSpan()` APIs instead.\n */\n getTransaction() {\n // Often, this span (if it exists at all) will be a transaction, but it's not guaranteed to be. Regardless, it will\n // have a pointer to the currently-active transaction.\n const span = this._span;\n // Cannot replace with getRootSpan because getRootSpan returns a span, not a transaction\n // Also, this method will be removed anyway.\n // eslint-disable-next-line deprecation/deprecation\n return span && span.transaction;\n }\n\n /**\n * @inheritDoc\n */\n setSession(session) {\n if (!session) {\n delete this._session;\n } else {\n this._session = session;\n }\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n getSession() {\n return this._session;\n }\n\n /**\n * @inheritDoc\n */\n update(captureContext) {\n if (!captureContext) {\n return this;\n }\n\n const scopeToMerge = typeof captureContext === 'function' ? captureContext(this) : captureContext;\n\n if (scopeToMerge instanceof Scope) {\n const scopeData = scopeToMerge.getScopeData();\n\n this._tags = { ...this._tags, ...scopeData.tags };\n this._extra = { ...this._extra, ...scopeData.extra };\n this._contexts = { ...this._contexts, ...scopeData.contexts };\n if (scopeData.user && Object.keys(scopeData.user).length) {\n this._user = scopeData.user;\n }\n if (scopeData.level) {\n this._level = scopeData.level;\n }\n if (scopeData.fingerprint.length) {\n this._fingerprint = scopeData.fingerprint;\n }\n if (scopeToMerge.getRequestSession()) {\n this._requestSession = scopeToMerge.getRequestSession();\n }\n if (scopeData.propagationContext) {\n this._propagationContext = scopeData.propagationContext;\n }\n } else if (isPlainObject(scopeToMerge)) {\n const scopeContext = captureContext ;\n this._tags = { ...this._tags, ...scopeContext.tags };\n this._extra = { ...this._extra, ...scopeContext.extra };\n this._contexts = { ...this._contexts, ...scopeContext.contexts };\n if (scopeContext.user) {\n this._user = scopeContext.user;\n }\n if (scopeContext.level) {\n this._level = scopeContext.level;\n }\n if (scopeContext.fingerprint) {\n this._fingerprint = scopeContext.fingerprint;\n }\n if (scopeContext.requestSession) {\n this._requestSession = scopeContext.requestSession;\n }\n if (scopeContext.propagationContext) {\n this._propagationContext = scopeContext.propagationContext;\n }\n }\n\n return this;\n }\n\n /**\n * @inheritDoc\n */\n clear() {\n this._breadcrumbs = [];\n this._tags = {};\n this._extra = {};\n this._user = {};\n this._contexts = {};\n this._level = undefined;\n this._transactionName = undefined;\n this._fingerprint = undefined;\n this._requestSession = undefined;\n this._span = undefined;\n this._session = undefined;\n this._notifyScopeListeners();\n this._attachments = [];\n this._propagationContext = generatePropagationContext();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n addBreadcrumb(breadcrumb, maxBreadcrumbs) {\n const maxCrumbs = typeof maxBreadcrumbs === 'number' ? maxBreadcrumbs : DEFAULT_MAX_BREADCRUMBS;\n\n // No data has been changed, so don't notify scope listeners\n if (maxCrumbs <= 0) {\n return this;\n }\n\n const mergedBreadcrumb = {\n timestamp: dateTimestampInSeconds(),\n ...breadcrumb,\n };\n\n const breadcrumbs = this._breadcrumbs;\n breadcrumbs.push(mergedBreadcrumb);\n this._breadcrumbs = breadcrumbs.length > maxCrumbs ? breadcrumbs.slice(-maxCrumbs) : breadcrumbs;\n\n this._notifyScopeListeners();\n\n return this;\n }\n\n /**\n * @inheritDoc\n */\n getLastBreadcrumb() {\n return this._breadcrumbs[this._breadcrumbs.length - 1];\n }\n\n /**\n * @inheritDoc\n */\n clearBreadcrumbs() {\n this._breadcrumbs = [];\n this._notifyScopeListeners();\n return this;\n }\n\n /**\n * @inheritDoc\n */\n addAttachment(attachment) {\n this._attachments.push(attachment);\n return this;\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `getScopeData()` instead.\n */\n getAttachments() {\n const data = this.getScopeData();\n\n return data.attachments;\n }\n\n /**\n * @inheritDoc\n */\n clearAttachments() {\n this._attachments = [];\n return this;\n }\n\n /** @inheritDoc */\n getScopeData() {\n const {\n _breadcrumbs,\n _attachments,\n _contexts,\n _tags,\n _extra,\n _user,\n _level,\n _fingerprint,\n _eventProcessors,\n _propagationContext,\n _sdkProcessingMetadata,\n _transactionName,\n _span,\n } = this;\n\n return {\n breadcrumbs: _breadcrumbs,\n attachments: _attachments,\n contexts: _contexts,\n tags: _tags,\n extra: _extra,\n user: _user,\n level: _level,\n fingerprint: _fingerprint || [],\n eventProcessors: _eventProcessors,\n propagationContext: _propagationContext,\n sdkProcessingMetadata: _sdkProcessingMetadata,\n transactionName: _transactionName,\n span: _span,\n };\n }\n\n /**\n * Applies data from the scope to the event and runs all event processors on it.\n *\n * @param event Event\n * @param hint Object containing additional information about the original exception, for use by the event processors.\n * @hidden\n * @deprecated Use `applyScopeDataToEvent()` directly\n */\n applyToEvent(\n event,\n hint = {},\n additionalEventProcessors = [],\n ) {\n applyScopeDataToEvent(event, this.getScopeData());\n\n // TODO (v8): Update this order to be: Global > Client > Scope\n const eventProcessors = [\n ...additionalEventProcessors,\n // eslint-disable-next-line deprecation/deprecation\n ...getGlobalEventProcessors(),\n ...this._eventProcessors,\n ];\n\n return notifyEventProcessors(eventProcessors, event, hint);\n }\n\n /**\n * Add data which will be accessible during event processing but won't get sent to Sentry\n */\n setSDKProcessingMetadata(newData) {\n this._sdkProcessingMetadata = { ...this._sdkProcessingMetadata, ...newData };\n\n return this;\n }\n\n /**\n * @inheritDoc\n */\n setPropagationContext(context) {\n this._propagationContext = context;\n return this;\n }\n\n /**\n * @inheritDoc\n */\n getPropagationContext() {\n return this._propagationContext;\n }\n\n /**\n * Capture an exception for this scope.\n *\n * @param exception The exception to capture.\n * @param hint Optinal additional data to attach to the Sentry event.\n * @returns the id of the captured Sentry event.\n */\n captureException(exception, hint) {\n const eventId = hint && hint.event_id ? hint.event_id : uuid4();\n\n if (!this._client) {\n logger.warn('No client configured on scope - will not capture exception!');\n return eventId;\n }\n\n const syntheticException = new Error('Sentry syntheticException');\n\n this._client.captureException(\n exception,\n {\n originalException: exception,\n syntheticException,\n ...hint,\n event_id: eventId,\n },\n this,\n );\n\n return eventId;\n }\n\n /**\n * Capture a message for this scope.\n *\n * @param message The message to capture.\n * @param level An optional severity level to report the message with.\n * @param hint Optional additional data to attach to the Sentry event.\n * @returns the id of the captured message.\n */\n captureMessage(message, level, hint) {\n const eventId = hint && hint.event_id ? hint.event_id : uuid4();\n\n if (!this._client) {\n logger.warn('No client configured on scope - will not capture message!');\n return eventId;\n }\n\n const syntheticException = new Error(message);\n\n this._client.captureMessage(\n message,\n level,\n {\n originalException: message,\n syntheticException,\n ...hint,\n event_id: eventId,\n },\n this,\n );\n\n return eventId;\n }\n\n /**\n * Captures a manually created event for this scope and sends it to Sentry.\n *\n * @param exception The event to capture.\n * @param hint Optional additional data to attach to the Sentry event.\n * @returns the id of the captured event.\n */\n captureEvent(event, hint) {\n const eventId = hint && hint.event_id ? hint.event_id : uuid4();\n\n if (!this._client) {\n logger.warn('No client configured on scope - will not capture event!');\n return eventId;\n }\n\n this._client.captureEvent(event, { ...hint, event_id: eventId }, this);\n\n return eventId;\n }\n\n /**\n * This will be called on every set call.\n */\n _notifyScopeListeners() {\n // We need this check for this._notifyingListeners to be able to work on scope during updates\n // If this check is not here we'll produce endless recursion when something is done with the scope\n // during the callback.\n if (!this._notifyingListeners) {\n this._notifyingListeners = true;\n this._scopeListeners.forEach(callback => {\n callback(this);\n });\n this._notifyingListeners = false;\n }\n }\n}\n\n/**\n * Get the global scope.\n * This scope is applied to _all_ events.\n */\nfunction getGlobalScope() {\n if (!globalScope) {\n globalScope = new Scope();\n }\n\n return globalScope;\n}\n\n/**\n * This is mainly needed for tests.\n * DO NOT USE this, as this is an internal API and subject to change.\n * @hidden\n */\nfunction setGlobalScope(scope) {\n globalScope = scope;\n}\n\nfunction generatePropagationContext() {\n return {\n traceId: uuid4(),\n spanId: uuid4().substring(16),\n };\n}\n\nexport { Scope, getGlobalScope, setGlobalScope };\n","const SDK_VERSION = '7.120.3';\n\nexport { SDK_VERSION };\n","import { isThenable, uuid4, dateTimestampInSeconds, consoleSandbox, logger, GLOBAL_OBJ, getGlobalSingleton } from '@sentry/utils';\nimport { DEFAULT_ENVIRONMENT } from './constants.js';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { Scope } from './scope.js';\nimport { closeSession, makeSession, updateSession } from './session.js';\nimport { SDK_VERSION } from './version.js';\n\n/**\n * API compatibility version of this hub.\n *\n * WARNING: This number should only be increased when the global interface\n * changes and new methods are introduced.\n *\n * @hidden\n */\nconst API_VERSION = parseFloat(SDK_VERSION);\n\n/**\n * Default maximum number of breadcrumbs added to an event. Can be overwritten\n * with {@link Options.maxBreadcrumbs}.\n */\nconst DEFAULT_BREADCRUMBS = 100;\n\n/**\n * @deprecated The `Hub` class will be removed in version 8 of the SDK in favour of `Scope` and `Client` objects.\n *\n * If you previously used the `Hub` class directly, replace it with `Scope` and `Client` objects. More information:\n * - [Multiple Sentry Instances](https://docs.sentry.io/platforms/javascript/best-practices/multiple-sentry-instances/)\n * - [Browser Extensions](https://docs.sentry.io/platforms/javascript/best-practices/browser-extensions/)\n *\n * Some of our APIs are typed with the Hub class instead of the interface (e.g. `getCurrentHub`). Most of them are deprecated\n * themselves and will also be removed in version 8. More information:\n * - [Migration Guide](https://github.com/getsentry/sentry-javascript/blob/develop/MIGRATION.md#deprecate-hub)\n */\n// eslint-disable-next-line deprecation/deprecation\nclass Hub {\n /** Is a {@link Layer}[] containing the client and scope */\n\n /** Contains the last event id of a captured event. */\n\n /**\n * Creates a new instance of the hub, will push one {@link Layer} into the\n * internal stack on creation.\n *\n * @param client bound to the hub.\n * @param scope bound to the hub.\n * @param version number, higher number means higher priority.\n *\n * @deprecated Instantiation of Hub objects is deprecated and the constructor will be removed in version 8 of the SDK.\n *\n * If you are currently using the Hub for multi-client use like so:\n *\n * ```\n * // OLD\n * const hub = new Hub();\n * hub.bindClient(client);\n * makeMain(hub)\n * ```\n *\n * instead initialize the client as follows:\n *\n * ```\n * // NEW\n * Sentry.withIsolationScope(() => {\n * Sentry.setCurrentClient(client);\n * client.init();\n * });\n * ```\n *\n * If you are using the Hub to capture events like so:\n *\n * ```\n * // OLD\n * const client = new Client();\n * const hub = new Hub(client);\n * hub.captureException()\n * ```\n *\n * instead capture isolated events as follows:\n *\n * ```\n * // NEW\n * const client = new Client();\n * const scope = new Scope();\n * scope.setClient(client);\n * scope.captureException();\n * ```\n */\n constructor(\n client,\n scope,\n isolationScope,\n _version = API_VERSION,\n ) {this._version = _version;\n let assignedScope;\n if (!scope) {\n assignedScope = new Scope();\n assignedScope.setClient(client);\n } else {\n assignedScope = scope;\n }\n\n let assignedIsolationScope;\n if (!isolationScope) {\n assignedIsolationScope = new Scope();\n assignedIsolationScope.setClient(client);\n } else {\n assignedIsolationScope = isolationScope;\n }\n\n this._stack = [{ scope: assignedScope }];\n\n if (client) {\n // eslint-disable-next-line deprecation/deprecation\n this.bindClient(client);\n }\n\n this._isolationScope = assignedIsolationScope;\n }\n\n /**\n * Checks if this hub's version is older than the given version.\n *\n * @param version A version number to compare to.\n * @return True if the given version is newer; otherwise false.\n *\n * @deprecated This will be removed in v8.\n */\n isOlderThan(version) {\n return this._version < version;\n }\n\n /**\n * This binds the given client to the current scope.\n * @param client An SDK client (client) instance.\n *\n * @deprecated Use `initAndBind()` directly, or `setCurrentClient()` and/or `client.init()` instead.\n */\n bindClient(client) {\n // eslint-disable-next-line deprecation/deprecation\n const top = this.getStackTop();\n top.client = client;\n top.scope.setClient(client);\n // eslint-disable-next-line deprecation/deprecation\n if (client && client.setupIntegrations) {\n // eslint-disable-next-line deprecation/deprecation\n client.setupIntegrations();\n }\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `withScope` instead.\n */\n pushScope() {\n // We want to clone the content of prev scope\n // eslint-disable-next-line deprecation/deprecation\n const scope = this.getScope().clone();\n // eslint-disable-next-line deprecation/deprecation\n this.getStack().push({\n // eslint-disable-next-line deprecation/deprecation\n client: this.getClient(),\n scope,\n });\n return scope;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `withScope` instead.\n */\n popScope() {\n // eslint-disable-next-line deprecation/deprecation\n if (this.getStack().length <= 1) return false;\n // eslint-disable-next-line deprecation/deprecation\n return !!this.getStack().pop();\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.withScope()` instead.\n */\n withScope(callback) {\n // eslint-disable-next-line deprecation/deprecation\n const scope = this.pushScope();\n\n let maybePromiseResult;\n try {\n maybePromiseResult = callback(scope);\n } catch (e) {\n // eslint-disable-next-line deprecation/deprecation\n this.popScope();\n throw e;\n }\n\n if (isThenable(maybePromiseResult)) {\n // @ts-expect-error - isThenable returns the wrong type\n return maybePromiseResult.then(\n res => {\n // eslint-disable-next-line deprecation/deprecation\n this.popScope();\n return res;\n },\n e => {\n // eslint-disable-next-line deprecation/deprecation\n this.popScope();\n throw e;\n },\n );\n }\n\n // eslint-disable-next-line deprecation/deprecation\n this.popScope();\n return maybePromiseResult;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.getClient()` instead.\n */\n getClient() {\n // eslint-disable-next-line deprecation/deprecation\n return this.getStackTop().client ;\n }\n\n /**\n * Returns the scope of the top stack.\n *\n * @deprecated Use `Sentry.getCurrentScope()` instead.\n */\n getScope() {\n // eslint-disable-next-line deprecation/deprecation\n return this.getStackTop().scope;\n }\n\n /**\n * @deprecated Use `Sentry.getIsolationScope()` instead.\n */\n getIsolationScope() {\n return this._isolationScope;\n }\n\n /**\n * Returns the scope stack for domains or the process.\n * @deprecated This will be removed in v8.\n */\n getStack() {\n return this._stack;\n }\n\n /**\n * Returns the topmost scope layer in the order domain > local > process.\n * @deprecated This will be removed in v8.\n */\n getStackTop() {\n return this._stack[this._stack.length - 1];\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.captureException()` instead.\n */\n captureException(exception, hint) {\n const eventId = (this._lastEventId = hint && hint.event_id ? hint.event_id : uuid4());\n const syntheticException = new Error('Sentry syntheticException');\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().captureException(exception, {\n originalException: exception,\n syntheticException,\n ...hint,\n event_id: eventId,\n });\n\n return eventId;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.captureMessage()` instead.\n */\n captureMessage(\n message,\n // eslint-disable-next-line deprecation/deprecation\n level,\n hint,\n ) {\n const eventId = (this._lastEventId = hint && hint.event_id ? hint.event_id : uuid4());\n const syntheticException = new Error(message);\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().captureMessage(message, level, {\n originalException: message,\n syntheticException,\n ...hint,\n event_id: eventId,\n });\n\n return eventId;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.captureEvent()` instead.\n */\n captureEvent(event, hint) {\n const eventId = hint && hint.event_id ? hint.event_id : uuid4();\n if (!event.type) {\n this._lastEventId = eventId;\n }\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().captureEvent(event, { ...hint, event_id: eventId });\n return eventId;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated This will be removed in v8.\n */\n lastEventId() {\n return this._lastEventId;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `Sentry.addBreadcrumb()` instead.\n */\n addBreadcrumb(breadcrumb, hint) {\n // eslint-disable-next-line deprecation/deprecation\n const { scope, client } = this.getStackTop();\n\n if (!client) return;\n\n const { beforeBreadcrumb = null, maxBreadcrumbs = DEFAULT_BREADCRUMBS } =\n (client.getOptions && client.getOptions()) || {};\n\n if (maxBreadcrumbs <= 0) return;\n\n const timestamp = dateTimestampInSeconds();\n const mergedBreadcrumb = { timestamp, ...breadcrumb };\n const finalBreadcrumb = beforeBreadcrumb\n ? (consoleSandbox(() => beforeBreadcrumb(mergedBreadcrumb, hint)) )\n : mergedBreadcrumb;\n\n if (finalBreadcrumb === null) return;\n\n if (client.emit) {\n client.emit('beforeAddBreadcrumb', finalBreadcrumb, hint);\n }\n\n // TODO(v8): I know this comment doesn't make much sense because the hub will be deprecated but I still wanted to\n // write it down. In theory, we would have to add the breadcrumbs to the isolation scope here, however, that would\n // duplicate all of the breadcrumbs. There was the possibility of adding breadcrumbs to both, the isolation scope\n // and the normal scope, and deduplicating it down the line in the event processing pipeline. However, that would\n // have been very fragile, because the breadcrumb objects would have needed to keep their identity all throughout\n // the event processing pipeline.\n // In the new implementation, the top level `Sentry.addBreadcrumb()` should ONLY write to the isolation scope.\n\n scope.addBreadcrumb(finalBreadcrumb, maxBreadcrumbs);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setUser()` instead.\n */\n setUser(user) {\n // TODO(v8): The top level `Sentry.setUser()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setUser(user);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setUser(user);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setTags()` instead.\n */\n setTags(tags) {\n // TODO(v8): The top level `Sentry.setTags()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setTags(tags);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setTags(tags);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setExtras()` instead.\n */\n setExtras(extras) {\n // TODO(v8): The top level `Sentry.setExtras()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setExtras(extras);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setExtras(extras);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setTag()` instead.\n */\n setTag(key, value) {\n // TODO(v8): The top level `Sentry.setTag()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setTag(key, value);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setTag(key, value);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setExtra()` instead.\n */\n setExtra(key, extra) {\n // TODO(v8): The top level `Sentry.setExtra()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setExtra(key, extra);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setExtra(key, extra);\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.setContext()` instead.\n */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n setContext(name, context) {\n // TODO(v8): The top level `Sentry.setContext()` function should write ONLY to the isolation scope.\n // eslint-disable-next-line deprecation/deprecation\n this.getScope().setContext(name, context);\n // eslint-disable-next-line deprecation/deprecation\n this.getIsolationScope().setContext(name, context);\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `getScope()` directly.\n */\n configureScope(callback) {\n // eslint-disable-next-line deprecation/deprecation\n const { scope, client } = this.getStackTop();\n if (client) {\n callback(scope);\n }\n }\n\n /**\n * @inheritDoc\n */\n // eslint-disable-next-line deprecation/deprecation\n run(callback) {\n // eslint-disable-next-line deprecation/deprecation\n const oldHub = makeMain(this);\n try {\n callback(this);\n } finally {\n // eslint-disable-next-line deprecation/deprecation\n makeMain(oldHub);\n }\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `Sentry.getClient().getIntegrationByName()` instead.\n */\n getIntegration(integration) {\n // eslint-disable-next-line deprecation/deprecation\n const client = this.getClient();\n if (!client) return null;\n try {\n // eslint-disable-next-line deprecation/deprecation\n return client.getIntegration(integration);\n } catch (_oO) {\n DEBUG_BUILD && logger.warn(`Cannot retrieve integration ${integration.id} from the current Hub`);\n return null;\n }\n }\n\n /**\n * Starts a new `Transaction` and returns it. This is the entry point to manual tracing instrumentation.\n *\n * A tree structure can be built by adding child spans to the transaction, and child spans to other spans. To start a\n * new child span within the transaction or any span, call the respective `.startChild()` method.\n *\n * Every child span must be finished before the transaction is finished, otherwise the unfinished spans are discarded.\n *\n * The transaction must be finished with a call to its `.end()` method, at which point the transaction with all its\n * finished child spans will be sent to Sentry.\n *\n * @param context Properties of the new `Transaction`.\n * @param customSamplingContext Information given to the transaction sampling function (along with context-dependent\n * default values). See {@link Options.tracesSampler}.\n *\n * @returns The transaction which was just started\n *\n * @deprecated Use `startSpan()`, `startSpanManual()` or `startInactiveSpan()` instead.\n */\n startTransaction(context, customSamplingContext) {\n const result = this._callExtensionMethod('startTransaction', context, customSamplingContext);\n\n if (DEBUG_BUILD && !result) {\n // eslint-disable-next-line deprecation/deprecation\n const client = this.getClient();\n if (!client) {\n logger.warn(\n \"Tracing extension 'startTransaction' is missing. You should 'init' the SDK before calling 'startTransaction'\",\n );\n } else {\n logger.warn(`Tracing extension 'startTransaction' has not been added. Call 'addTracingExtensions' before calling 'init':\nSentry.addTracingExtensions();\nSentry.init({...});\n`);\n }\n }\n\n return result;\n }\n\n /**\n * @inheritDoc\n * @deprecated Use `spanToTraceHeader()` instead.\n */\n traceHeaders() {\n return this._callExtensionMethod('traceHeaders');\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use top level `captureSession` instead.\n */\n captureSession(endSession = false) {\n // both send the update and pull the session from the scope\n if (endSession) {\n // eslint-disable-next-line deprecation/deprecation\n return this.endSession();\n }\n\n // only send the update\n this._sendSessionUpdate();\n }\n\n /**\n * @inheritDoc\n * @deprecated Use top level `endSession` instead.\n */\n endSession() {\n // eslint-disable-next-line deprecation/deprecation\n const layer = this.getStackTop();\n const scope = layer.scope;\n const session = scope.getSession();\n if (session) {\n closeSession(session);\n }\n this._sendSessionUpdate();\n\n // the session is over; take it off of the scope\n scope.setSession();\n }\n\n /**\n * @inheritDoc\n * @deprecated Use top level `startSession` instead.\n */\n startSession(context) {\n // eslint-disable-next-line deprecation/deprecation\n const { scope, client } = this.getStackTop();\n const { release, environment = DEFAULT_ENVIRONMENT } = (client && client.getOptions()) || {};\n\n // Will fetch userAgent if called from browser sdk\n const { userAgent } = GLOBAL_OBJ.navigator || {};\n\n const session = makeSession({\n release,\n environment,\n user: scope.getUser(),\n ...(userAgent && { userAgent }),\n ...context,\n });\n\n // End existing session if there's one\n const currentSession = scope.getSession && scope.getSession();\n if (currentSession && currentSession.status === 'ok') {\n updateSession(currentSession, { status: 'exited' });\n }\n // eslint-disable-next-line deprecation/deprecation\n this.endSession();\n\n // Afterwards we set the new session on the scope\n scope.setSession(session);\n\n return session;\n }\n\n /**\n * Returns if default PII should be sent to Sentry and propagated in ourgoing requests\n * when Tracing is used.\n *\n * @deprecated Use top-level `getClient().getOptions().sendDefaultPii` instead. This function\n * only unnecessarily increased API surface but only wrapped accessing the option.\n */\n shouldSendDefaultPii() {\n // eslint-disable-next-line deprecation/deprecation\n const client = this.getClient();\n const options = client && client.getOptions();\n return Boolean(options && options.sendDefaultPii);\n }\n\n /**\n * Sends the current Session on the scope\n */\n _sendSessionUpdate() {\n // eslint-disable-next-line deprecation/deprecation\n const { scope, client } = this.getStackTop();\n\n const session = scope.getSession();\n if (session && client && client.captureSession) {\n client.captureSession(session);\n }\n }\n\n /**\n * Calls global extension method and binding current instance to the function call\n */\n // @ts-expect-error Function lacks ending return statement and return type does not include 'undefined'. ts(2366)\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n _callExtensionMethod(method, ...args) {\n const carrier = getMainCarrier();\n const sentry = carrier.__SENTRY__;\n if (sentry && sentry.extensions && typeof sentry.extensions[method] === 'function') {\n return sentry.extensions[method].apply(this, args);\n }\n DEBUG_BUILD && logger.warn(`Extension method ${method} couldn't be found, doing nothing.`);\n }\n}\n\n/**\n * Returns the global shim registry.\n *\n * FIXME: This function is problematic, because despite always returning a valid Carrier,\n * it has an optional `__SENTRY__` property, which then in turn requires us to always perform an unnecessary check\n * at the call-site. We always access the carrier through this function, so we can guarantee that `__SENTRY__` is there.\n **/\nfunction getMainCarrier() {\n GLOBAL_OBJ.__SENTRY__ = GLOBAL_OBJ.__SENTRY__ || {\n extensions: {},\n hub: undefined,\n };\n return GLOBAL_OBJ;\n}\n\n/**\n * Replaces the current main hub with the passed one on the global object\n *\n * @returns The old replaced hub\n *\n * @deprecated Use `setCurrentClient()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction makeMain(hub) {\n const registry = getMainCarrier();\n const oldHub = getHubFromCarrier(registry);\n setHubOnCarrier(registry, hub);\n return oldHub;\n}\n\n/**\n * Returns the default hub instance.\n *\n * If a hub is already registered in the global carrier but this module\n * contains a more recent version, it replaces the registered version.\n * Otherwise, the currently registered hub will be returned.\n *\n * @deprecated Use the respective replacement method directly instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction getCurrentHub() {\n // Get main carrier (global for every environment)\n const registry = getMainCarrier();\n\n if (registry.__SENTRY__ && registry.__SENTRY__.acs) {\n const hub = registry.__SENTRY__.acs.getCurrentHub();\n\n if (hub) {\n return hub;\n }\n }\n\n // Return hub that lives on a global object\n return getGlobalHub(registry);\n}\n\n/**\n * Get the currently active isolation scope.\n * The isolation scope is active for the current exection context,\n * meaning that it will remain stable for the same Hub.\n */\nfunction getIsolationScope() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().getIsolationScope();\n}\n\n// eslint-disable-next-line deprecation/deprecation\nfunction getGlobalHub(registry = getMainCarrier()) {\n // If there's no hub, or its an old API, assign a new one\n\n if (\n !hasHubOnCarrier(registry) ||\n // eslint-disable-next-line deprecation/deprecation\n getHubFromCarrier(registry).isOlderThan(API_VERSION)\n ) {\n // eslint-disable-next-line deprecation/deprecation\n setHubOnCarrier(registry, new Hub());\n }\n\n // Return hub that lives on a global object\n return getHubFromCarrier(registry);\n}\n\n/**\n * @private Private API with no semver guarantees!\n *\n * If the carrier does not contain a hub, a new hub is created with the global hub client and scope.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction ensureHubOnCarrier(carrier, parent = getGlobalHub()) {\n // If there's no hub on current domain, or it's an old API, assign a new one\n if (\n !hasHubOnCarrier(carrier) ||\n // eslint-disable-next-line deprecation/deprecation\n getHubFromCarrier(carrier).isOlderThan(API_VERSION)\n ) {\n // eslint-disable-next-line deprecation/deprecation\n const client = parent.getClient();\n // eslint-disable-next-line deprecation/deprecation\n const scope = parent.getScope();\n // eslint-disable-next-line deprecation/deprecation\n const isolationScope = parent.getIsolationScope();\n // eslint-disable-next-line deprecation/deprecation\n setHubOnCarrier(carrier, new Hub(client, scope.clone(), isolationScope.clone()));\n }\n}\n\n/**\n * @private Private API with no semver guarantees!\n *\n * Sets the global async context strategy\n */\nfunction setAsyncContextStrategy(strategy) {\n // Get main carrier (global for every environment)\n const registry = getMainCarrier();\n registry.__SENTRY__ = registry.__SENTRY__ || {};\n registry.__SENTRY__.acs = strategy;\n}\n\n/**\n * Runs the supplied callback in its own async context. Async Context strategies are defined per SDK.\n *\n * @param callback The callback to run in its own async context\n * @param options Options to pass to the async context strategy\n * @returns The result of the callback\n */\nfunction runWithAsyncContext(callback, options = {}) {\n const registry = getMainCarrier();\n\n if (registry.__SENTRY__ && registry.__SENTRY__.acs) {\n return registry.__SENTRY__.acs.runWithAsyncContext(callback, options);\n }\n\n // if there was no strategy, fallback to just calling the callback\n return callback();\n}\n\n/**\n * This will tell whether a carrier has a hub on it or not\n * @param carrier object\n */\nfunction hasHubOnCarrier(carrier) {\n return !!(carrier && carrier.__SENTRY__ && carrier.__SENTRY__.hub);\n}\n\n/**\n * This will create a new {@link Hub} and add to the passed object on\n * __SENTRY__.hub.\n * @param carrier object\n * @hidden\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction getHubFromCarrier(carrier) {\n // eslint-disable-next-line deprecation/deprecation\n return getGlobalSingleton('hub', () => new Hub(), carrier);\n}\n\n/**\n * This will set passed {@link Hub} on the passed object's __SENTRY__.hub attribute\n * @param carrier object\n * @param hub Hub\n * @returns A boolean indicating success or failure\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setHubOnCarrier(carrier, hub) {\n if (!carrier) return false;\n const __SENTRY__ = (carrier.__SENTRY__ = carrier.__SENTRY__ || {});\n __SENTRY__.hub = hub;\n return true;\n}\n\nexport { API_VERSION, Hub, ensureHubOnCarrier, getCurrentHub, getHubFromCarrier, getIsolationScope, getMainCarrier, makeMain, runWithAsyncContext, setAsyncContextStrategy, setHubOnCarrier };\n","import { node } from './node-stack-trace.js';\nexport { filenameIsInApp } from './node-stack-trace.js';\n\nconst STACKTRACE_FRAME_LIMIT = 50;\n// Used to sanitize webpack (error: *) wrapped stack errors\nconst WEBPACK_ERROR_REGEXP = /\\(error: (.*)\\)/;\nconst STRIP_FRAME_REGEXP = /captureMessage|captureException/;\n\n/**\n * Creates a stack parser with the supplied line parsers\n *\n * StackFrames are returned in the correct order for Sentry Exception\n * frames and with Sentry SDK internal frames removed from the top and bottom\n *\n */\nfunction createStackParser(...parsers) {\n const sortedParsers = parsers.sort((a, b) => a[0] - b[0]).map(p => p[1]);\n\n return (stack, skipFirst = 0) => {\n const frames = [];\n const lines = stack.split('\\n');\n\n for (let i = skipFirst; i < lines.length; i++) {\n const line = lines[i];\n // Ignore lines over 1kb as they are unlikely to be stack frames.\n // Many of the regular expressions use backtracking which results in run time that increases exponentially with\n // input size. Huge strings can result in hangs/Denial of Service:\n // https://github.com/getsentry/sentry-javascript/issues/2286\n if (line.length > 1024) {\n continue;\n }\n\n // https://github.com/getsentry/sentry-javascript/issues/5459\n // Remove webpack (error: *) wrappers\n const cleanedLine = WEBPACK_ERROR_REGEXP.test(line) ? line.replace(WEBPACK_ERROR_REGEXP, '$1') : line;\n\n // https://github.com/getsentry/sentry-javascript/issues/7813\n // Skip Error: lines\n if (cleanedLine.match(/\\S*Error: /)) {\n continue;\n }\n\n for (const parser of sortedParsers) {\n const frame = parser(cleanedLine);\n\n if (frame) {\n frames.push(frame);\n break;\n }\n }\n\n if (frames.length >= STACKTRACE_FRAME_LIMIT) {\n break;\n }\n }\n\n return stripSentryFramesAndReverse(frames);\n };\n}\n\n/**\n * Gets a stack parser implementation from Options.stackParser\n * @see Options\n *\n * If options contains an array of line parsers, it is converted into a parser\n */\nfunction stackParserFromStackParserOptions(stackParser) {\n if (Array.isArray(stackParser)) {\n return createStackParser(...stackParser);\n }\n return stackParser;\n}\n\n/**\n * Removes Sentry frames from the top and bottom of the stack if present and enforces a limit of max number of frames.\n * Assumes stack input is ordered from top to bottom and returns the reverse representation so call site of the\n * function that caused the crash is the last frame in the array.\n * @hidden\n */\nfunction stripSentryFramesAndReverse(stack) {\n if (!stack.length) {\n return [];\n }\n\n const localStack = Array.from(stack);\n\n // If stack starts with one of our API calls, remove it (starts, meaning it's the top of the stack - aka last call)\n if (/sentryWrapped/.test(localStack[localStack.length - 1].function || '')) {\n localStack.pop();\n }\n\n // Reversing in the middle of the procedure allows us to just pop the values off the stack\n localStack.reverse();\n\n // If stack ends with one of our internal API calls, remove it (ends, meaning it's the bottom of the stack - aka top-most call)\n if (STRIP_FRAME_REGEXP.test(localStack[localStack.length - 1].function || '')) {\n localStack.pop();\n\n // When using synthetic events, we will have a 2 levels deep stack, as `new Error('Sentry syntheticException')`\n // is produced within the hub itself, making it:\n //\n // Sentry.captureException()\n // getCurrentHub().captureException()\n //\n // instead of just the top `Sentry` call itself.\n // This forces us to possibly strip an additional frame in the exact same was as above.\n if (STRIP_FRAME_REGEXP.test(localStack[localStack.length - 1].function || '')) {\n localStack.pop();\n }\n }\n\n return localStack.slice(0, STACKTRACE_FRAME_LIMIT).map(frame => ({\n ...frame,\n filename: frame.filename || localStack[localStack.length - 1].filename,\n function: frame.function || '?',\n }));\n}\n\nconst defaultFunctionName = '';\n\n/**\n * Safely extract function name from itself\n */\nfunction getFunctionName(fn) {\n try {\n if (!fn || typeof fn !== 'function') {\n return defaultFunctionName;\n }\n return fn.name || defaultFunctionName;\n } catch (e) {\n // Just accessing custom props in some Selenium environments\n // can cause a \"Permission denied\" exception (see raven-js#495).\n return defaultFunctionName;\n }\n}\n\n/**\n * Node.js stack line parser\n *\n * This is in @sentry/utils so it can be used from the Electron SDK in the browser for when `nodeIntegration == true`.\n * This allows it to be used without referencing or importing any node specific code which causes bundlers to complain\n */\nfunction nodeStackLineParser(getModule) {\n return [90, node(getModule)];\n}\n\nexport { createStackParser, getFunctionName, nodeStackLineParser, stackParserFromStackParserOptions, stripSentryFramesAndReverse };\n","import { isNaN, isVueViewModel, isSyntheticEvent } from './is.js';\nimport { memoBuilder } from './memo.js';\nimport { convertToPlainObject } from './object.js';\nimport { getFunctionName } from './stacktrace.js';\n\n/**\n * Recursively normalizes the given object.\n *\n * - Creates a copy to prevent original input mutation\n * - Skips non-enumerable properties\n * - When stringifying, calls `toJSON` if implemented\n * - Removes circular references\n * - Translates non-serializable values (`undefined`/`NaN`/functions) to serializable format\n * - Translates known global objects/classes to a string representations\n * - Takes care of `Error` object serialization\n * - Optionally limits depth of final output\n * - Optionally limits number of properties/elements included in any single object/array\n *\n * @param input The object to be normalized.\n * @param depth The max depth to which to normalize the object. (Anything deeper stringified whole.)\n * @param maxProperties The max number of elements or properties to be included in any single array or\n * object in the normallized output.\n * @returns A normalized version of the object, or `\"**non-serializable**\"` if any errors are thrown during normalization.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction normalize(input, depth = 100, maxProperties = +Infinity) {\n try {\n // since we're at the outermost level, we don't provide a key\n return visit('', input, depth, maxProperties);\n } catch (err) {\n return { ERROR: `**non-serializable** (${err})` };\n }\n}\n\n/** JSDoc */\nfunction normalizeToSize(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n object,\n // Default Node.js REPL depth\n depth = 3,\n // 100kB, as 200kB is max payload size, so half sounds reasonable\n maxSize = 100 * 1024,\n) {\n const normalized = normalize(object, depth);\n\n if (jsonSize(normalized) > maxSize) {\n return normalizeToSize(object, depth - 1, maxSize);\n }\n\n return normalized ;\n}\n\n/**\n * Visits a node to perform normalization on it\n *\n * @param key The key corresponding to the given node\n * @param value The node to be visited\n * @param depth Optional number indicating the maximum recursion depth\n * @param maxProperties Optional maximum number of properties/elements included in any single object/array\n * @param memo Optional Memo class handling decycling\n */\nfunction visit(\n key,\n value,\n depth = +Infinity,\n maxProperties = +Infinity,\n memo = memoBuilder(),\n) {\n const [memoize, unmemoize] = memo;\n\n // Get the simple cases out of the way first\n if (\n value == null || // this matches null and undefined -> eqeq not eqeqeq\n (['number', 'boolean', 'string'].includes(typeof value) && !isNaN(value))\n ) {\n return value ;\n }\n\n const stringified = stringifyValue(key, value);\n\n // Anything we could potentially dig into more (objects or arrays) will have come back as `\"[object XXXX]\"`.\n // Everything else will have already been serialized, so if we don't see that pattern, we're done.\n if (!stringified.startsWith('[object ')) {\n return stringified;\n }\n\n // From here on, we can assert that `value` is either an object or an array.\n\n // Do not normalize objects that we know have already been normalized. As a general rule, the\n // \"__sentry_skip_normalization__\" property should only be used sparingly and only should only be set on objects that\n // have already been normalized.\n if ((value )['__sentry_skip_normalization__']) {\n return value ;\n }\n\n // We can set `__sentry_override_normalization_depth__` on an object to ensure that from there\n // We keep a certain amount of depth.\n // This should be used sparingly, e.g. we use it for the redux integration to ensure we get a certain amount of state.\n const remainingDepth =\n typeof (value )['__sentry_override_normalization_depth__'] === 'number'\n ? ((value )['__sentry_override_normalization_depth__'] )\n : depth;\n\n // We're also done if we've reached the max depth\n if (remainingDepth === 0) {\n // At this point we know `serialized` is a string of the form `\"[object XXXX]\"`. Clean it up so it's just `\"[XXXX]\"`.\n return stringified.replace('object ', '');\n }\n\n // If we've already visited this branch, bail out, as it's circular reference. If not, note that we're seeing it now.\n if (memoize(value)) {\n return '[Circular ~]';\n }\n\n // If the value has a `toJSON` method, we call it to extract more information\n const valueWithToJSON = value ;\n if (valueWithToJSON && typeof valueWithToJSON.toJSON === 'function') {\n try {\n const jsonValue = valueWithToJSON.toJSON();\n // We need to normalize the return value of `.toJSON()` in case it has circular references\n return visit('', jsonValue, remainingDepth - 1, maxProperties, memo);\n } catch (err) {\n // pass (The built-in `toJSON` failed, but we can still try to do it ourselves)\n }\n }\n\n // At this point we know we either have an object or an array, we haven't seen it before, and we're going to recurse\n // because we haven't yet reached the max depth. Create an accumulator to hold the results of visiting each\n // property/entry, and keep track of the number of items we add to it.\n const normalized = (Array.isArray(value) ? [] : {}) ;\n let numAdded = 0;\n\n // Before we begin, convert`Error` and`Event` instances into plain objects, since some of each of their relevant\n // properties are non-enumerable and otherwise would get missed.\n const visitable = convertToPlainObject(value );\n\n for (const visitKey in visitable) {\n // Avoid iterating over fields in the prototype if they've somehow been exposed to enumeration.\n if (!Object.prototype.hasOwnProperty.call(visitable, visitKey)) {\n continue;\n }\n\n if (numAdded >= maxProperties) {\n normalized[visitKey] = '[MaxProperties ~]';\n break;\n }\n\n // Recursively visit all the child nodes\n const visitValue = visitable[visitKey];\n normalized[visitKey] = visit(visitKey, visitValue, remainingDepth - 1, maxProperties, memo);\n\n numAdded++;\n }\n\n // Once we've visited all the branches, remove the parent from memo storage\n unmemoize(value);\n\n // Return accumulated values\n return normalized;\n}\n\n/* eslint-disable complexity */\n/**\n * Stringify the given value. Handles various known special values and types.\n *\n * Not meant to be used on simple primitives which already have a string representation, as it will, for example, turn\n * the number 1231 into \"[Object Number]\", nor on `null`, as it will throw.\n *\n * @param value The value to stringify\n * @returns A stringified representation of the given value\n */\nfunction stringifyValue(\n key,\n // this type is a tiny bit of a cheat, since this function does handle NaN (which is technically a number), but for\n // our internal use, it'll do\n value,\n) {\n try {\n if (key === 'domain' && value && typeof value === 'object' && (value )._events) {\n return '[Domain]';\n }\n\n if (key === 'domainEmitter') {\n return '[DomainEmitter]';\n }\n\n // It's safe to use `global`, `window`, and `document` here in this manner, as we are asserting using `typeof` first\n // which won't throw if they are not present.\n\n if (typeof global !== 'undefined' && value === global) {\n return '[Global]';\n }\n\n // eslint-disable-next-line no-restricted-globals\n if (typeof window !== 'undefined' && value === window) {\n return '[Window]';\n }\n\n // eslint-disable-next-line no-restricted-globals\n if (typeof document !== 'undefined' && value === document) {\n return '[Document]';\n }\n\n if (isVueViewModel(value)) {\n return '[VueViewModel]';\n }\n\n // React's SyntheticEvent thingy\n if (isSyntheticEvent(value)) {\n return '[SyntheticEvent]';\n }\n\n if (typeof value === 'number' && value !== value) {\n return '[NaN]';\n }\n\n if (typeof value === 'function') {\n return `[Function: ${getFunctionName(value)}]`;\n }\n\n if (typeof value === 'symbol') {\n return `[${String(value)}]`;\n }\n\n // stringified BigInts are indistinguishable from regular numbers, so we need to label them to avoid confusion\n if (typeof value === 'bigint') {\n return `[BigInt: ${String(value)}]`;\n }\n\n // Now that we've knocked out all the special cases and the primitives, all we have left are objects. Simply casting\n // them to strings means that instances of classes which haven't defined their `toStringTag` will just come out as\n // `\"[object Object]\"`. If we instead look at the constructor's name (which is the same as the name of the class),\n // we can make sure that only plain objects come out that way.\n const objName = getConstructorName(value);\n\n // Handle HTML Elements\n if (/^HTML(\\w*)Element$/.test(objName)) {\n return `[HTMLElement: ${objName}]`;\n }\n\n return `[object ${objName}]`;\n } catch (err) {\n return `**non-serializable** (${err})`;\n }\n}\n/* eslint-enable complexity */\n\nfunction getConstructorName(value) {\n const prototype = Object.getPrototypeOf(value);\n\n return prototype ? prototype.constructor.name : 'null prototype';\n}\n\n/** Calculates bytes size of input string */\nfunction utf8Length(value) {\n // eslint-disable-next-line no-bitwise\n return ~-encodeURI(value).split(/%..|./).length;\n}\n\n/** Calculates bytes size of input object */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction jsonSize(value) {\n return utf8Length(JSON.stringify(value));\n}\n\n/**\n * Normalizes URLs in exceptions and stacktraces to a base path so Sentry can fingerprint\n * across platforms and working directory.\n *\n * @param url The URL to be normalized.\n * @param basePath The application base path.\n * @returns The normalized URL.\n */\nfunction normalizeUrlToBase(url, basePath) {\n const escapedBase = basePath\n // Backslash to forward\n .replace(/\\\\/g, '/')\n // Escape RegExp special characters\n .replace(/[|\\\\{}()[\\]^$+*?.]/g, '\\\\$&');\n\n let newUrl = url;\n try {\n newUrl = decodeURI(url);\n } catch (_Oo) {\n // Sometime this breaks\n }\n return (\n newUrl\n .replace(/\\\\/g, '/')\n .replace(/webpack:\\/?/g, '') // Remove intermediate base path\n // eslint-disable-next-line @sentry-internal/sdk/no-regexp-constructor\n .replace(new RegExp(`(file://)?/*${escapedBase}/*`, 'ig'), 'app:///')\n );\n}\n\nexport { normalize, normalizeToSize, normalizeUrlToBase, visit as walk };\n","/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n/* eslint-disable @typescript-eslint/no-explicit-any */\n\n/**\n * Helper to decycle json objects\n */\nfunction memoBuilder() {\n const hasWeakSet = typeof WeakSet === 'function';\n const inner = hasWeakSet ? new WeakSet() : [];\n function memoize(obj) {\n if (hasWeakSet) {\n if (inner.has(obj)) {\n return true;\n }\n inner.add(obj);\n return false;\n }\n // eslint-disable-next-line @typescript-eslint/prefer-for-of\n for (let i = 0; i < inner.length; i++) {\n const value = inner[i];\n if (value === obj) {\n return true;\n }\n }\n inner.push(obj);\n return false;\n }\n\n function unmemoize(obj) {\n if (hasWeakSet) {\n inner.delete(obj);\n } else {\n for (let i = 0; i < inner.length; i++) {\n if (inner[i] === obj) {\n inner.splice(i, 1);\n break;\n }\n }\n }\n }\n return [memoize, unmemoize];\n}\n\nexport { memoBuilder };\n","import { uuid4, dateTimestampInSeconds, addExceptionMechanism, truncate, GLOBAL_OBJ, normalize } from '@sentry/utils';\nimport { DEFAULT_ENVIRONMENT } from '../constants.js';\nimport { getGlobalEventProcessors, notifyEventProcessors } from '../eventProcessors.js';\nimport { getGlobalScope, Scope } from '../scope.js';\nimport { mergeScopeData, applyScopeDataToEvent } from './applyScopeDataToEvent.js';\nimport { spanToJSON } from './spanUtils.js';\n\n/**\n * This type makes sure that we get either a CaptureContext, OR an EventHint.\n * It does not allow mixing them, which could lead to unexpected outcomes, e.g. this is disallowed:\n * { user: { id: '123' }, mechanism: { handled: false } }\n */\n\n/**\n * Adds common information to events.\n *\n * The information includes release and environment from `options`,\n * breadcrumbs and context (extra, tags and user) from the scope.\n *\n * Information that is already present in the event is never overwritten. For\n * nested objects, such as the context, keys are merged.\n *\n * Note: This also triggers callbacks for `addGlobalEventProcessor`, but not `beforeSend`.\n *\n * @param event The original event.\n * @param hint May contain additional information about the original exception.\n * @param scope A scope containing event metadata.\n * @returns A new event with more information.\n * @hidden\n */\nfunction prepareEvent(\n options,\n event,\n hint,\n scope,\n client,\n isolationScope,\n) {\n const { normalizeDepth = 3, normalizeMaxBreadth = 1000 } = options;\n const prepared = {\n ...event,\n event_id: event.event_id || hint.event_id || uuid4(),\n timestamp: event.timestamp || dateTimestampInSeconds(),\n };\n const integrations = hint.integrations || options.integrations.map(i => i.name);\n\n applyClientOptions(prepared, options);\n applyIntegrationsMetadata(prepared, integrations);\n\n // Only put debug IDs onto frames for error events.\n if (event.type === undefined) {\n applyDebugIds(prepared, options.stackParser);\n }\n\n // If we have scope given to us, use it as the base for further modifications.\n // This allows us to prevent unnecessary copying of data if `captureContext` is not provided.\n const finalScope = getFinalScope(scope, hint.captureContext);\n\n if (hint.mechanism) {\n addExceptionMechanism(prepared, hint.mechanism);\n }\n\n const clientEventProcessors = client && client.getEventProcessors ? client.getEventProcessors() : [];\n\n // This should be the last thing called, since we want that\n // {@link Hub.addEventProcessor} gets the finished prepared event.\n // Merge scope data together\n const data = getGlobalScope().getScopeData();\n\n if (isolationScope) {\n const isolationData = isolationScope.getScopeData();\n mergeScopeData(data, isolationData);\n }\n\n if (finalScope) {\n const finalScopeData = finalScope.getScopeData();\n mergeScopeData(data, finalScopeData);\n }\n\n const attachments = [...(hint.attachments || []), ...data.attachments];\n if (attachments.length) {\n hint.attachments = attachments;\n }\n\n applyScopeDataToEvent(prepared, data);\n\n // TODO (v8): Update this order to be: Global > Client > Scope\n const eventProcessors = [\n ...clientEventProcessors,\n // eslint-disable-next-line deprecation/deprecation\n ...getGlobalEventProcessors(),\n // Run scope event processors _after_ all other processors\n ...data.eventProcessors,\n ];\n\n const result = notifyEventProcessors(eventProcessors, prepared, hint);\n\n return result.then(evt => {\n if (evt) {\n // We apply the debug_meta field only after all event processors have ran, so that if any event processors modified\n // file names (e.g.the RewriteFrames integration) the filename -> debug ID relationship isn't destroyed.\n // This should not cause any PII issues, since we're only moving data that is already on the event and not adding\n // any new data\n applyDebugMeta(evt);\n }\n\n if (typeof normalizeDepth === 'number' && normalizeDepth > 0) {\n return normalizeEvent(evt, normalizeDepth, normalizeMaxBreadth);\n }\n return evt;\n });\n}\n\n/**\n * Enhances event using the client configuration.\n * It takes care of all \"static\" values like environment, release and `dist`,\n * as well as truncating overly long values.\n * @param event event instance to be enhanced\n */\nfunction applyClientOptions(event, options) {\n const { environment, release, dist, maxValueLength = 250 } = options;\n\n if (!('environment' in event)) {\n event.environment = 'environment' in options ? environment : DEFAULT_ENVIRONMENT;\n }\n\n if (event.release === undefined && release !== undefined) {\n event.release = release;\n }\n\n if (event.dist === undefined && dist !== undefined) {\n event.dist = dist;\n }\n\n if (event.message) {\n event.message = truncate(event.message, maxValueLength);\n }\n\n const exception = event.exception && event.exception.values && event.exception.values[0];\n if (exception && exception.value) {\n exception.value = truncate(exception.value, maxValueLength);\n }\n\n const request = event.request;\n if (request && request.url) {\n request.url = truncate(request.url, maxValueLength);\n }\n}\n\nconst debugIdStackParserCache = new WeakMap();\n\n/**\n * Puts debug IDs into the stack frames of an error event.\n */\nfunction applyDebugIds(event, stackParser) {\n const debugIdMap = GLOBAL_OBJ._sentryDebugIds;\n\n if (!debugIdMap) {\n return;\n }\n\n let debugIdStackFramesCache;\n const cachedDebugIdStackFrameCache = debugIdStackParserCache.get(stackParser);\n if (cachedDebugIdStackFrameCache) {\n debugIdStackFramesCache = cachedDebugIdStackFrameCache;\n } else {\n debugIdStackFramesCache = new Map();\n debugIdStackParserCache.set(stackParser, debugIdStackFramesCache);\n }\n\n // Build a map of filename -> debug_id\n const filenameDebugIdMap = Object.keys(debugIdMap).reduce((acc, debugIdStackTrace) => {\n let parsedStack;\n const cachedParsedStack = debugIdStackFramesCache.get(debugIdStackTrace);\n if (cachedParsedStack) {\n parsedStack = cachedParsedStack;\n } else {\n parsedStack = stackParser(debugIdStackTrace);\n debugIdStackFramesCache.set(debugIdStackTrace, parsedStack);\n }\n\n for (let i = parsedStack.length - 1; i >= 0; i--) {\n const stackFrame = parsedStack[i];\n if (stackFrame.filename) {\n acc[stackFrame.filename] = debugIdMap[debugIdStackTrace];\n break;\n }\n }\n return acc;\n }, {});\n\n try {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n event.exception.values.forEach(exception => {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n exception.stacktrace.frames.forEach(frame => {\n if (frame.filename) {\n frame.debug_id = filenameDebugIdMap[frame.filename];\n }\n });\n });\n } catch (e) {\n // To save bundle size we're just try catching here instead of checking for the existence of all the different objects.\n }\n}\n\n/**\n * Moves debug IDs from the stack frames of an error event into the debug_meta field.\n */\nfunction applyDebugMeta(event) {\n // Extract debug IDs and filenames from the stack frames on the event.\n const filenameDebugIdMap = {};\n try {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n event.exception.values.forEach(exception => {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n exception.stacktrace.frames.forEach(frame => {\n if (frame.debug_id) {\n if (frame.abs_path) {\n filenameDebugIdMap[frame.abs_path] = frame.debug_id;\n } else if (frame.filename) {\n filenameDebugIdMap[frame.filename] = frame.debug_id;\n }\n delete frame.debug_id;\n }\n });\n });\n } catch (e) {\n // To save bundle size we're just try catching here instead of checking for the existence of all the different objects.\n }\n\n if (Object.keys(filenameDebugIdMap).length === 0) {\n return;\n }\n\n // Fill debug_meta information\n event.debug_meta = event.debug_meta || {};\n event.debug_meta.images = event.debug_meta.images || [];\n const images = event.debug_meta.images;\n Object.keys(filenameDebugIdMap).forEach(filename => {\n images.push({\n type: 'sourcemap',\n code_file: filename,\n debug_id: filenameDebugIdMap[filename],\n });\n });\n}\n\n/**\n * This function adds all used integrations to the SDK info in the event.\n * @param event The event that will be filled with all integrations.\n */\nfunction applyIntegrationsMetadata(event, integrationNames) {\n if (integrationNames.length > 0) {\n event.sdk = event.sdk || {};\n event.sdk.integrations = [...(event.sdk.integrations || []), ...integrationNames];\n }\n}\n\n/**\n * Applies `normalize` function on necessary `Event` attributes to make them safe for serialization.\n * Normalized keys:\n * - `breadcrumbs.data`\n * - `user`\n * - `contexts`\n * - `extra`\n * @param event Event\n * @returns Normalized event\n */\nfunction normalizeEvent(event, depth, maxBreadth) {\n if (!event) {\n return null;\n }\n\n const normalized = {\n ...event,\n ...(event.breadcrumbs && {\n breadcrumbs: event.breadcrumbs.map(b => ({\n ...b,\n ...(b.data && {\n data: normalize(b.data, depth, maxBreadth),\n }),\n })),\n }),\n ...(event.user && {\n user: normalize(event.user, depth, maxBreadth),\n }),\n ...(event.contexts && {\n contexts: normalize(event.contexts, depth, maxBreadth),\n }),\n ...(event.extra && {\n extra: normalize(event.extra, depth, maxBreadth),\n }),\n };\n\n // event.contexts.trace stores information about a Transaction. Similarly,\n // event.spans[] stores information about child Spans. Given that a\n // Transaction is conceptually a Span, normalization should apply to both\n // Transactions and Spans consistently.\n // For now the decision is to skip normalization of Transactions and Spans,\n // so this block overwrites the normalized event to add back the original\n // Transaction information prior to normalization.\n if (event.contexts && event.contexts.trace && normalized.contexts) {\n normalized.contexts.trace = event.contexts.trace;\n\n // event.contexts.trace.data may contain circular/dangerous data so we need to normalize it\n if (event.contexts.trace.data) {\n normalized.contexts.trace.data = normalize(event.contexts.trace.data, depth, maxBreadth);\n }\n }\n\n // event.spans[].data may contain circular/dangerous data so we need to normalize it\n if (event.spans) {\n normalized.spans = event.spans.map(span => {\n const data = spanToJSON(span).data;\n\n if (data) {\n // This is a bit weird, as we generally have `Span` instances here, but to be safe we do not assume so\n // eslint-disable-next-line deprecation/deprecation\n span.data = normalize(data, depth, maxBreadth);\n }\n\n return span;\n });\n }\n\n return normalized;\n}\n\nfunction getFinalScope(scope, captureContext) {\n if (!captureContext) {\n return scope;\n }\n\n const finalScope = scope ? scope.clone() : new Scope();\n finalScope.update(captureContext);\n return finalScope;\n}\n\n/**\n * Parse either an `EventHint` directly, or convert a `CaptureContext` to an `EventHint`.\n * This is used to allow to update method signatures that used to accept a `CaptureContext` but should now accept an `EventHint`.\n */\nfunction parseEventHintOrCaptureContext(\n hint,\n) {\n if (!hint) {\n return undefined;\n }\n\n // If you pass a Scope or `() => Scope` as CaptureContext, we just return this as captureContext\n if (hintIsScopeOrFunction(hint)) {\n return { captureContext: hint };\n }\n\n if (hintIsScopeContext(hint)) {\n return {\n captureContext: hint,\n };\n }\n\n return hint;\n}\n\nfunction hintIsScopeOrFunction(\n hint,\n) {\n return hint instanceof Scope || typeof hint === 'function';\n}\n\nconst captureContextKeys = [\n 'user',\n 'level',\n 'extra',\n 'contexts',\n 'tags',\n 'fingerprint',\n 'requestSession',\n 'propagationContext',\n] ;\n\nfunction hintIsScopeContext(hint) {\n return Object.keys(hint).some(key => captureContextKeys.includes(key ));\n}\n\nexport { applyDebugIds, applyDebugMeta, parseEventHintOrCaptureContext, prepareEvent };\n","import { logger, uuid4, timestampInSeconds, isThenable, GLOBAL_OBJ } from '@sentry/utils';\nimport { DEFAULT_ENVIRONMENT } from './constants.js';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { getCurrentHub, runWithAsyncContext, getIsolationScope } from './hub.js';\nimport { makeSession, updateSession, closeSession } from './session.js';\nimport { parseEventHintOrCaptureContext } from './utils/prepareEvent.js';\n\n/**\n * Captures an exception event and sends it to Sentry.\n *\n * @param exception The exception to capture.\n * @param hint Optional additional data to attach to the Sentry event.\n * @returns the id of the captured Sentry event.\n */\nfunction captureException(\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n exception,\n hint,\n) {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().captureException(exception, parseEventHintOrCaptureContext(hint));\n}\n\n/**\n * Captures a message event and sends it to Sentry.\n *\n * @param exception The exception to capture.\n * @param captureContext Define the level of the message or pass in additional data to attach to the message.\n * @returns the id of the captured message.\n */\nfunction captureMessage(\n message,\n // eslint-disable-next-line deprecation/deprecation\n captureContext,\n) {\n // This is necessary to provide explicit scopes upgrade, without changing the original\n // arity of the `captureMessage(message, level)` method.\n const level = typeof captureContext === 'string' ? captureContext : undefined;\n const context = typeof captureContext !== 'string' ? { captureContext } : undefined;\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().captureMessage(message, level, context);\n}\n\n/**\n * Captures a manually created event and sends it to Sentry.\n *\n * @param exception The event to send to Sentry.\n * @param hint Optional additional data to attach to the Sentry event.\n * @returns the id of the captured event.\n */\nfunction captureEvent(event, hint) {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().captureEvent(event, hint);\n}\n\n/**\n * Callback to set context information onto the scope.\n * @param callback Callback function that receives Scope.\n *\n * @deprecated Use getCurrentScope() directly.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction configureScope(callback) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().configureScope(callback);\n}\n\n/**\n * Records a new breadcrumb which will be attached to future events.\n *\n * Breadcrumbs will be added to subsequent events to provide more context on\n * user's actions prior to an error or crash.\n *\n * @param breadcrumb The breadcrumb to record.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction addBreadcrumb(breadcrumb, hint) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().addBreadcrumb(breadcrumb, hint);\n}\n\n/**\n * Sets context data with the given name.\n * @param name of the context\n * @param context Any kind of data. This data will be normalized.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any, deprecation/deprecation\nfunction setContext(name, context) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setContext(name, context);\n}\n\n/**\n * Set an object that will be merged sent as extra data with the event.\n * @param extras Extras object to merge into current context.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setExtras(extras) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setExtras(extras);\n}\n\n/**\n * Set key:value that will be sent as extra data with the event.\n * @param key String of extra\n * @param extra Any kind of data. This data will be normalized.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setExtra(key, extra) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setExtra(key, extra);\n}\n\n/**\n * Set an object that will be merged sent as tags data with the event.\n * @param tags Tags context object to merge into current context.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setTags(tags) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setTags(tags);\n}\n\n/**\n * Set key:value that will be sent as tags data with the event.\n *\n * Can also be used to unset a tag, by passing `undefined`.\n *\n * @param key String key of tag\n * @param value Value of tag\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setTag(key, value) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setTag(key, value);\n}\n\n/**\n * Updates user context information for future events.\n *\n * @param user User context object to be set in the current context. Pass `null` to unset the user.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction setUser(user) {\n // eslint-disable-next-line deprecation/deprecation\n getCurrentHub().setUser(user);\n}\n\n/**\n * Creates a new scope with and executes the given operation within.\n * The scope is automatically removed once the operation\n * finishes or throws.\n *\n * This is essentially a convenience function for:\n *\n * pushScope();\n * callback();\n * popScope();\n */\n\n/**\n * Either creates a new active scope, or sets the given scope as active scope in the given callback.\n */\nfunction withScope(\n ...rest\n) {\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n\n // If a scope is defined, we want to make this the active scope instead of the default one\n if (rest.length === 2) {\n const [scope, callback] = rest;\n if (!scope) {\n // eslint-disable-next-line deprecation/deprecation\n return hub.withScope(callback);\n }\n\n // eslint-disable-next-line deprecation/deprecation\n return hub.withScope(() => {\n // eslint-disable-next-line deprecation/deprecation\n hub.getStackTop().scope = scope ;\n return callback(scope );\n });\n }\n\n // eslint-disable-next-line deprecation/deprecation\n return hub.withScope(rest[0]);\n}\n\n/**\n * Attempts to fork the current isolation scope and the current scope based on the current async context strategy. If no\n * async context strategy is set, the isolation scope and the current scope will not be forked (this is currently the\n * case, for example, in the browser).\n *\n * Usage of this function in environments without async context strategy is discouraged and may lead to unexpected behaviour.\n *\n * This function is intended for Sentry SDK and SDK integration development. It is not recommended to be used in \"normal\"\n * applications directly because it comes with pitfalls. Use at your own risk!\n *\n * @param callback The callback in which the passed isolation scope is active. (Note: In environments without async\n * context strategy, the currently active isolation scope may change within execution of the callback.)\n * @returns The same value that `callback` returns.\n */\nfunction withIsolationScope(callback) {\n return runWithAsyncContext(() => {\n return callback(getIsolationScope());\n });\n}\n\n/**\n * Forks the current scope and sets the provided span as active span in the context of the provided callback.\n *\n * @param span Spans started in the context of the provided callback will be children of this span.\n * @param callback Execution context in which the provided span will be active. Is passed the newly forked scope.\n * @returns the value returned from the provided callback function.\n */\nfunction withActiveSpan(span, callback) {\n return withScope(scope => {\n // eslint-disable-next-line deprecation/deprecation\n scope.setSpan(span);\n return callback(scope);\n });\n}\n\n/**\n * Starts a new `Transaction` and returns it. This is the entry point to manual tracing instrumentation.\n *\n * A tree structure can be built by adding child spans to the transaction, and child spans to other spans. To start a\n * new child span within the transaction or any span, call the respective `.startChild()` method.\n *\n * Every child span must be finished before the transaction is finished, otherwise the unfinished spans are discarded.\n *\n * The transaction must be finished with a call to its `.end()` method, at which point the transaction with all its\n * finished child spans will be sent to Sentry.\n *\n * NOTE: This function should only be used for *manual* instrumentation. Auto-instrumentation should call\n * `startTransaction` directly on the hub.\n *\n * @param context Properties of the new `Transaction`.\n * @param customSamplingContext Information given to the transaction sampling function (along with context-dependent\n * default values). See {@link Options.tracesSampler}.\n *\n * @returns The transaction which was just started\n *\n * @deprecated Use `startSpan()`, `startSpanManual()` or `startInactiveSpan()` instead.\n */\nfunction startTransaction(\n context,\n customSamplingContext,\n // eslint-disable-next-line deprecation/deprecation\n) {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().startTransaction({ ...context }, customSamplingContext);\n}\n\n/**\n * Create a cron monitor check in and send it to Sentry.\n *\n * @param checkIn An object that describes a check in.\n * @param upsertMonitorConfig An optional object that describes a monitor config. Use this if you want\n * to create a monitor automatically when sending a check in.\n */\nfunction captureCheckIn(checkIn, upsertMonitorConfig) {\n const scope = getCurrentScope();\n const client = getClient();\n if (!client) {\n DEBUG_BUILD && logger.warn('Cannot capture check-in. No client defined.');\n } else if (!client.captureCheckIn) {\n DEBUG_BUILD && logger.warn('Cannot capture check-in. Client does not support sending check-ins.');\n } else {\n return client.captureCheckIn(checkIn, upsertMonitorConfig, scope);\n }\n\n return uuid4();\n}\n\n/**\n * Wraps a callback with a cron monitor check in. The check in will be sent to Sentry when the callback finishes.\n *\n * @param monitorSlug The distinct slug of the monitor.\n * @param upsertMonitorConfig An optional object that describes a monitor config. Use this if you want\n * to create a monitor automatically when sending a check in.\n */\nfunction withMonitor(\n monitorSlug,\n callback,\n upsertMonitorConfig,\n) {\n const checkInId = captureCheckIn({ monitorSlug, status: 'in_progress' }, upsertMonitorConfig);\n const now = timestampInSeconds();\n\n function finishCheckIn(status) {\n captureCheckIn({ monitorSlug, status, checkInId, duration: timestampInSeconds() - now });\n }\n\n let maybePromiseResult;\n try {\n maybePromiseResult = callback();\n } catch (e) {\n finishCheckIn('error');\n throw e;\n }\n\n if (isThenable(maybePromiseResult)) {\n Promise.resolve(maybePromiseResult).then(\n () => {\n finishCheckIn('ok');\n },\n () => {\n finishCheckIn('error');\n },\n );\n } else {\n finishCheckIn('ok');\n }\n\n return maybePromiseResult;\n}\n\n/**\n * Call `flush()` on the current client, if there is one. See {@link Client.flush}.\n *\n * @param timeout Maximum time in ms the client should wait to flush its event queue. Omitting this parameter will cause\n * the client to wait until all events are sent before resolving the promise.\n * @returns A promise which resolves to `true` if the queue successfully drains before the timeout, or `false` if it\n * doesn't (or if there's no client defined).\n */\nasync function flush(timeout) {\n const client = getClient();\n if (client) {\n return client.flush(timeout);\n }\n DEBUG_BUILD && logger.warn('Cannot flush events. No client defined.');\n return Promise.resolve(false);\n}\n\n/**\n * Call `close()` on the current client, if there is one. See {@link Client.close}.\n *\n * @param timeout Maximum time in ms the client should wait to flush its event queue before shutting down. Omitting this\n * parameter will cause the client to wait until all events are sent before disabling itself.\n * @returns A promise which resolves to `true` if the queue successfully drains before the timeout, or `false` if it\n * doesn't (or if there's no client defined).\n */\nasync function close(timeout) {\n const client = getClient();\n if (client) {\n return client.close(timeout);\n }\n DEBUG_BUILD && logger.warn('Cannot flush events and disable SDK. No client defined.');\n return Promise.resolve(false);\n}\n\n/**\n * This is the getter for lastEventId.\n *\n * @returns The last event id of a captured event.\n */\nfunction lastEventId() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().lastEventId();\n}\n\n/**\n * Get the currently active client.\n */\nfunction getClient() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().getClient();\n}\n\n/**\n * Returns true if Sentry has been properly initialized.\n */\nfunction isInitialized() {\n return !!getClient();\n}\n\n/**\n * Get the currently active scope.\n */\nfunction getCurrentScope() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentHub().getScope();\n}\n\n/**\n * Start a session on the current isolation scope.\n *\n * @param context (optional) additional properties to be applied to the returned session object\n *\n * @returns the new active session\n */\nfunction startSession(context) {\n const client = getClient();\n const isolationScope = getIsolationScope();\n const currentScope = getCurrentScope();\n\n const { release, environment = DEFAULT_ENVIRONMENT } = (client && client.getOptions()) || {};\n\n // Will fetch userAgent if called from browser sdk\n const { userAgent } = GLOBAL_OBJ.navigator || {};\n\n const session = makeSession({\n release,\n environment,\n user: currentScope.getUser() || isolationScope.getUser(),\n ...(userAgent && { userAgent }),\n ...context,\n });\n\n // End existing session if there's one\n const currentSession = isolationScope.getSession();\n if (currentSession && currentSession.status === 'ok') {\n updateSession(currentSession, { status: 'exited' });\n }\n\n endSession();\n\n // Afterwards we set the new session on the scope\n isolationScope.setSession(session);\n\n // TODO (v8): Remove this and only use the isolation scope(?).\n // For v7 though, we can't \"soft-break\" people using getCurrentHub().getScope().setSession()\n currentScope.setSession(session);\n\n return session;\n}\n\n/**\n * End the session on the current isolation scope.\n */\nfunction endSession() {\n const isolationScope = getIsolationScope();\n const currentScope = getCurrentScope();\n\n const session = currentScope.getSession() || isolationScope.getSession();\n if (session) {\n closeSession(session);\n }\n _sendSessionUpdate();\n\n // the session is over; take it off of the scope\n isolationScope.setSession();\n\n // TODO (v8): Remove this and only use the isolation scope(?).\n // For v7 though, we can't \"soft-break\" people using getCurrentHub().getScope().setSession()\n currentScope.setSession();\n}\n\n/**\n * Sends the current Session on the scope\n */\nfunction _sendSessionUpdate() {\n const isolationScope = getIsolationScope();\n const currentScope = getCurrentScope();\n const client = getClient();\n // TODO (v8): Remove currentScope and only use the isolation scope(?).\n // For v7 though, we can't \"soft-break\" people using getCurrentHub().getScope().setSession()\n const session = currentScope.getSession() || isolationScope.getSession();\n if (session && client && client.captureSession) {\n client.captureSession(session);\n }\n}\n\n/**\n * Sends the current session on the scope to Sentry\n *\n * @param end If set the session will be marked as exited and removed from the scope.\n * Defaults to `false`.\n */\nfunction captureSession(end = false) {\n // both send the update and pull the session from the scope\n if (end) {\n endSession();\n return;\n }\n\n // only send the update\n _sendSessionUpdate();\n}\n\nexport { addBreadcrumb, captureCheckIn, captureEvent, captureException, captureMessage, captureSession, close, configureScope, endSession, flush, getClient, getCurrentScope, isInitialized, lastEventId, setContext, setExtra, setExtras, setTag, setTags, setUser, startSession, startTransaction, withActiveSpan, withIsolationScope, withMonitor, withScope };\n","import { arrayify, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { addGlobalEventProcessor } from './eventProcessors.js';\nimport { getClient } from './exports.js';\nimport { getCurrentHub } from './hub.js';\n\nconst installedIntegrations = [];\n\n/** Map of integrations assigned to a client */\n\n/**\n * Remove duplicates from the given array, preferring the last instance of any duplicate. Not guaranteed to\n * preseve the order of integrations in the array.\n *\n * @private\n */\nfunction filterDuplicates(integrations) {\n const integrationsByName = {};\n\n integrations.forEach(currentInstance => {\n const { name } = currentInstance;\n\n const existingInstance = integrationsByName[name];\n\n // We want integrations later in the array to overwrite earlier ones of the same type, except that we never want a\n // default instance to overwrite an existing user instance\n if (existingInstance && !existingInstance.isDefaultInstance && currentInstance.isDefaultInstance) {\n return;\n }\n\n integrationsByName[name] = currentInstance;\n });\n\n return Object.keys(integrationsByName).map(k => integrationsByName[k]);\n}\n\n/** Gets integrations to install */\nfunction getIntegrationsToSetup(options) {\n const defaultIntegrations = options.defaultIntegrations || [];\n const userIntegrations = options.integrations;\n\n // We flag default instances, so that later we can tell them apart from any user-created instances of the same class\n defaultIntegrations.forEach(integration => {\n integration.isDefaultInstance = true;\n });\n\n let integrations;\n\n if (Array.isArray(userIntegrations)) {\n integrations = [...defaultIntegrations, ...userIntegrations];\n } else if (typeof userIntegrations === 'function') {\n integrations = arrayify(userIntegrations(defaultIntegrations));\n } else {\n integrations = defaultIntegrations;\n }\n\n const finalIntegrations = filterDuplicates(integrations);\n\n // The `Debug` integration prints copies of the `event` and `hint` which will be passed to `beforeSend` or\n // `beforeSendTransaction`. It therefore has to run after all other integrations, so that the changes of all event\n // processors will be reflected in the printed values. For lack of a more elegant way to guarantee that, we therefore\n // locate it and, assuming it exists, pop it out of its current spot and shove it onto the end of the array.\n const debugIndex = findIndex(finalIntegrations, integration => integration.name === 'Debug');\n if (debugIndex !== -1) {\n const [debugInstance] = finalIntegrations.splice(debugIndex, 1);\n finalIntegrations.push(debugInstance);\n }\n\n return finalIntegrations;\n}\n\n/**\n * Given a list of integration instances this installs them all. When `withDefaults` is set to `true` then all default\n * integrations are added unless they were already provided before.\n * @param integrations array of integration instances\n * @param withDefault should enable default integrations\n */\nfunction setupIntegrations(client, integrations) {\n const integrationIndex = {};\n\n integrations.forEach(integration => {\n // guard against empty provided integrations\n if (integration) {\n setupIntegration(client, integration, integrationIndex);\n }\n });\n\n return integrationIndex;\n}\n\n/**\n * Execute the `afterAllSetup` hooks of the given integrations.\n */\nfunction afterSetupIntegrations(client, integrations) {\n for (const integration of integrations) {\n // guard against empty provided integrations\n if (integration && integration.afterAllSetup) {\n integration.afterAllSetup(client);\n }\n }\n}\n\n/** Setup a single integration. */\nfunction setupIntegration(client, integration, integrationIndex) {\n if (integrationIndex[integration.name]) {\n DEBUG_BUILD && logger.log(`Integration skipped because it was already installed: ${integration.name}`);\n return;\n }\n integrationIndex[integration.name] = integration;\n\n // `setupOnce` is only called the first time\n if (installedIntegrations.indexOf(integration.name) === -1) {\n // eslint-disable-next-line deprecation/deprecation\n integration.setupOnce(addGlobalEventProcessor, getCurrentHub);\n installedIntegrations.push(integration.name);\n }\n\n // `setup` is run for each client\n if (integration.setup && typeof integration.setup === 'function') {\n integration.setup(client);\n }\n\n if (client.on && typeof integration.preprocessEvent === 'function') {\n const callback = integration.preprocessEvent.bind(integration) ;\n client.on('preprocessEvent', (event, hint) => callback(event, hint, client));\n }\n\n if (client.addEventProcessor && typeof integration.processEvent === 'function') {\n const callback = integration.processEvent.bind(integration) ;\n\n const processor = Object.assign((event, hint) => callback(event, hint, client), {\n id: integration.name,\n });\n\n client.addEventProcessor(processor);\n }\n\n DEBUG_BUILD && logger.log(`Integration installed: ${integration.name}`);\n}\n\n/** Add an integration to the current hub's client. */\nfunction addIntegration(integration) {\n const client = getClient();\n\n if (!client || !client.addIntegration) {\n DEBUG_BUILD && logger.warn(`Cannot add integration \"${integration.name}\" because no SDK Client is available.`);\n return;\n }\n\n client.addIntegration(integration);\n}\n\n// Polyfill for Array.findIndex(), which is not supported in ES5\nfunction findIndex(arr, callback) {\n for (let i = 0; i < arr.length; i++) {\n if (callback(arr[i]) === true) {\n return i;\n }\n }\n\n return -1;\n}\n\n/**\n * Convert a new integration function to the legacy class syntax.\n * In v8, we can remove this and instead export the integration functions directly.\n *\n * @deprecated This will be removed in v8!\n */\nfunction convertIntegrationFnToClass(\n name,\n fn,\n) {\n return Object.assign(\n function ConvertedIntegration(...args) {\n return fn(...args);\n },\n { id: name },\n ) ;\n}\n\n/**\n * Define an integration function that can be used to create an integration instance.\n * Note that this by design hides the implementation details of the integration, as they are considered internal.\n */\nfunction defineIntegration(fn) {\n return fn;\n}\n\nexport { addIntegration, afterSetupIntegrations, convertIntegrationFnToClass, defineIntegration, getIntegrationsToSetup, installedIntegrations, setupIntegration, setupIntegrations };\n","import { logger, getEventDescription, stringMatchesSomePattern } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { convertIntegrationFnToClass, defineIntegration } from '../integration.js';\n\n// \"Script error.\" is hard coded into browsers for errors that it can't read.\n// this is the result of a script being pulled in from an external domain and CORS.\nconst DEFAULT_IGNORE_ERRORS = [\n /^Script error\\.?$/,\n /^Javascript error: Script error\\.? on line 0$/,\n /^ResizeObserver loop completed with undelivered notifications.$/,\n /^Cannot redefine property: googletag$/,\n];\n\nconst DEFAULT_IGNORE_TRANSACTIONS = [\n /^.*\\/healthcheck$/,\n /^.*\\/healthy$/,\n /^.*\\/live$/,\n /^.*\\/ready$/,\n /^.*\\/heartbeat$/,\n /^.*\\/health$/,\n /^.*\\/healthz$/,\n];\n\n/** Options for the InboundFilters integration */\n\nconst INTEGRATION_NAME = 'InboundFilters';\nconst _inboundFiltersIntegration = ((options = {}) => {\n return {\n name: INTEGRATION_NAME,\n // TODO v8: Remove this\n setupOnce() {}, // eslint-disable-line @typescript-eslint/no-empty-function\n processEvent(event, _hint, client) {\n const clientOptions = client.getOptions();\n const mergedOptions = _mergeOptions(options, clientOptions);\n return _shouldDropEvent(event, mergedOptions) ? null : event;\n },\n };\n}) ;\n\nconst inboundFiltersIntegration = defineIntegration(_inboundFiltersIntegration);\n\n/**\n * Inbound filters configurable by the user.\n * @deprecated Use `inboundFiltersIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst InboundFilters = convertIntegrationFnToClass(\n INTEGRATION_NAME,\n inboundFiltersIntegration,\n)\n\n;\n\nfunction _mergeOptions(\n internalOptions = {},\n clientOptions = {},\n) {\n return {\n allowUrls: [...(internalOptions.allowUrls || []), ...(clientOptions.allowUrls || [])],\n denyUrls: [...(internalOptions.denyUrls || []), ...(clientOptions.denyUrls || [])],\n ignoreErrors: [\n ...(internalOptions.ignoreErrors || []),\n ...(clientOptions.ignoreErrors || []),\n ...(internalOptions.disableErrorDefaults ? [] : DEFAULT_IGNORE_ERRORS),\n ],\n ignoreTransactions: [\n ...(internalOptions.ignoreTransactions || []),\n ...(clientOptions.ignoreTransactions || []),\n ...(internalOptions.disableTransactionDefaults ? [] : DEFAULT_IGNORE_TRANSACTIONS),\n ],\n ignoreInternal: internalOptions.ignoreInternal !== undefined ? internalOptions.ignoreInternal : true,\n };\n}\n\nfunction _shouldDropEvent(event, options) {\n if (options.ignoreInternal && _isSentryError(event)) {\n DEBUG_BUILD &&\n logger.warn(`Event dropped due to being internal Sentry Error.\\nEvent: ${getEventDescription(event)}`);\n return true;\n }\n if (_isIgnoredError(event, options.ignoreErrors)) {\n DEBUG_BUILD &&\n logger.warn(\n `Event dropped due to being matched by \\`ignoreErrors\\` option.\\nEvent: ${getEventDescription(event)}`,\n );\n return true;\n }\n if (_isIgnoredTransaction(event, options.ignoreTransactions)) {\n DEBUG_BUILD &&\n logger.warn(\n `Event dropped due to being matched by \\`ignoreTransactions\\` option.\\nEvent: ${getEventDescription(event)}`,\n );\n return true;\n }\n if (_isDeniedUrl(event, options.denyUrls)) {\n DEBUG_BUILD &&\n logger.warn(\n `Event dropped due to being matched by \\`denyUrls\\` option.\\nEvent: ${getEventDescription(\n event,\n )}.\\nUrl: ${_getEventFilterUrl(event)}`,\n );\n return true;\n }\n if (!_isAllowedUrl(event, options.allowUrls)) {\n DEBUG_BUILD &&\n logger.warn(\n `Event dropped due to not being matched by \\`allowUrls\\` option.\\nEvent: ${getEventDescription(\n event,\n )}.\\nUrl: ${_getEventFilterUrl(event)}`,\n );\n return true;\n }\n return false;\n}\n\nfunction _isIgnoredError(event, ignoreErrors) {\n // If event.type, this is not an error\n if (event.type || !ignoreErrors || !ignoreErrors.length) {\n return false;\n }\n\n return _getPossibleEventMessages(event).some(message => stringMatchesSomePattern(message, ignoreErrors));\n}\n\nfunction _isIgnoredTransaction(event, ignoreTransactions) {\n if (event.type !== 'transaction' || !ignoreTransactions || !ignoreTransactions.length) {\n return false;\n }\n\n const name = event.transaction;\n return name ? stringMatchesSomePattern(name, ignoreTransactions) : false;\n}\n\nfunction _isDeniedUrl(event, denyUrls) {\n // TODO: Use Glob instead?\n if (!denyUrls || !denyUrls.length) {\n return false;\n }\n const url = _getEventFilterUrl(event);\n return !url ? false : stringMatchesSomePattern(url, denyUrls);\n}\n\nfunction _isAllowedUrl(event, allowUrls) {\n // TODO: Use Glob instead?\n if (!allowUrls || !allowUrls.length) {\n return true;\n }\n const url = _getEventFilterUrl(event);\n return !url ? true : stringMatchesSomePattern(url, allowUrls);\n}\n\nfunction _getPossibleEventMessages(event) {\n const possibleMessages = [];\n\n if (event.message) {\n possibleMessages.push(event.message);\n }\n\n let lastException;\n try {\n // @ts-expect-error Try catching to save bundle size\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n lastException = event.exception.values[event.exception.values.length - 1];\n } catch (e) {\n // try catching to save bundle size checking existence of variables\n }\n\n if (lastException) {\n if (lastException.value) {\n possibleMessages.push(lastException.value);\n if (lastException.type) {\n possibleMessages.push(`${lastException.type}: ${lastException.value}`);\n }\n }\n }\n\n if (DEBUG_BUILD && possibleMessages.length === 0) {\n logger.error(`Could not extract message for event ${getEventDescription(event)}`);\n }\n\n return possibleMessages;\n}\n\nfunction _isSentryError(event) {\n try {\n // @ts-expect-error can't be a sentry error if undefined\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return event.exception.values[0].type === 'SentryError';\n } catch (e) {\n // ignore\n }\n return false;\n}\n\nfunction _getLastValidUrl(frames = []) {\n for (let i = frames.length - 1; i >= 0; i--) {\n const frame = frames[i];\n\n if (frame && frame.filename !== '' && frame.filename !== '[native code]') {\n return frame.filename || null;\n }\n }\n\n return null;\n}\n\nfunction _getEventFilterUrl(event) {\n try {\n let frames;\n try {\n // @ts-expect-error we only care about frames if the whole thing here is defined\n frames = event.exception.values[0].stacktrace.frames;\n } catch (e) {\n // ignore\n }\n return frames ? _getLastValidUrl(frames) : null;\n } catch (oO) {\n DEBUG_BUILD && logger.error(`Cannot extract url for event ${getEventDescription(event)}`);\n return null;\n }\n}\n\nexport { InboundFilters, inboundFiltersIntegration };\n","import { getOriginalFunction } from '@sentry/utils';\nimport { getClient } from '../exports.js';\nimport { convertIntegrationFnToClass, defineIntegration } from '../integration.js';\n\nlet originalFunctionToString;\n\nconst INTEGRATION_NAME = 'FunctionToString';\n\nconst SETUP_CLIENTS = new WeakMap();\n\nconst _functionToStringIntegration = (() => {\n return {\n name: INTEGRATION_NAME,\n setupOnce() {\n // eslint-disable-next-line @typescript-eslint/unbound-method\n originalFunctionToString = Function.prototype.toString;\n\n // intrinsics (like Function.prototype) might be immutable in some environments\n // e.g. Node with --frozen-intrinsics, XS (an embedded JavaScript engine) or SES (a JavaScript proposal)\n try {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n Function.prototype.toString = function ( ...args) {\n const originalFunction = getOriginalFunction(this);\n const context =\n SETUP_CLIENTS.has(getClient() ) && originalFunction !== undefined ? originalFunction : this;\n return originalFunctionToString.apply(context, args);\n };\n } catch (e) {\n // ignore errors here, just don't patch this\n }\n },\n setup(client) {\n SETUP_CLIENTS.set(client, true);\n },\n };\n}) ;\n\n/**\n * Patch toString calls to return proper name for wrapped functions.\n *\n * ```js\n * Sentry.init({\n * integrations: [\n * functionToStringIntegration(),\n * ],\n * });\n * ```\n */\nconst functionToStringIntegration = defineIntegration(_functionToStringIntegration);\n\n/**\n * Patch toString calls to return proper name for wrapped functions.\n *\n * @deprecated Use `functionToStringIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst FunctionToString = convertIntegrationFnToClass(\n INTEGRATION_NAME,\n functionToStringIntegration,\n) ;\n\n// eslint-disable-next-line deprecation/deprecation\n\nexport { FunctionToString, functionToStringIntegration };\n","import { DEBUG_BUILD } from './debug-build.js';\nimport { consoleSandbox, logger } from './logger.js';\n\n/** Regular expression used to parse a Dsn. */\nconst DSN_REGEX = /^(?:(\\w+):)\\/\\/(?:(\\w+)(?::(\\w+)?)?@)([\\w.-]+)(?::(\\d+))?\\/(.+)/;\n\nfunction isValidProtocol(protocol) {\n return protocol === 'http' || protocol === 'https';\n}\n\n/**\n * Renders the string representation of this Dsn.\n *\n * By default, this will render the public representation without the password\n * component. To get the deprecated private representation, set `withPassword`\n * to true.\n *\n * @param withPassword When set to true, the password will be included.\n */\nfunction dsnToString(dsn, withPassword = false) {\n const { host, path, pass, port, projectId, protocol, publicKey } = dsn;\n return (\n `${protocol}://${publicKey}${withPassword && pass ? `:${pass}` : ''}` +\n `@${host}${port ? `:${port}` : ''}/${path ? `${path}/` : path}${projectId}`\n );\n}\n\n/**\n * Parses a Dsn from a given string.\n *\n * @param str A Dsn as string\n * @returns Dsn as DsnComponents or undefined if @param str is not a valid DSN string\n */\nfunction dsnFromString(str) {\n const match = DSN_REGEX.exec(str);\n\n if (!match) {\n // This should be logged to the console\n consoleSandbox(() => {\n // eslint-disable-next-line no-console\n console.error(`Invalid Sentry Dsn: ${str}`);\n });\n return undefined;\n }\n\n const [protocol, publicKey, pass = '', host, port = '', lastPath] = match.slice(1);\n let path = '';\n let projectId = lastPath;\n\n const split = projectId.split('/');\n if (split.length > 1) {\n path = split.slice(0, -1).join('/');\n projectId = split.pop() ;\n }\n\n if (projectId) {\n const projectMatch = projectId.match(/^\\d+/);\n if (projectMatch) {\n projectId = projectMatch[0];\n }\n }\n\n return dsnFromComponents({ host, pass, path, projectId, port, protocol: protocol , publicKey });\n}\n\nfunction dsnFromComponents(components) {\n return {\n protocol: components.protocol,\n publicKey: components.publicKey || '',\n pass: components.pass || '',\n host: components.host,\n port: components.port || '',\n path: components.path || '',\n projectId: components.projectId,\n };\n}\n\nfunction validateDsn(dsn) {\n if (!DEBUG_BUILD) {\n return true;\n }\n\n const { port, projectId, protocol } = dsn;\n\n const requiredComponents = ['protocol', 'publicKey', 'host', 'projectId'];\n const hasMissingRequiredComponent = requiredComponents.find(component => {\n if (!dsn[component]) {\n logger.error(`Invalid Sentry Dsn: ${component} missing`);\n return true;\n }\n return false;\n });\n\n if (hasMissingRequiredComponent) {\n return false;\n }\n\n if (!projectId.match(/^\\d+$/)) {\n logger.error(`Invalid Sentry Dsn: Invalid projectId ${projectId}`);\n return false;\n }\n\n if (!isValidProtocol(protocol)) {\n logger.error(`Invalid Sentry Dsn: Invalid protocol ${protocol}`);\n return false;\n }\n\n if (port && isNaN(parseInt(port, 10))) {\n logger.error(`Invalid Sentry Dsn: Invalid port ${port}`);\n return false;\n }\n\n return true;\n}\n\n/**\n * Creates a valid Sentry Dsn object, identifying a Sentry instance and project.\n * @returns a valid DsnComponents object or `undefined` if @param from is an invalid DSN source\n */\nfunction makeDsn(from) {\n const components = typeof from === 'string' ? dsnFromString(from) : dsnFromComponents(from);\n if (!components || !validateDsn(components)) {\n return undefined;\n }\n return components;\n}\n\nexport { dsnFromString, dsnToString, makeDsn };\n","import { makeDsn, dsnToString, urlEncode } from '@sentry/utils';\n\nconst SENTRY_API_VERSION = '7';\n\n/** Returns the prefix to construct Sentry ingestion API endpoints. */\nfunction getBaseApiEndpoint(dsn) {\n const protocol = dsn.protocol ? `${dsn.protocol}:` : '';\n const port = dsn.port ? `:${dsn.port}` : '';\n return `${protocol}//${dsn.host}${port}${dsn.path ? `/${dsn.path}` : ''}/api/`;\n}\n\n/** Returns the ingest API endpoint for target. */\nfunction _getIngestEndpoint(dsn) {\n return `${getBaseApiEndpoint(dsn)}${dsn.projectId}/envelope/`;\n}\n\n/** Returns a URL-encoded string with auth config suitable for a query string. */\nfunction _encodedAuth(dsn, sdkInfo) {\n return urlEncode({\n // We send only the minimum set of required information. See\n // https://github.com/getsentry/sentry-javascript/issues/2572.\n sentry_key: dsn.publicKey,\n sentry_version: SENTRY_API_VERSION,\n ...(sdkInfo && { sentry_client: `${sdkInfo.name}/${sdkInfo.version}` }),\n });\n}\n\n/**\n * Returns the envelope endpoint URL with auth in the query string.\n *\n * Sending auth as part of the query string and not as custom HTTP headers avoids CORS preflight requests.\n */\nfunction getEnvelopeEndpointWithUrlEncodedAuth(\n dsn,\n // TODO (v8): Remove `tunnelOrOptions` in favor of `options`, and use the substitute code below\n // options: ClientOptions = {} as ClientOptions,\n tunnelOrOptions = {} ,\n) {\n // TODO (v8): Use this code instead\n // const { tunnel, _metadata = {} } = options;\n // return tunnel ? tunnel : `${_getIngestEndpoint(dsn)}?${_encodedAuth(dsn, _metadata.sdk)}`;\n\n const tunnel = typeof tunnelOrOptions === 'string' ? tunnelOrOptions : tunnelOrOptions.tunnel;\n const sdkInfo =\n typeof tunnelOrOptions === 'string' || !tunnelOrOptions._metadata ? undefined : tunnelOrOptions._metadata.sdk;\n\n return tunnel ? tunnel : `${_getIngestEndpoint(dsn)}?${_encodedAuth(dsn, sdkInfo)}`;\n}\n\n/** Returns the url to the report dialog endpoint. */\nfunction getReportDialogEndpoint(\n dsnLike,\n dialogOptions\n\n,\n) {\n const dsn = makeDsn(dsnLike);\n if (!dsn) {\n return '';\n }\n\n const endpoint = `${getBaseApiEndpoint(dsn)}embed/error-page/`;\n\n let encodedOptions = `dsn=${dsnToString(dsn)}`;\n for (const key in dialogOptions) {\n if (key === 'dsn') {\n continue;\n }\n\n if (key === 'onClose') {\n continue;\n }\n\n if (key === 'user') {\n const user = dialogOptions.user;\n if (!user) {\n continue;\n }\n if (user.name) {\n encodedOptions += `&name=${encodeURIComponent(user.name)}`;\n }\n if (user.email) {\n encodedOptions += `&email=${encodeURIComponent(user.email)}`;\n }\n } else {\n encodedOptions += `&${encodeURIComponent(key)}=${encodeURIComponent(dialogOptions[key] )}`;\n }\n }\n\n return `${endpoint}?${encodedOptions}`;\n}\n\nexport { getEnvelopeEndpointWithUrlEncodedAuth, getReportDialogEndpoint };\n","import { DEBUG_BUILD } from './debug-build.js';\nimport { logger } from './logger.js';\nimport { getGlobalObject } from './worldwide.js';\n\n// eslint-disable-next-line deprecation/deprecation\nconst WINDOW = getGlobalObject();\n\n/**\n * Tells whether current environment supports ErrorEvent objects\n * {@link supportsErrorEvent}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsErrorEvent() {\n try {\n new ErrorEvent('');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n/**\n * Tells whether current environment supports DOMError objects\n * {@link supportsDOMError}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsDOMError() {\n try {\n // Chrome: VM89:1 Uncaught TypeError: Failed to construct 'DOMError':\n // 1 argument required, but only 0 present.\n // @ts-expect-error It really needs 1 argument, not 0.\n new DOMError('');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n/**\n * Tells whether current environment supports DOMException objects\n * {@link supportsDOMException}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsDOMException() {\n try {\n new DOMException('');\n return true;\n } catch (e) {\n return false;\n }\n}\n\n/**\n * Tells whether current environment supports Fetch API\n * {@link supportsFetch}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsFetch() {\n if (!('fetch' in WINDOW)) {\n return false;\n }\n\n try {\n new Headers();\n new Request('http://www.example.com');\n new Response();\n return true;\n } catch (e) {\n return false;\n }\n}\n/**\n * isNativeFetch checks if the given function is a native implementation of fetch()\n */\n// eslint-disable-next-line @typescript-eslint/ban-types\nfunction isNativeFetch(func) {\n return func && /^function fetch\\(\\)\\s+\\{\\s+\\[native code\\]\\s+\\}$/.test(func.toString());\n}\n\n/**\n * Tells whether current environment supports Fetch API natively\n * {@link supportsNativeFetch}.\n *\n * @returns true if `window.fetch` is natively implemented, false otherwise\n */\nfunction supportsNativeFetch() {\n if (typeof EdgeRuntime === 'string') {\n return true;\n }\n\n if (!supportsFetch()) {\n return false;\n }\n\n // Fast path to avoid DOM I/O\n // eslint-disable-next-line @typescript-eslint/unbound-method\n if (isNativeFetch(WINDOW.fetch)) {\n return true;\n }\n\n // window.fetch is implemented, but is polyfilled or already wrapped (e.g: by a chrome extension)\n // so create a \"pure\" iframe to see if that has native fetch\n let result = false;\n const doc = WINDOW.document;\n // eslint-disable-next-line deprecation/deprecation\n if (doc && typeof (doc.createElement ) === 'function') {\n try {\n const sandbox = doc.createElement('iframe');\n sandbox.hidden = true;\n doc.head.appendChild(sandbox);\n if (sandbox.contentWindow && sandbox.contentWindow.fetch) {\n // eslint-disable-next-line @typescript-eslint/unbound-method\n result = isNativeFetch(sandbox.contentWindow.fetch);\n }\n doc.head.removeChild(sandbox);\n } catch (err) {\n DEBUG_BUILD &&\n logger.warn('Could not create sandbox iframe for pure fetch check, bailing to window.fetch: ', err);\n }\n }\n\n return result;\n}\n\n/**\n * Tells whether current environment supports ReportingObserver API\n * {@link supportsReportingObserver}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsReportingObserver() {\n return 'ReportingObserver' in WINDOW;\n}\n\n/**\n * Tells whether current environment supports Referrer Policy API\n * {@link supportsReferrerPolicy}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsReferrerPolicy() {\n // Despite all stars in the sky saying that Edge supports old draft syntax, aka 'never', 'always', 'origin' and 'default'\n // (see https://caniuse.com/#feat=referrer-policy),\n // it doesn't. And it throws an exception instead of ignoring this parameter...\n // REF: https://github.com/getsentry/raven-js/issues/1233\n\n if (!supportsFetch()) {\n return false;\n }\n\n try {\n new Request('_', {\n referrerPolicy: 'origin' ,\n });\n return true;\n } catch (e) {\n return false;\n }\n}\n\nexport { isNativeFetch, supportsDOMError, supportsDOMException, supportsErrorEvent, supportsFetch, supportsNativeFetch, supportsReferrerPolicy, supportsReportingObserver };\n","import { getGlobalObject } from '../worldwide.js';\n\n// Based on https://github.com/angular/angular.js/pull/13945/files\n\n// eslint-disable-next-line deprecation/deprecation\nconst WINDOW = getGlobalObject();\n\n/**\n * Tells whether current environment supports History API\n * {@link supportsHistory}.\n *\n * @returns Answer to the given question.\n */\nfunction supportsHistory() {\n // NOTE: in Chrome App environment, touching history.pushState, *even inside\n // a try/catch block*, will cause Chrome to output an error to console.error\n // borrowed from: https://github.com/angular/angular.js/pull/13945/files\n /* eslint-disable @typescript-eslint/no-unsafe-member-access */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const chromeVar = (WINDOW ).chrome;\n const isChromePackagedApp = chromeVar && chromeVar.app && chromeVar.app.runtime;\n /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n const hasHistoryApi = 'history' in WINDOW && !!WINDOW.history.pushState && !!WINDOW.history.replaceState;\n\n return !isChromePackagedApp && hasHistoryApi;\n}\n\nexport { supportsHistory };\n","import { DEBUG_BUILD } from '../debug-build.js';\nimport { logger } from '../logger.js';\nimport { getFunctionName } from '../stacktrace.js';\n\n// We keep the handlers globally\nconst handlers = {};\nconst instrumented = {};\n\n/** Add a handler function. */\nfunction addHandler(type, handler) {\n handlers[type] = handlers[type] || [];\n (handlers[type] ).push(handler);\n}\n\n/**\n * Reset all instrumentation handlers.\n * This can be used by tests to ensure we have a clean slate of instrumentation handlers.\n */\nfunction resetInstrumentationHandlers() {\n Object.keys(handlers).forEach(key => {\n handlers[key ] = undefined;\n });\n}\n\n/** Maybe run an instrumentation function, unless it was already called. */\nfunction maybeInstrument(type, instrumentFn) {\n if (!instrumented[type]) {\n instrumentFn();\n instrumented[type] = true;\n }\n}\n\n/** Trigger handlers for a given instrumentation type. */\nfunction triggerHandlers(type, data) {\n const typeHandlers = type && handlers[type];\n if (!typeHandlers) {\n return;\n }\n\n for (const handler of typeHandlers) {\n try {\n handler(data);\n } catch (e) {\n DEBUG_BUILD &&\n logger.error(\n `Error while triggering instrumentation handler.\\nType: ${type}\\nName: ${getFunctionName(handler)}\\nError:`,\n e,\n );\n }\n }\n}\n\nexport { addHandler, maybeInstrument, resetInstrumentationHandlers, triggerHandlers };\n","import { fill } from '../object.js';\nimport '../debug-build.js';\nimport '../logger.js';\nimport { GLOBAL_OBJ } from '../worldwide.js';\nimport { supportsHistory } from '../vendor/supportsHistory.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\nconst WINDOW = GLOBAL_OBJ ;\n\nlet lastHref;\n\n/**\n * Add an instrumentation handler for when a fetch request happens.\n * The handler function is called once when the request starts and once when it ends,\n * which can be identified by checking if it has an `endTimestamp`.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addHistoryInstrumentationHandler(handler) {\n const type = 'history';\n addHandler(type, handler);\n maybeInstrument(type, instrumentHistory);\n}\n\nfunction instrumentHistory() {\n if (!supportsHistory()) {\n return;\n }\n\n const oldOnPopState = WINDOW.onpopstate;\n WINDOW.onpopstate = function ( ...args) {\n const to = WINDOW.location.href;\n // keep track of the current URL state, as we always receive only the updated state\n const from = lastHref;\n lastHref = to;\n const handlerData = { from, to };\n triggerHandlers('history', handlerData);\n if (oldOnPopState) {\n // Apparently this can throw in Firefox when incorrectly implemented plugin is installed.\n // https://github.com/getsentry/sentry-javascript/issues/3344\n // https://github.com/bugsnag/bugsnag-js/issues/469\n try {\n return oldOnPopState.apply(this, args);\n } catch (_oO) {\n // no-empty\n }\n }\n };\n\n function historyReplacementFunction(originalHistoryFunction) {\n return function ( ...args) {\n const url = args.length > 2 ? args[2] : undefined;\n if (url) {\n // coerce to string (this is what pushState does)\n const from = lastHref;\n const to = String(url);\n // keep track of the current URL state, as we always receive only the updated state\n lastHref = to;\n const handlerData = { from, to };\n triggerHandlers('history', handlerData);\n }\n return originalHistoryFunction.apply(this, args);\n };\n }\n\n fill(WINDOW.history, 'pushState', historyReplacementFunction);\n fill(WINDOW.history, 'replaceState', historyReplacementFunction);\n}\n\nexport { addHistoryInstrumentationHandler };\n","import { dsnToString } from './dsn.js';\nimport { normalize } from './normalize.js';\nimport { dropUndefinedKeys } from './object.js';\n\n/**\n * Creates an envelope.\n * Make sure to always explicitly provide the generic to this function\n * so that the envelope types resolve correctly.\n */\nfunction createEnvelope(headers, items = []) {\n return [headers, items] ;\n}\n\n/**\n * Add an item to an envelope.\n * Make sure to always explicitly provide the generic to this function\n * so that the envelope types resolve correctly.\n */\nfunction addItemToEnvelope(envelope, newItem) {\n const [headers, items] = envelope;\n return [headers, [...items, newItem]] ;\n}\n\n/**\n * Convenience function to loop through the items and item types of an envelope.\n * (This function was mostly created because working with envelope types is painful at the moment)\n *\n * If the callback returns true, the rest of the items will be skipped.\n */\nfunction forEachEnvelopeItem(\n envelope,\n callback,\n) {\n const envelopeItems = envelope[1];\n\n for (const envelopeItem of envelopeItems) {\n const envelopeItemType = envelopeItem[0].type;\n const result = callback(envelopeItem, envelopeItemType);\n\n if (result) {\n return true;\n }\n }\n\n return false;\n}\n\n/**\n * Returns true if the envelope contains any of the given envelope item types\n */\nfunction envelopeContainsItemType(envelope, types) {\n return forEachEnvelopeItem(envelope, (_, type) => types.includes(type));\n}\n\n/**\n * Encode a string to UTF8.\n */\nfunction encodeUTF8(input, textEncoder) {\n const utf8 = textEncoder || new TextEncoder();\n return utf8.encode(input);\n}\n\n/**\n * Serializes an envelope.\n */\nfunction serializeEnvelope(envelope, textEncoder) {\n const [envHeaders, items] = envelope;\n\n // Initially we construct our envelope as a string and only convert to binary chunks if we encounter binary data\n let parts = JSON.stringify(envHeaders);\n\n function append(next) {\n if (typeof parts === 'string') {\n parts = typeof next === 'string' ? parts + next : [encodeUTF8(parts, textEncoder), next];\n } else {\n parts.push(typeof next === 'string' ? encodeUTF8(next, textEncoder) : next);\n }\n }\n\n for (const item of items) {\n const [itemHeaders, payload] = item;\n\n append(`\\n${JSON.stringify(itemHeaders)}\\n`);\n\n if (typeof payload === 'string' || payload instanceof Uint8Array) {\n append(payload);\n } else {\n let stringifiedPayload;\n try {\n stringifiedPayload = JSON.stringify(payload);\n } catch (e) {\n // In case, despite all our efforts to keep `payload` circular-dependency-free, `JSON.strinify()` still\n // fails, we try again after normalizing it again with infinite normalization depth. This of course has a\n // performance impact but in this case a performance hit is better than throwing.\n stringifiedPayload = JSON.stringify(normalize(payload));\n }\n append(stringifiedPayload);\n }\n }\n\n return typeof parts === 'string' ? parts : concatBuffers(parts);\n}\n\nfunction concatBuffers(buffers) {\n const totalLength = buffers.reduce((acc, buf) => acc + buf.length, 0);\n\n const merged = new Uint8Array(totalLength);\n let offset = 0;\n for (const buffer of buffers) {\n merged.set(buffer, offset);\n offset += buffer.length;\n }\n\n return merged;\n}\n\n/**\n * Parses an envelope\n */\nfunction parseEnvelope(\n env,\n textEncoder,\n textDecoder,\n) {\n let buffer = typeof env === 'string' ? textEncoder.encode(env) : env;\n\n function readBinary(length) {\n const bin = buffer.subarray(0, length);\n // Replace the buffer with the remaining data excluding trailing newline\n buffer = buffer.subarray(length + 1);\n return bin;\n }\n\n function readJson() {\n let i = buffer.indexOf(0xa);\n // If we couldn't find a newline, we must have found the end of the buffer\n if (i < 0) {\n i = buffer.length;\n }\n\n return JSON.parse(textDecoder.decode(readBinary(i))) ;\n }\n\n const envelopeHeader = readJson();\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const items = [];\n\n while (buffer.length) {\n const itemHeader = readJson();\n const binaryLength = typeof itemHeader.length === 'number' ? itemHeader.length : undefined;\n\n items.push([itemHeader, binaryLength ? readBinary(binaryLength) : readJson()]);\n }\n\n return [envelopeHeader, items];\n}\n\n/**\n * Creates attachment envelope items\n */\nfunction createAttachmentEnvelopeItem(\n attachment,\n textEncoder,\n) {\n const buffer = typeof attachment.data === 'string' ? encodeUTF8(attachment.data, textEncoder) : attachment.data;\n\n return [\n dropUndefinedKeys({\n type: 'attachment',\n length: buffer.length,\n filename: attachment.filename,\n content_type: attachment.contentType,\n attachment_type: attachment.attachmentType,\n }),\n buffer,\n ];\n}\n\nconst ITEM_TYPE_TO_DATA_CATEGORY_MAP = {\n session: 'session',\n sessions: 'session',\n attachment: 'attachment',\n transaction: 'transaction',\n event: 'error',\n client_report: 'internal',\n user_report: 'default',\n profile: 'profile',\n replay_event: 'replay',\n replay_recording: 'replay',\n check_in: 'monitor',\n feedback: 'feedback',\n span: 'span',\n statsd: 'metric_bucket',\n};\n\n/**\n * Maps the type of an envelope item to a data category.\n */\nfunction envelopeItemTypeToDataCategory(type) {\n return ITEM_TYPE_TO_DATA_CATEGORY_MAP[type];\n}\n\n/** Extracts the minimal SDK info from the metadata or an events */\nfunction getSdkMetadataForEnvelopeHeader(metadataOrEvent) {\n if (!metadataOrEvent || !metadataOrEvent.sdk) {\n return;\n }\n const { name, version } = metadataOrEvent.sdk;\n return { name, version };\n}\n\n/**\n * Creates event envelope headers, based on event, sdk info and tunnel\n * Note: This function was extracted from the core package to make it available in Replay\n */\nfunction createEventEnvelopeHeaders(\n event,\n sdkInfo,\n tunnel,\n dsn,\n) {\n const dynamicSamplingContext = event.sdkProcessingMetadata && event.sdkProcessingMetadata.dynamicSamplingContext;\n return {\n event_id: event.event_id ,\n sent_at: new Date().toISOString(),\n ...(sdkInfo && { sdk: sdkInfo }),\n ...(!!tunnel && dsn && { dsn: dsnToString(dsn) }),\n ...(dynamicSamplingContext && {\n trace: dropUndefinedKeys({ ...dynamicSamplingContext }),\n }),\n };\n}\n\nexport { addItemToEnvelope, createAttachmentEnvelopeItem, createEnvelope, createEventEnvelopeHeaders, envelopeContainsItemType, envelopeItemTypeToDataCategory, forEachEnvelopeItem, getSdkMetadataForEnvelopeHeader, parseEnvelope, serializeEnvelope };\n","/** An error emitted by Sentry SDKs and related utilities. */\nclass SentryError extends Error {\n /** Display name of this error instance. */\n\n constructor( message, logLevel = 'warn') {\n super(message);this.message = message;\n this.name = new.target.prototype.constructor.name;\n // This sets the prototype to be `Error`, not `SentryError`. It's unclear why we do this, but commenting this line\n // out causes various (seemingly totally unrelated) playwright tests consistently time out. FYI, this makes\n // instances of `SentryError` fail `obj instanceof SentryError` checks.\n Object.setPrototypeOf(this, new.target.prototype);\n this.logLevel = logLevel;\n }\n}\n\nexport { SentryError };\n","import { makeDsn, logger, checkOrSetAlreadyCaught, isParameterizedString, isPrimitive, resolvedSyncPromise, addItemToEnvelope, createAttachmentEnvelopeItem, SyncPromise, rejectedSyncPromise, SentryError, isThenable, isPlainObject } from '@sentry/utils';\nimport { getEnvelopeEndpointWithUrlEncodedAuth } from './api.js';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { createEventEnvelope, createSessionEnvelope } from './envelope.js';\nimport { getClient } from './exports.js';\nimport { getIsolationScope } from './hub.js';\nimport { setupIntegration, afterSetupIntegrations, setupIntegrations } from './integration.js';\nimport { createMetricEnvelope } from './metrics/envelope.js';\nimport { updateSession } from './session.js';\nimport { getDynamicSamplingContextFromClient } from './tracing/dynamicSamplingContext.js';\nimport { prepareEvent } from './utils/prepareEvent.js';\n\nconst ALREADY_SEEN_ERROR = \"Not capturing exception because it's already been captured.\";\n\n/**\n * Base implementation for all JavaScript SDK clients.\n *\n * Call the constructor with the corresponding options\n * specific to the client subclass. To access these options later, use\n * {@link Client.getOptions}.\n *\n * If a Dsn is specified in the options, it will be parsed and stored. Use\n * {@link Client.getDsn} to retrieve the Dsn at any moment. In case the Dsn is\n * invalid, the constructor will throw a {@link SentryException}. Note that\n * without a valid Dsn, the SDK will not send any events to Sentry.\n *\n * Before sending an event, it is passed through\n * {@link BaseClient._prepareEvent} to add SDK information and scope data\n * (breadcrumbs and context). To add more custom information, override this\n * method and extend the resulting prepared event.\n *\n * To issue automatically created events (e.g. via instrumentation), use\n * {@link Client.captureEvent}. It will prepare the event and pass it through\n * the callback lifecycle. To issue auto-breadcrumbs, use\n * {@link Client.addBreadcrumb}.\n *\n * @example\n * class NodeClient extends BaseClient {\n * public constructor(options: NodeOptions) {\n * super(options);\n * }\n *\n * // ...\n * }\n */\nclass BaseClient {\n /**\n * A reference to a metrics aggregator\n *\n * @experimental Note this is alpha API. It may experience breaking changes in the future.\n */\n\n /** Options passed to the SDK. */\n\n /** The client Dsn, if specified in options. Without this Dsn, the SDK will be disabled. */\n\n /** Array of set up integrations. */\n\n /** Indicates whether this client's integrations have been set up. */\n\n /** Number of calls being processed */\n\n /** Holds flushable */\n\n // eslint-disable-next-line @typescript-eslint/ban-types\n\n /**\n * Initializes this client instance.\n *\n * @param options Options for the client.\n */\n constructor(options) {\n this._options = options;\n this._integrations = {};\n this._integrationsInitialized = false;\n this._numProcessing = 0;\n this._outcomes = {};\n this._hooks = {};\n this._eventProcessors = [];\n\n if (options.dsn) {\n this._dsn = makeDsn(options.dsn);\n } else {\n DEBUG_BUILD && logger.warn('No DSN provided, client will not send events.');\n }\n\n if (this._dsn) {\n const url = getEnvelopeEndpointWithUrlEncodedAuth(this._dsn, options);\n this._transport = options.transport({\n tunnel: this._options.tunnel,\n recordDroppedEvent: this.recordDroppedEvent.bind(this),\n ...options.transportOptions,\n url,\n });\n }\n }\n\n /**\n * @inheritDoc\n */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/explicit-module-boundary-types\n captureException(exception, hint, scope) {\n // ensure we haven't captured this very object before\n if (checkOrSetAlreadyCaught(exception)) {\n DEBUG_BUILD && logger.log(ALREADY_SEEN_ERROR);\n return;\n }\n\n let eventId = hint && hint.event_id;\n\n this._process(\n this.eventFromException(exception, hint)\n .then(event => this._captureEvent(event, hint, scope))\n .then(result => {\n eventId = result;\n }),\n );\n\n return eventId;\n }\n\n /**\n * @inheritDoc\n */\n captureMessage(\n message,\n // eslint-disable-next-line deprecation/deprecation\n level,\n hint,\n scope,\n ) {\n let eventId = hint && hint.event_id;\n\n const eventMessage = isParameterizedString(message) ? message : String(message);\n\n const promisedEvent = isPrimitive(message)\n ? this.eventFromMessage(eventMessage, level, hint)\n : this.eventFromException(message, hint);\n\n this._process(\n promisedEvent\n .then(event => this._captureEvent(event, hint, scope))\n .then(result => {\n eventId = result;\n }),\n );\n\n return eventId;\n }\n\n /**\n * @inheritDoc\n */\n captureEvent(event, hint, scope) {\n // ensure we haven't captured this very object before\n if (hint && hint.originalException && checkOrSetAlreadyCaught(hint.originalException)) {\n DEBUG_BUILD && logger.log(ALREADY_SEEN_ERROR);\n return;\n }\n\n let eventId = hint && hint.event_id;\n\n const sdkProcessingMetadata = event.sdkProcessingMetadata || {};\n const capturedSpanScope = sdkProcessingMetadata.capturedSpanScope;\n\n this._process(\n this._captureEvent(event, hint, capturedSpanScope || scope).then(result => {\n eventId = result;\n }),\n );\n\n return eventId;\n }\n\n /**\n * @inheritDoc\n */\n captureSession(session) {\n if (!(typeof session.release === 'string')) {\n DEBUG_BUILD && logger.warn('Discarded session because of missing or non-string release');\n } else {\n this.sendSession(session);\n // After sending, we set init false to indicate it's not the first occurrence\n updateSession(session, { init: false });\n }\n }\n\n /**\n * @inheritDoc\n */\n getDsn() {\n return this._dsn;\n }\n\n /**\n * @inheritDoc\n */\n getOptions() {\n return this._options;\n }\n\n /**\n * @see SdkMetadata in @sentry/types\n *\n * @return The metadata of the SDK\n */\n getSdkMetadata() {\n return this._options._metadata;\n }\n\n /**\n * @inheritDoc\n */\n getTransport() {\n return this._transport;\n }\n\n /**\n * @inheritDoc\n */\n flush(timeout) {\n const transport = this._transport;\n if (transport) {\n if (this.metricsAggregator) {\n this.metricsAggregator.flush();\n }\n return this._isClientDoneProcessing(timeout).then(clientFinished => {\n return transport.flush(timeout).then(transportFlushed => clientFinished && transportFlushed);\n });\n } else {\n return resolvedSyncPromise(true);\n }\n }\n\n /**\n * @inheritDoc\n */\n close(timeout) {\n return this.flush(timeout).then(result => {\n this.getOptions().enabled = false;\n if (this.metricsAggregator) {\n this.metricsAggregator.close();\n }\n return result;\n });\n }\n\n /** Get all installed event processors. */\n getEventProcessors() {\n return this._eventProcessors;\n }\n\n /** @inheritDoc */\n addEventProcessor(eventProcessor) {\n this._eventProcessors.push(eventProcessor);\n }\n\n /**\n * This is an internal function to setup all integrations that should run on the client.\n * @deprecated Use `client.init()` instead.\n */\n setupIntegrations(forceInitialize) {\n if ((forceInitialize && !this._integrationsInitialized) || (this._isEnabled() && !this._integrationsInitialized)) {\n this._setupIntegrations();\n }\n }\n\n /** @inheritdoc */\n init() {\n if (this._isEnabled()) {\n this._setupIntegrations();\n }\n }\n\n /**\n * Gets an installed integration by its `id`.\n *\n * @returns The installed integration or `undefined` if no integration with that `id` was installed.\n * @deprecated Use `getIntegrationByName()` instead.\n */\n getIntegrationById(integrationId) {\n return this.getIntegrationByName(integrationId);\n }\n\n /**\n * Gets an installed integration by its name.\n *\n * @returns The installed integration or `undefined` if no integration with that `name` was installed.\n */\n getIntegrationByName(integrationName) {\n return this._integrations[integrationName] ;\n }\n\n /**\n * Returns the client's instance of the given integration class, it any.\n * @deprecated Use `getIntegrationByName()` instead.\n */\n getIntegration(integration) {\n try {\n return (this._integrations[integration.id] ) || null;\n } catch (_oO) {\n DEBUG_BUILD && logger.warn(`Cannot retrieve integration ${integration.id} from the current Client`);\n return null;\n }\n }\n\n /**\n * @inheritDoc\n */\n addIntegration(integration) {\n const isAlreadyInstalled = this._integrations[integration.name];\n\n // This hook takes care of only installing if not already installed\n setupIntegration(this, integration, this._integrations);\n // Here we need to check manually to make sure to not run this multiple times\n if (!isAlreadyInstalled) {\n afterSetupIntegrations(this, [integration]);\n }\n }\n\n /**\n * @inheritDoc\n */\n sendEvent(event, hint = {}) {\n this.emit('beforeSendEvent', event, hint);\n\n let env = createEventEnvelope(event, this._dsn, this._options._metadata, this._options.tunnel);\n\n for (const attachment of hint.attachments || []) {\n env = addItemToEnvelope(\n env,\n createAttachmentEnvelopeItem(\n attachment,\n this._options.transportOptions && this._options.transportOptions.textEncoder,\n ),\n );\n }\n\n const promise = this._sendEnvelope(env);\n if (promise) {\n promise.then(sendResponse => this.emit('afterSendEvent', event, sendResponse), null);\n }\n }\n\n /**\n * @inheritDoc\n */\n sendSession(session) {\n const env = createSessionEnvelope(session, this._dsn, this._options._metadata, this._options.tunnel);\n\n // _sendEnvelope should not throw\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n this._sendEnvelope(env);\n }\n\n /**\n * @inheritDoc\n */\n recordDroppedEvent(reason, category, eventOrCount) {\n if (this._options.sendClientReports) {\n // TODO v9: We do not need the `event` passed as third argument anymore, and can possibly remove this overload\n // If event is passed as third argument, we assume this is a count of 1\n const count = typeof eventOrCount === 'number' ? eventOrCount : 1;\n\n // We want to track each category (error, transaction, session, replay_event) separately\n // but still keep the distinction between different type of outcomes.\n // We could use nested maps, but it's much easier to read and type this way.\n // A correct type for map-based implementation if we want to go that route\n // would be `Partial>>>`\n // With typescript 4.1 we could even use template literal types\n const key = `${reason}:${category}`;\n DEBUG_BUILD && logger.log(`Recording outcome: \"${key}\"${count > 1 ? ` (${count} times)` : ''}`);\n this._outcomes[key] = (this._outcomes[key] || 0) + count;\n }\n }\n\n /**\n * @inheritDoc\n */\n captureAggregateMetrics(metricBucketItems) {\n DEBUG_BUILD && logger.log(`Flushing aggregated metrics, number of metrics: ${metricBucketItems.length}`);\n const metricsEnvelope = createMetricEnvelope(\n metricBucketItems,\n this._dsn,\n this._options._metadata,\n this._options.tunnel,\n );\n\n // _sendEnvelope should not throw\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n this._sendEnvelope(metricsEnvelope);\n }\n\n // Keep on() & emit() signatures in sync with types' client.ts interface\n /* eslint-disable @typescript-eslint/unified-signatures */\n\n /** @inheritdoc */\n\n /** @inheritdoc */\n on(hook, callback) {\n if (!this._hooks[hook]) {\n this._hooks[hook] = [];\n }\n\n // @ts-expect-error We assue the types are correct\n this._hooks[hook].push(callback);\n }\n\n /** @inheritdoc */\n\n /** @inheritdoc */\n emit(hook, ...rest) {\n if (this._hooks[hook]) {\n this._hooks[hook].forEach(callback => callback(...rest));\n }\n }\n\n /* eslint-enable @typescript-eslint/unified-signatures */\n\n /** Setup integrations for this client. */\n _setupIntegrations() {\n const { integrations } = this._options;\n this._integrations = setupIntegrations(this, integrations);\n afterSetupIntegrations(this, integrations);\n\n // TODO v8: We don't need this flag anymore\n this._integrationsInitialized = true;\n }\n\n /** Updates existing session based on the provided event */\n _updateSessionFromEvent(session, event) {\n let crashed = false;\n let errored = false;\n const exceptions = event.exception && event.exception.values;\n\n if (exceptions) {\n errored = true;\n\n for (const ex of exceptions) {\n const mechanism = ex.mechanism;\n if (mechanism && mechanism.handled === false) {\n crashed = true;\n break;\n }\n }\n }\n\n // A session is updated and that session update is sent in only one of the two following scenarios:\n // 1. Session with non terminal status and 0 errors + an error occurred -> Will set error count to 1 and send update\n // 2. Session with non terminal status and 1 error + a crash occurred -> Will set status crashed and send update\n const sessionNonTerminal = session.status === 'ok';\n const shouldUpdateAndSend = (sessionNonTerminal && session.errors === 0) || (sessionNonTerminal && crashed);\n\n if (shouldUpdateAndSend) {\n updateSession(session, {\n ...(crashed && { status: 'crashed' }),\n errors: session.errors || Number(errored || crashed),\n });\n this.captureSession(session);\n }\n }\n\n /**\n * Determine if the client is finished processing. Returns a promise because it will wait `timeout` ms before saying\n * \"no\" (resolving to `false`) in order to give the client a chance to potentially finish first.\n *\n * @param timeout The time, in ms, after which to resolve to `false` if the client is still busy. Passing `0` (or not\n * passing anything) will make the promise wait as long as it takes for processing to finish before resolving to\n * `true`.\n * @returns A promise which will resolve to `true` if processing is already done or finishes before the timeout, and\n * `false` otherwise\n */\n _isClientDoneProcessing(timeout) {\n return new SyncPromise(resolve => {\n let ticked = 0;\n const tick = 1;\n\n const interval = setInterval(() => {\n if (this._numProcessing == 0) {\n clearInterval(interval);\n resolve(true);\n } else {\n ticked += tick;\n if (timeout && ticked >= timeout) {\n clearInterval(interval);\n resolve(false);\n }\n }\n }, tick);\n });\n }\n\n /** Determines whether this SDK is enabled and a transport is present. */\n _isEnabled() {\n return this.getOptions().enabled !== false && this._transport !== undefined;\n }\n\n /**\n * Adds common information to events.\n *\n * The information includes release and environment from `options`,\n * breadcrumbs and context (extra, tags and user) from the scope.\n *\n * Information that is already present in the event is never overwritten. For\n * nested objects, such as the context, keys are merged.\n *\n * @param event The original event.\n * @param hint May contain additional information about the original exception.\n * @param scope A scope containing event metadata.\n * @returns A new event with more information.\n */\n _prepareEvent(\n event,\n hint,\n scope,\n isolationScope = getIsolationScope(),\n ) {\n const options = this.getOptions();\n const integrations = Object.keys(this._integrations);\n if (!hint.integrations && integrations.length > 0) {\n hint.integrations = integrations;\n }\n\n this.emit('preprocessEvent', event, hint);\n\n return prepareEvent(options, event, hint, scope, this, isolationScope).then(evt => {\n if (evt === null) {\n return evt;\n }\n\n const propagationContext = {\n ...isolationScope.getPropagationContext(),\n ...(scope ? scope.getPropagationContext() : undefined),\n };\n\n const trace = evt.contexts && evt.contexts.trace;\n if (!trace && propagationContext) {\n const { traceId: trace_id, spanId, parentSpanId, dsc } = propagationContext;\n evt.contexts = {\n trace: {\n trace_id,\n span_id: spanId,\n parent_span_id: parentSpanId,\n },\n ...evt.contexts,\n };\n\n const dynamicSamplingContext = dsc ? dsc : getDynamicSamplingContextFromClient(trace_id, this, scope);\n\n evt.sdkProcessingMetadata = {\n dynamicSamplingContext,\n ...evt.sdkProcessingMetadata,\n };\n }\n return evt;\n });\n }\n\n /**\n * Processes the event and logs an error in case of rejection\n * @param event\n * @param hint\n * @param scope\n */\n _captureEvent(event, hint = {}, scope) {\n return this._processEvent(event, hint, scope).then(\n finalEvent => {\n return finalEvent.event_id;\n },\n reason => {\n if (DEBUG_BUILD) {\n // If something's gone wrong, log the error as a warning. If it's just us having used a `SentryError` for\n // control flow, log just the message (no stack) as a log-level log.\n const sentryError = reason ;\n if (sentryError.logLevel === 'log') {\n logger.log(sentryError.message);\n } else {\n logger.warn(sentryError);\n }\n }\n return undefined;\n },\n );\n }\n\n /**\n * Processes an event (either error or message) and sends it to Sentry.\n *\n * This also adds breadcrumbs and context information to the event. However,\n * platform specific meta data (such as the User's IP address) must be added\n * by the SDK implementor.\n *\n *\n * @param event The event to send to Sentry.\n * @param hint May contain additional information about the original exception.\n * @param scope A scope containing event metadata.\n * @returns A SyncPromise that resolves with the event or rejects in case event was/will not be send.\n */\n _processEvent(event, hint, scope) {\n const options = this.getOptions();\n const { sampleRate } = options;\n\n const isTransaction = isTransactionEvent(event);\n const isError = isErrorEvent(event);\n const eventType = event.type || 'error';\n const beforeSendLabel = `before send for type \\`${eventType}\\``;\n\n // 1.0 === 100% events are sent\n // 0.0 === 0% events are sent\n // Sampling for transaction happens somewhere else\n if (isError && typeof sampleRate === 'number' && Math.random() > sampleRate) {\n this.recordDroppedEvent('sample_rate', 'error', event);\n return rejectedSyncPromise(\n new SentryError(\n `Discarding event because it's not included in the random sample (sampling rate = ${sampleRate})`,\n 'log',\n ),\n );\n }\n\n const dataCategory = eventType === 'replay_event' ? 'replay' : eventType;\n\n const sdkProcessingMetadata = event.sdkProcessingMetadata || {};\n const capturedSpanIsolationScope = sdkProcessingMetadata.capturedSpanIsolationScope;\n\n return this._prepareEvent(event, hint, scope, capturedSpanIsolationScope)\n .then(prepared => {\n if (prepared === null) {\n this.recordDroppedEvent('event_processor', dataCategory, event);\n throw new SentryError('An event processor returned `null`, will not send event.', 'log');\n }\n\n const isInternalException = hint.data && (hint.data ).__sentry__ === true;\n if (isInternalException) {\n return prepared;\n }\n\n const result = processBeforeSend(options, prepared, hint);\n return _validateBeforeSendResult(result, beforeSendLabel);\n })\n .then(processedEvent => {\n if (processedEvent === null) {\n this.recordDroppedEvent('before_send', dataCategory, event);\n if (isTransaction) {\n const spans = event.spans || [];\n // the transaction itself counts as one span, plus all the child spans that are added\n const spanCount = 1 + spans.length;\n this.recordDroppedEvent('before_send', 'span', spanCount);\n }\n throw new SentryError(`${beforeSendLabel} returned \\`null\\`, will not send event.`, 'log');\n }\n\n const session = scope && scope.getSession();\n if (!isTransaction && session) {\n this._updateSessionFromEvent(session, processedEvent);\n }\n\n if (isTransaction) {\n const spanCountBefore =\n (processedEvent.sdkProcessingMetadata && processedEvent.sdkProcessingMetadata.spanCountBeforeProcessing) ||\n 0;\n const spanCountAfter = processedEvent.spans ? processedEvent.spans.length : 0;\n\n const droppedSpanCount = spanCountBefore - spanCountAfter;\n if (droppedSpanCount > 0) {\n this.recordDroppedEvent('before_send', 'span', droppedSpanCount);\n }\n }\n\n // None of the Sentry built event processor will update transaction name,\n // so if the transaction name has been changed by an event processor, we know\n // it has to come from custom event processor added by a user\n const transactionInfo = processedEvent.transaction_info;\n if (isTransaction && transactionInfo && processedEvent.transaction !== event.transaction) {\n const source = 'custom';\n processedEvent.transaction_info = {\n ...transactionInfo,\n source,\n };\n }\n\n this.sendEvent(processedEvent, hint);\n return processedEvent;\n })\n .then(null, reason => {\n if (reason instanceof SentryError) {\n throw reason;\n }\n\n this.captureException(reason, {\n data: {\n __sentry__: true,\n },\n originalException: reason,\n });\n throw new SentryError(\n `Event processing pipeline threw an error, original event will not be sent. Details have been sent as a new event.\\nReason: ${reason}`,\n );\n });\n }\n\n /**\n * Occupies the client with processing and event\n */\n _process(promise) {\n this._numProcessing++;\n void promise.then(\n value => {\n this._numProcessing--;\n return value;\n },\n reason => {\n this._numProcessing--;\n return reason;\n },\n );\n }\n\n /**\n * @inheritdoc\n */\n _sendEnvelope(envelope) {\n this.emit('beforeEnvelope', envelope);\n\n if (this._isEnabled() && this._transport) {\n return this._transport.send(envelope).then(null, reason => {\n DEBUG_BUILD && logger.error('Error while sending event:', reason);\n });\n } else {\n DEBUG_BUILD && logger.error('Transport disabled');\n }\n }\n\n /**\n * Clears outcomes on this client and returns them.\n */\n _clearOutcomes() {\n const outcomes = this._outcomes;\n this._outcomes = {};\n return Object.keys(outcomes).map(key => {\n const [reason, category] = key.split(':') ;\n return {\n reason,\n category,\n quantity: outcomes[key],\n };\n });\n }\n\n /**\n * @inheritDoc\n */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/explicit-module-boundary-types\n\n}\n\n/**\n * Verifies that return value of configured `beforeSend` or `beforeSendTransaction` is of expected type, and returns the value if so.\n */\nfunction _validateBeforeSendResult(\n beforeSendResult,\n beforeSendLabel,\n) {\n const invalidValueError = `${beforeSendLabel} must return \\`null\\` or a valid event.`;\n if (isThenable(beforeSendResult)) {\n return beforeSendResult.then(\n event => {\n if (!isPlainObject(event) && event !== null) {\n throw new SentryError(invalidValueError);\n }\n return event;\n },\n e => {\n throw new SentryError(`${beforeSendLabel} rejected with ${e}`);\n },\n );\n } else if (!isPlainObject(beforeSendResult) && beforeSendResult !== null) {\n throw new SentryError(invalidValueError);\n }\n return beforeSendResult;\n}\n\n/**\n * Process the matching `beforeSendXXX` callback.\n */\nfunction processBeforeSend(\n options,\n event,\n hint,\n) {\n const { beforeSend, beforeSendTransaction } = options;\n\n if (isErrorEvent(event) && beforeSend) {\n return beforeSend(event, hint);\n }\n\n if (isTransactionEvent(event) && beforeSendTransaction) {\n if (event.spans) {\n // We store the # of spans before processing in SDK metadata,\n // so we can compare it afterwards to determine how many spans were dropped\n const spanCountBefore = event.spans.length;\n event.sdkProcessingMetadata = {\n ...event.sdkProcessingMetadata,\n spanCountBeforeProcessing: spanCountBefore,\n };\n }\n return beforeSendTransaction(event, hint);\n }\n\n return event;\n}\n\nfunction isErrorEvent(event) {\n return event.type === undefined;\n}\n\nfunction isTransactionEvent(event) {\n return event.type === 'transaction';\n}\n\n/**\n * Add an event processor to the current client.\n * This event processor will run for all events processed by this client.\n */\nfunction addEventProcessor(callback) {\n const client = getClient();\n\n if (!client || !client.addEventProcessor) {\n return;\n }\n\n client.addEventProcessor(callback);\n}\n\nexport { BaseClient, addEventProcessor };\n","import { SDK_VERSION } from '../version.js';\n\n/**\n * A builder for the SDK metadata in the options for the SDK initialization.\n *\n * Note: This function is identical to `buildMetadata` in Remix and NextJS and SvelteKit.\n * We don't extract it for bundle size reasons.\n * @see https://github.com/getsentry/sentry-javascript/pull/7404\n * @see https://github.com/getsentry/sentry-javascript/pull/4196\n *\n * If you make changes to this function consider updating the others as well.\n *\n * @param options SDK options object that gets mutated\n * @param names list of package names\n */\nfunction applySdkMetadata(options, name, names = [name], source = 'npm') {\n const metadata = options._metadata || {};\n\n if (!metadata.sdk) {\n metadata.sdk = {\n name: `sentry.javascript.${name}`,\n packages: names.map(name => ({\n name: `${source}:@sentry/${name}`,\n version: SDK_VERSION,\n })),\n version: SDK_VERSION,\n };\n }\n\n options._metadata = metadata;\n}\n\nexport { applySdkMetadata };\n","/**\n * This serves as a build time flag that will be true by default, but false in non-debug builds or if users replace `__SENTRY_DEBUG__` in their generated code.\n *\n * ATTENTION: This constant must never cross package boundaries (i.e. be exported) to guarantee that it can be used for tree shaking.\n */\nconst DEBUG_BUILD = (typeof __SENTRY_DEBUG__ === 'undefined' || __SENTRY_DEBUG__);\n\nexport { DEBUG_BUILD };\n","import { getClient } from '@sentry/core';\nimport { addExceptionMechanism, resolvedSyncPromise, isErrorEvent, isDOMError, isDOMException, addExceptionTypeValue, isError, isPlainObject, isEvent, isParameterizedString, normalizeToSize, extractExceptionKeysForMessage } from '@sentry/utils';\n\n/**\n * This function creates an exception from a JavaScript Error\n */\nfunction exceptionFromError(stackParser, ex) {\n // Get the frames first since Opera can lose the stack if we touch anything else first\n const frames = parseStackFrames(stackParser, ex);\n\n const exception = {\n type: ex && ex.name,\n value: extractMessage(ex),\n };\n\n if (frames.length) {\n exception.stacktrace = { frames };\n }\n\n if (exception.type === undefined && exception.value === '') {\n exception.value = 'Unrecoverable error caught';\n }\n\n return exception;\n}\n\n/**\n * @hidden\n */\nfunction eventFromPlainObject(\n stackParser,\n exception,\n syntheticException,\n isUnhandledRejection,\n) {\n const client = getClient();\n const normalizeDepth = client && client.getOptions().normalizeDepth;\n\n const event = {\n exception: {\n values: [\n {\n type: isEvent(exception) ? exception.constructor.name : isUnhandledRejection ? 'UnhandledRejection' : 'Error',\n value: getNonErrorObjectExceptionValue(exception, { isUnhandledRejection }),\n },\n ],\n },\n extra: {\n __serialized__: normalizeToSize(exception, normalizeDepth),\n },\n };\n\n if (syntheticException) {\n const frames = parseStackFrames(stackParser, syntheticException);\n if (frames.length) {\n // event.exception.values[0] has been set above\n (event.exception ).values[0].stacktrace = { frames };\n }\n }\n\n return event;\n}\n\n/**\n * @hidden\n */\nfunction eventFromError(stackParser, ex) {\n return {\n exception: {\n values: [exceptionFromError(stackParser, ex)],\n },\n };\n}\n\n/** Parses stack frames from an error */\nfunction parseStackFrames(\n stackParser,\n ex,\n) {\n // Access and store the stacktrace property before doing ANYTHING\n // else to it because Opera is not very good at providing it\n // reliably in other circumstances.\n const stacktrace = ex.stacktrace || ex.stack || '';\n\n const popSize = getPopSize(ex);\n\n try {\n return stackParser(stacktrace, popSize);\n } catch (e) {\n // no-empty\n }\n\n return [];\n}\n\n// Based on our own mapping pattern - https://github.com/getsentry/sentry/blob/9f08305e09866c8bd6d0c24f5b0aabdd7dd6c59c/src/sentry/lang/javascript/errormapping.py#L83-L108\nconst reactMinifiedRegexp = /Minified React error #\\d+;/i;\n\nfunction getPopSize(ex) {\n if (ex) {\n if (typeof ex.framesToPop === 'number') {\n return ex.framesToPop;\n }\n\n if (reactMinifiedRegexp.test(ex.message)) {\n return 1;\n }\n }\n\n return 0;\n}\n\n/**\n * There are cases where stacktrace.message is an Event object\n * https://github.com/getsentry/sentry-javascript/issues/1949\n * In this specific case we try to extract stacktrace.message.error.message\n */\nfunction extractMessage(ex) {\n const message = ex && ex.message;\n if (!message) {\n return 'No error message';\n }\n if (message.error && typeof message.error.message === 'string') {\n return message.error.message;\n }\n return message;\n}\n\n/**\n * Creates an {@link Event} from all inputs to `captureException` and non-primitive inputs to `captureMessage`.\n * @hidden\n */\nfunction eventFromException(\n stackParser,\n exception,\n hint,\n attachStacktrace,\n) {\n const syntheticException = (hint && hint.syntheticException) || undefined;\n const event = eventFromUnknownInput(stackParser, exception, syntheticException, attachStacktrace);\n addExceptionMechanism(event); // defaults to { type: 'generic', handled: true }\n event.level = 'error';\n if (hint && hint.event_id) {\n event.event_id = hint.event_id;\n }\n return resolvedSyncPromise(event);\n}\n\n/**\n * Builds and Event from a Message\n * @hidden\n */\nfunction eventFromMessage(\n stackParser,\n message,\n // eslint-disable-next-line deprecation/deprecation\n level = 'info',\n hint,\n attachStacktrace,\n) {\n const syntheticException = (hint && hint.syntheticException) || undefined;\n const event = eventFromString(stackParser, message, syntheticException, attachStacktrace);\n event.level = level;\n if (hint && hint.event_id) {\n event.event_id = hint.event_id;\n }\n return resolvedSyncPromise(event);\n}\n\n/**\n * @hidden\n */\nfunction eventFromUnknownInput(\n stackParser,\n exception,\n syntheticException,\n attachStacktrace,\n isUnhandledRejection,\n) {\n let event;\n\n if (isErrorEvent(exception ) && (exception ).error) {\n // If it is an ErrorEvent with `error` property, extract it to get actual Error\n const errorEvent = exception ;\n return eventFromError(stackParser, errorEvent.error );\n }\n\n // If it is a `DOMError` (which is a legacy API, but still supported in some browsers) then we just extract the name\n // and message, as it doesn't provide anything else. According to the spec, all `DOMExceptions` should also be\n // `Error`s, but that's not the case in IE11, so in that case we treat it the same as we do a `DOMError`.\n //\n // https://developer.mozilla.org/en-US/docs/Web/API/DOMError\n // https://developer.mozilla.org/en-US/docs/Web/API/DOMException\n // https://webidl.spec.whatwg.org/#es-DOMException-specialness\n if (isDOMError(exception) || isDOMException(exception )) {\n const domException = exception ;\n\n if ('stack' in (exception )) {\n event = eventFromError(stackParser, exception );\n } else {\n const name = domException.name || (isDOMError(domException) ? 'DOMError' : 'DOMException');\n const message = domException.message ? `${name}: ${domException.message}` : name;\n event = eventFromString(stackParser, message, syntheticException, attachStacktrace);\n addExceptionTypeValue(event, message);\n }\n if ('code' in domException) {\n // eslint-disable-next-line deprecation/deprecation\n event.tags = { ...event.tags, 'DOMException.code': `${domException.code}` };\n }\n\n return event;\n }\n if (isError(exception)) {\n // we have a real Error object, do nothing\n return eventFromError(stackParser, exception);\n }\n if (isPlainObject(exception) || isEvent(exception)) {\n // If it's a plain object or an instance of `Event` (the built-in JS kind, not this SDK's `Event` type), serialize\n // it manually. This will allow us to group events based on top-level keys which is much better than creating a new\n // group on any key/value change.\n const objectException = exception ;\n event = eventFromPlainObject(stackParser, objectException, syntheticException, isUnhandledRejection);\n addExceptionMechanism(event, {\n synthetic: true,\n });\n return event;\n }\n\n // If none of previous checks were valid, then it means that it's not:\n // - an instance of DOMError\n // - an instance of DOMException\n // - an instance of Event\n // - an instance of Error\n // - a valid ErrorEvent (one with an error property)\n // - a plain Object\n //\n // So bail out and capture it as a simple message:\n event = eventFromString(stackParser, exception , syntheticException, attachStacktrace);\n addExceptionTypeValue(event, `${exception}`, undefined);\n addExceptionMechanism(event, {\n synthetic: true,\n });\n\n return event;\n}\n\n/**\n * @hidden\n */\nfunction eventFromString(\n stackParser,\n message,\n syntheticException,\n attachStacktrace,\n) {\n const event = {};\n\n if (attachStacktrace && syntheticException) {\n const frames = parseStackFrames(stackParser, syntheticException);\n if (frames.length) {\n event.exception = {\n values: [{ value: message, stacktrace: { frames } }],\n };\n }\n }\n\n if (isParameterizedString(message)) {\n const { __sentry_template_string__, __sentry_template_values__ } = message;\n\n event.logentry = {\n message: __sentry_template_string__,\n params: __sentry_template_values__,\n };\n return event;\n }\n\n event.message = message;\n return event;\n}\n\nfunction getNonErrorObjectExceptionValue(\n exception,\n { isUnhandledRejection },\n) {\n const keys = extractExceptionKeysForMessage(exception);\n const captureType = isUnhandledRejection ? 'promise rejection' : 'exception';\n\n // Some ErrorEvent instances do not have an `error` property, which is why they are not handled before\n // We still want to try to get a decent message for these cases\n if (isErrorEvent(exception)) {\n return `Event \\`ErrorEvent\\` captured as ${captureType} with message \\`${exception.message}\\``;\n }\n\n if (isEvent(exception)) {\n const className = getObjectClassName(exception);\n return `Event \\`${className}\\` (type=${exception.type}) captured as ${captureType}`;\n }\n\n return `Object captured as ${captureType} with keys: ${keys}`;\n}\n\nfunction getObjectClassName(obj) {\n try {\n const prototype = Object.getPrototypeOf(obj);\n return prototype ? prototype.constructor.name : undefined;\n } catch (e) {\n // ignore errors here\n }\n}\n\nexport { eventFromError, eventFromException, eventFromMessage, eventFromPlainObject, eventFromString, eventFromUnknownInput, exceptionFromError, parseStackFrames };\n","import '@sentry-internal/tracing';\nimport { withScope, captureException } from '@sentry/core';\nimport { GLOBAL_OBJ, getOriginalFunction, markFunctionWrapped, addNonEnumerableProperty, addExceptionTypeValue, addExceptionMechanism } from '@sentry/utils';\n\nconst WINDOW = GLOBAL_OBJ ;\n\nlet ignoreOnError = 0;\n\n/**\n * @hidden\n */\nfunction shouldIgnoreOnError() {\n return ignoreOnError > 0;\n}\n\n/**\n * @hidden\n */\nfunction ignoreNextOnError() {\n // onerror should trigger before setTimeout\n ignoreOnError++;\n setTimeout(() => {\n ignoreOnError--;\n });\n}\n\n/**\n * Instruments the given function and sends an event to Sentry every time the\n * function throws an exception.\n *\n * @param fn A function to wrap. It is generally safe to pass an unbound function, because the returned wrapper always\n * has a correct `this` context.\n * @returns The wrapped function.\n * @hidden\n */\nfunction wrap(\n fn,\n options\n\n = {},\n before,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n) {\n // for future readers what this does is wrap a function and then create\n // a bi-directional wrapping between them.\n //\n // example: wrapped = wrap(original);\n // original.__sentry_wrapped__ -> wrapped\n // wrapped.__sentry_original__ -> original\n\n if (typeof fn !== 'function') {\n return fn;\n }\n\n try {\n // if we're dealing with a function that was previously wrapped, return\n // the original wrapper.\n const wrapper = fn.__sentry_wrapped__;\n if (wrapper) {\n if (typeof wrapper === 'function') {\n return wrapper;\n } else {\n // If we find that the `__sentry_wrapped__` function is not a function at the time of accessing it, it means\n // that something messed with it. In that case we want to return the originally passed function.\n return fn;\n }\n }\n\n // We don't wanna wrap it twice\n if (getOriginalFunction(fn)) {\n return fn;\n }\n } catch (e) {\n // Just accessing custom props in some Selenium environments\n // can cause a \"Permission denied\" exception (see raven-js#495).\n // Bail on wrapping and return the function as-is (defers to window.onerror).\n return fn;\n }\n\n /* eslint-disable prefer-rest-params */\n // It is important that `sentryWrapped` is not an arrow function to preserve the context of `this`\n const sentryWrapped = function () {\n const args = Array.prototype.slice.call(arguments);\n\n try {\n if (before && typeof before === 'function') {\n before.apply(this, arguments);\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-member-access\n const wrappedArguments = args.map((arg) => wrap(arg, options));\n\n // Attempt to invoke user-land function\n // NOTE: If you are a Sentry user, and you are seeing this stack frame, it\n // means the sentry.javascript SDK caught an error invoking your application code. This\n // is expected behavior and NOT indicative of a bug with sentry.javascript.\n return fn.apply(this, wrappedArguments);\n } catch (ex) {\n ignoreNextOnError();\n\n withScope(scope => {\n scope.addEventProcessor(event => {\n if (options.mechanism) {\n addExceptionTypeValue(event, undefined, undefined);\n addExceptionMechanism(event, options.mechanism);\n }\n\n event.extra = {\n ...event.extra,\n arguments: args,\n };\n\n return event;\n });\n\n captureException(ex);\n });\n\n throw ex;\n }\n };\n /* eslint-enable prefer-rest-params */\n\n // Accessing some objects may throw\n // ref: https://github.com/getsentry/sentry-javascript/issues/1168\n try {\n for (const property in fn) {\n if (Object.prototype.hasOwnProperty.call(fn, property)) {\n sentryWrapped[property] = fn[property];\n }\n }\n } catch (_oO) {} // eslint-disable-line no-empty\n\n // Signal that this function has been wrapped/filled already\n // for both debugging and to prevent it to being wrapped/filled twice\n markFunctionWrapped(sentryWrapped, fn);\n\n addNonEnumerableProperty(fn, '__sentry_wrapped__', sentryWrapped);\n\n // Restore original function name (not all browsers allow that)\n try {\n const descriptor = Object.getOwnPropertyDescriptor(sentryWrapped, 'name') ;\n if (descriptor.configurable) {\n Object.defineProperty(sentryWrapped, 'name', {\n get() {\n return fn.name;\n },\n });\n }\n // eslint-disable-next-line no-empty\n } catch (_oO) {}\n\n return sentryWrapped;\n}\n\nexport { WINDOW, ignoreNextOnError, shouldIgnoreOnError, wrap };\n","import { BaseClient, applySdkMetadata } from '@sentry/core';\nimport { getSDKSource, logger, createClientReportEnvelope, dsnToString } from '@sentry/utils';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { eventFromException, eventFromMessage } from './eventbuilder.js';\nimport { WINDOW } from './helpers.js';\nimport { createUserFeedbackEnvelope } from './userfeedback.js';\n\n/**\n * Configuration options for the Sentry Browser SDK.\n * @see @sentry/types Options for more information.\n */\n\n/**\n * The Sentry Browser SDK Client.\n *\n * @see BrowserOptions for documentation on configuration options.\n * @see SentryClient for usage documentation.\n */\nclass BrowserClient extends BaseClient {\n /**\n * Creates a new Browser SDK instance.\n *\n * @param options Configuration options for this SDK.\n */\n constructor(options) {\n const sdkSource = WINDOW.SENTRY_SDK_SOURCE || getSDKSource();\n applySdkMetadata(options, 'browser', ['browser'], sdkSource);\n\n super(options);\n\n if (options.sendClientReports && WINDOW.document) {\n WINDOW.document.addEventListener('visibilitychange', () => {\n if (WINDOW.document.visibilityState === 'hidden') {\n this._flushOutcomes();\n }\n });\n }\n }\n\n /**\n * @inheritDoc\n */\n eventFromException(exception, hint) {\n return eventFromException(this._options.stackParser, exception, hint, this._options.attachStacktrace);\n }\n\n /**\n * @inheritDoc\n */\n eventFromMessage(\n message,\n // eslint-disable-next-line deprecation/deprecation\n level = 'info',\n hint,\n ) {\n return eventFromMessage(this._options.stackParser, message, level, hint, this._options.attachStacktrace);\n }\n\n /**\n * Sends user feedback to Sentry.\n */\n captureUserFeedback(feedback) {\n if (!this._isEnabled()) {\n DEBUG_BUILD && logger.warn('SDK not enabled, will not capture user feedback.');\n return;\n }\n\n const envelope = createUserFeedbackEnvelope(feedback, {\n metadata: this.getSdkMetadata(),\n dsn: this.getDsn(),\n tunnel: this.getOptions().tunnel,\n });\n\n // _sendEnvelope should not throw\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n this._sendEnvelope(envelope);\n }\n\n /**\n * @inheritDoc\n */\n _prepareEvent(event, hint, scope) {\n event.platform = event.platform || 'javascript';\n return super._prepareEvent(event, hint, scope);\n }\n\n /**\n * Sends client reports as an envelope.\n */\n _flushOutcomes() {\n const outcomes = this._clearOutcomes();\n\n if (outcomes.length === 0) {\n DEBUG_BUILD && logger.log('No outcomes to send');\n return;\n }\n\n // This is really the only place where we want to check for a DSN and only send outcomes then\n if (!this._dsn) {\n DEBUG_BUILD && logger.log('No dsn provided, will not send outcomes');\n return;\n }\n\n DEBUG_BUILD && logger.log('Sending outcomes:', outcomes);\n\n const envelope = createClientReportEnvelope(outcomes, this._options.tunnel && dsnToString(this._dsn));\n\n // _sendEnvelope should not throw\n // eslint-disable-next-line @typescript-eslint/no-floating-promises\n this._sendEnvelope(envelope);\n }\n}\n\nexport { BrowserClient };\n","import { getSdkMetadataForEnvelopeHeader, dsnToString, createEnvelope, createEventEnvelopeHeaders } from '@sentry/utils';\n\n/**\n * Apply SdkInfo (name, version, packages, integrations) to the corresponding event key.\n * Merge with existing data if any.\n **/\nfunction enhanceEventWithSdkInfo(event, sdkInfo) {\n if (!sdkInfo) {\n return event;\n }\n event.sdk = event.sdk || {};\n event.sdk.name = event.sdk.name || sdkInfo.name;\n event.sdk.version = event.sdk.version || sdkInfo.version;\n event.sdk.integrations = [...(event.sdk.integrations || []), ...(sdkInfo.integrations || [])];\n event.sdk.packages = [...(event.sdk.packages || []), ...(sdkInfo.packages || [])];\n return event;\n}\n\n/** Creates an envelope from a Session */\nfunction createSessionEnvelope(\n session,\n dsn,\n metadata,\n tunnel,\n) {\n const sdkInfo = getSdkMetadataForEnvelopeHeader(metadata);\n const envelopeHeaders = {\n sent_at: new Date().toISOString(),\n ...(sdkInfo && { sdk: sdkInfo }),\n ...(!!tunnel && dsn && { dsn: dsnToString(dsn) }),\n };\n\n const envelopeItem =\n 'aggregates' in session ? [{ type: 'sessions' }, session] : [{ type: 'session' }, session.toJSON()];\n\n return createEnvelope(envelopeHeaders, [envelopeItem]);\n}\n\n/**\n * Create an Envelope from an event.\n */\nfunction createEventEnvelope(\n event,\n dsn,\n metadata,\n tunnel,\n) {\n const sdkInfo = getSdkMetadataForEnvelopeHeader(metadata);\n\n /*\n Note: Due to TS, event.type may be `replay_event`, theoretically.\n In practice, we never call `createEventEnvelope` with `replay_event` type,\n and we'd have to adjut a looot of types to make this work properly.\n We want to avoid casting this around, as that could lead to bugs (e.g. when we add another type)\n So the safe choice is to really guard against the replay_event type here.\n */\n const eventType = event.type && event.type !== 'replay_event' ? event.type : 'event';\n\n enhanceEventWithSdkInfo(event, metadata && metadata.sdk);\n\n const envelopeHeaders = createEventEnvelopeHeaders(event, sdkInfo, tunnel, dsn);\n\n // Prevent this data (which, if it exists, was used in earlier steps in the processing pipeline) from being sent to\n // sentry. (Note: Our use of this property comes and goes with whatever we might be debugging, whatever hacks we may\n // have temporarily added, etc. Even if we don't happen to be using it at some point in the future, let's not get rid\n // of this `delete`, lest we miss putting it back in the next time the property is in use.)\n delete event.sdkProcessingMetadata;\n\n const eventItem = [{ type: eventType }, event];\n return createEnvelope(envelopeHeaders, [eventItem]);\n}\n\nexport { createEventEnvelope, createSessionEnvelope };\n","import { dsnToString, createEnvelope } from '@sentry/utils';\nimport { serializeMetricBuckets } from './utils.js';\n\n/**\n * Create envelope from a metric aggregate.\n */\nfunction createMetricEnvelope(\n metricBucketItems,\n dsn,\n metadata,\n tunnel,\n) {\n const headers = {\n sent_at: new Date().toISOString(),\n };\n\n if (metadata && metadata.sdk) {\n headers.sdk = {\n name: metadata.sdk.name,\n version: metadata.sdk.version,\n };\n }\n\n if (!!tunnel && dsn) {\n headers.dsn = dsnToString(dsn);\n }\n\n const item = createMetricEnvelopeItem(metricBucketItems);\n return createEnvelope(headers, [item]);\n}\n\nfunction createMetricEnvelopeItem(metricBucketItems) {\n const payload = serializeMetricBuckets(metricBucketItems);\n const metricHeaders = {\n type: 'statsd',\n length: payload.length,\n };\n return [metricHeaders, payload];\n}\n\nexport { createMetricEnvelope };\n","import { dropUndefinedKeys } from '@sentry/utils';\n\n/**\n * Generate bucket key from metric properties.\n */\nfunction getBucketKey(\n metricType,\n name,\n unit,\n tags,\n) {\n const stringifiedTags = Object.entries(dropUndefinedKeys(tags)).sort((a, b) => a[0].localeCompare(b[0]));\n return `${metricType}${name}${unit}${stringifiedTags}`;\n}\n\n/* eslint-disable no-bitwise */\n/**\n * Simple hash function for strings.\n */\nfunction simpleHash(s) {\n let rv = 0;\n for (let i = 0; i < s.length; i++) {\n const c = s.charCodeAt(i);\n rv = (rv << 5) - rv + c;\n rv &= rv;\n }\n return rv >>> 0;\n}\n/* eslint-enable no-bitwise */\n\n/**\n * Serialize metrics buckets into a string based on statsd format.\n *\n * Example of format:\n * metric.name@second:1:1.2|d|#a:value,b:anothervalue|T12345677\n * Segments:\n * name: metric.name\n * unit: second\n * value: [1, 1.2]\n * type of metric: d (distribution)\n * tags: { a: value, b: anothervalue }\n * timestamp: 12345677\n */\nfunction serializeMetricBuckets(metricBucketItems) {\n let out = '';\n for (const item of metricBucketItems) {\n const tagEntries = Object.entries(item.tags);\n const maybeTags = tagEntries.length > 0 ? `|#${tagEntries.map(([key, value]) => `${key}:${value}`).join(',')}` : '';\n out += `${item.name}@${item.unit}:${item.metric}|${item.metricType}${maybeTags}|T${item.timestamp}\\n`;\n }\n return out;\n}\n\n/** Sanitizes units */\nfunction sanitizeUnit(unit) {\n return unit.replace(/[^\\w]+/gi, '_');\n}\n\n/** Sanitizes metric keys */\nfunction sanitizeMetricKey(key) {\n return key.replace(/[^\\w\\-.]+/gi, '_');\n}\n\nfunction sanitizeTagKey(key) {\n return key.replace(/[^\\w\\-./]+/gi, '');\n}\n\nconst tagValueReplacements = [\n ['\\n', '\\\\n'],\n ['\\r', '\\\\r'],\n ['\\t', '\\\\t'],\n ['\\\\', '\\\\\\\\'],\n ['|', '\\\\u{7c}'],\n [',', '\\\\u{2c}'],\n];\n\nfunction getCharOrReplacement(input) {\n for (const [search, replacement] of tagValueReplacements) {\n if (input === search) {\n return replacement;\n }\n }\n\n return input;\n}\n\nfunction sanitizeTagValue(value) {\n return [...value].reduce((acc, char) => acc + getCharOrReplacement(char), '');\n}\n\n/**\n * Sanitizes tags.\n */\nfunction sanitizeTags(unsanitizedTags) {\n const tags = {};\n for (const key in unsanitizedTags) {\n if (Object.prototype.hasOwnProperty.call(unsanitizedTags, key)) {\n const sanitizedKey = sanitizeTagKey(key);\n tags[sanitizedKey] = sanitizeTagValue(String(unsanitizedTags[key]));\n }\n }\n return tags;\n}\n\nexport { getBucketKey, sanitizeMetricKey, sanitizeTags, sanitizeUnit, serializeMetricBuckets, simpleHash };\n","/*\n * This module exists for optimizations in the build process through rollup and terser. We define some global\n * constants, which can be overridden during build. By guarding certain pieces of code with functions that return these\n * constants, we can control whether or not they appear in the final bundle. (Any code guarded by a false condition will\n * never run, and will hence be dropped during treeshaking.) The two primary uses for this are stripping out calls to\n * `logger` and preventing node-related code from appearing in browser bundles.\n *\n * Attention:\n * This file should not be used to define constants/flags that are intended to be used for tree-shaking conducted by\n * users. These flags should live in their respective packages, as we identified user tooling (specifically webpack)\n * having issues tree-shaking these constants across package boundaries.\n * An example for this is the __SENTRY_DEBUG__ constant. It is declared in each package individually because we want\n * users to be able to shake away expressions that it guards.\n */\n\n/**\n * Figures out if we're building a browser bundle.\n *\n * @returns true if this is a browser bundle build.\n */\nfunction isBrowserBundle() {\n return typeof __SENTRY_BROWSER_BUNDLE__ !== 'undefined' && !!__SENTRY_BROWSER_BUNDLE__;\n}\n\n/**\n * Get source of SDK.\n */\nfunction getSDKSource() {\n // @ts-expect-error \"npm\" is injected by rollup during build process\n return \"npm\";\n}\n\nexport { getSDKSource, isBrowserBundle };\n","import { dsnToString, createEnvelope } from '@sentry/utils';\n\n/**\n * Creates an envelope from a user feedback.\n */\nfunction createUserFeedbackEnvelope(\n feedback,\n {\n metadata,\n tunnel,\n dsn,\n }\n\n,\n) {\n const headers = {\n event_id: feedback.event_id,\n sent_at: new Date().toISOString(),\n ...(metadata &&\n metadata.sdk && {\n sdk: {\n name: metadata.sdk.name,\n version: metadata.sdk.version,\n },\n }),\n ...(!!tunnel && !!dsn && { dsn: dsnToString(dsn) }),\n };\n const item = createUserFeedbackEnvelopeItem(feedback);\n\n return createEnvelope(headers, [item]);\n}\n\nfunction createUserFeedbackEnvelopeItem(feedback) {\n const feedbackHeaders = {\n type: 'user_report',\n };\n return [feedbackHeaders, feedback];\n}\n\nexport { createUserFeedbackEnvelope };\n","import { createEnvelope } from './envelope.js';\nimport { dateTimestampInSeconds } from './time.js';\n\n/**\n * Creates client report envelope\n * @param discarded_events An array of discard events\n * @param dsn A DSN that can be set on the header. Optional.\n */\nfunction createClientReportEnvelope(\n discarded_events,\n dsn,\n timestamp,\n) {\n const clientReportItem = [\n { type: 'client_report' },\n {\n timestamp: timestamp || dateTimestampInSeconds(),\n discarded_events,\n },\n ];\n return createEnvelope(dsn ? { dsn } : {}, [clientReportItem]);\n}\n\nexport { createClientReportEnvelope };\n","import { CONSOLE_LEVELS, originalConsoleMethods } from '../logger.js';\nimport { fill } from '../object.js';\nimport { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\n/**\n * Add an instrumentation handler for when a console.xxx method is called.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addConsoleInstrumentationHandler(handler) {\n const type = 'console';\n addHandler(type, handler);\n maybeInstrument(type, instrumentConsole);\n}\n\nfunction instrumentConsole() {\n if (!('console' in GLOBAL_OBJ)) {\n return;\n }\n\n CONSOLE_LEVELS.forEach(function (level) {\n if (!(level in GLOBAL_OBJ.console)) {\n return;\n }\n\n fill(GLOBAL_OBJ.console, level, function (originalConsoleMethod) {\n originalConsoleMethods[level] = originalConsoleMethod;\n\n return function (...args) {\n const handlerData = { args, level };\n triggerHandlers('console', handlerData);\n\n const log = originalConsoleMethods[level];\n log && log.apply(GLOBAL_OBJ.console, args);\n };\n });\n });\n}\n\nexport { addConsoleInstrumentationHandler };\n","import { uuid4 } from '../misc.js';\nimport { fill, addNonEnumerableProperty } from '../object.js';\nimport { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\nconst WINDOW = GLOBAL_OBJ ;\nconst DEBOUNCE_DURATION = 1000;\n\nlet debounceTimerID;\nlet lastCapturedEventType;\nlet lastCapturedEventTargetId;\n\n/**\n * Add an instrumentation handler for when a click or a keypress happens.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addClickKeypressInstrumentationHandler(handler) {\n const type = 'dom';\n addHandler(type, handler);\n maybeInstrument(type, instrumentDOM);\n}\n\n/** Exported for tests only. */\nfunction instrumentDOM() {\n if (!WINDOW.document) {\n return;\n }\n\n // Make it so that any click or keypress that is unhandled / bubbled up all the way to the document triggers our dom\n // handlers. (Normally we have only one, which captures a breadcrumb for each click or keypress.) Do this before\n // we instrument `addEventListener` so that we don't end up attaching this handler twice.\n const triggerDOMHandler = triggerHandlers.bind(null, 'dom');\n const globalDOMEventHandler = makeDOMEventHandler(triggerDOMHandler, true);\n WINDOW.document.addEventListener('click', globalDOMEventHandler, false);\n WINDOW.document.addEventListener('keypress', globalDOMEventHandler, false);\n\n // After hooking into click and keypress events bubbled up to `document`, we also hook into user-handled\n // clicks & keypresses, by adding an event listener of our own to any element to which they add a listener. That\n // way, whenever one of their handlers is triggered, ours will be, too. (This is needed because their handler\n // could potentially prevent the event from bubbling up to our global listeners. This way, our handler are still\n // guaranteed to fire at least once.)\n ['EventTarget', 'Node'].forEach((target) => {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n const proto = (WINDOW )[target] && (WINDOW )[target].prototype;\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, no-prototype-builtins\n if (!proto || !proto.hasOwnProperty || !proto.hasOwnProperty('addEventListener')) {\n return;\n }\n\n fill(proto, 'addEventListener', function (originalAddEventListener) {\n return function (\n\n type,\n listener,\n options,\n ) {\n if (type === 'click' || type == 'keypress') {\n try {\n const el = this ;\n const handlers = (el.__sentry_instrumentation_handlers__ = el.__sentry_instrumentation_handlers__ || {});\n const handlerForType = (handlers[type] = handlers[type] || { refCount: 0 });\n\n if (!handlerForType.handler) {\n const handler = makeDOMEventHandler(triggerDOMHandler);\n handlerForType.handler = handler;\n originalAddEventListener.call(this, type, handler, options);\n }\n\n handlerForType.refCount++;\n } catch (e) {\n // Accessing dom properties is always fragile.\n // Also allows us to skip `addEventListenrs` calls with no proper `this` context.\n }\n }\n\n return originalAddEventListener.call(this, type, listener, options);\n };\n });\n\n fill(\n proto,\n 'removeEventListener',\n function (originalRemoveEventListener) {\n return function (\n\n type,\n listener,\n options,\n ) {\n if (type === 'click' || type == 'keypress') {\n try {\n const el = this ;\n const handlers = el.__sentry_instrumentation_handlers__ || {};\n const handlerForType = handlers[type];\n\n if (handlerForType) {\n handlerForType.refCount--;\n // If there are no longer any custom handlers of the current type on this element, we can remove ours, too.\n if (handlerForType.refCount <= 0) {\n originalRemoveEventListener.call(this, type, handlerForType.handler, options);\n handlerForType.handler = undefined;\n delete handlers[type]; // eslint-disable-line @typescript-eslint/no-dynamic-delete\n }\n\n // If there are no longer any custom handlers of any type on this element, cleanup everything.\n if (Object.keys(handlers).length === 0) {\n delete el.__sentry_instrumentation_handlers__;\n }\n }\n } catch (e) {\n // Accessing dom properties is always fragile.\n // Also allows us to skip `addEventListenrs` calls with no proper `this` context.\n }\n }\n\n return originalRemoveEventListener.call(this, type, listener, options);\n };\n },\n );\n });\n}\n\n/**\n * Check whether the event is similar to the last captured one. For example, two click events on the same button.\n */\nfunction isSimilarToLastCapturedEvent(event) {\n // If both events have different type, then user definitely performed two separate actions. e.g. click + keypress.\n if (event.type !== lastCapturedEventType) {\n return false;\n }\n\n try {\n // If both events have the same type, it's still possible that actions were performed on different targets.\n // e.g. 2 clicks on different buttons.\n if (!event.target || (event.target )._sentryId !== lastCapturedEventTargetId) {\n return false;\n }\n } catch (e) {\n // just accessing `target` property can throw an exception in some rare circumstances\n // see: https://github.com/getsentry/sentry-javascript/issues/838\n }\n\n // If both events have the same type _and_ same `target` (an element which triggered an event, _not necessarily_\n // to which an event listener was attached), we treat them as the same action, as we want to capture\n // only one breadcrumb. e.g. multiple clicks on the same button, or typing inside a user input box.\n return true;\n}\n\n/**\n * Decide whether an event should be captured.\n * @param event event to be captured\n */\nfunction shouldSkipDOMEvent(eventType, target) {\n // We are only interested in filtering `keypress` events for now.\n if (eventType !== 'keypress') {\n return false;\n }\n\n if (!target || !target.tagName) {\n return true;\n }\n\n // Only consider keypress events on actual input elements. This will disregard keypresses targeting body\n // e.g.tabbing through elements, hotkeys, etc.\n if (target.tagName === 'INPUT' || target.tagName === 'TEXTAREA' || target.isContentEditable) {\n return false;\n }\n\n return true;\n}\n\n/**\n * Wraps addEventListener to capture UI breadcrumbs\n */\nfunction makeDOMEventHandler(\n handler,\n globalListener = false,\n) {\n return (event) => {\n // It's possible this handler might trigger multiple times for the same\n // event (e.g. event propagation through node ancestors).\n // Ignore if we've already captured that event.\n if (!event || event['_sentryCaptured']) {\n return;\n }\n\n const target = getEventTarget(event);\n\n // We always want to skip _some_ events.\n if (shouldSkipDOMEvent(event.type, target)) {\n return;\n }\n\n // Mark event as \"seen\"\n addNonEnumerableProperty(event, '_sentryCaptured', true);\n\n if (target && !target._sentryId) {\n // Add UUID to event target so we can identify if\n addNonEnumerableProperty(target, '_sentryId', uuid4());\n }\n\n const name = event.type === 'keypress' ? 'input' : event.type;\n\n // If there is no last captured event, it means that we can safely capture the new event and store it for future comparisons.\n // If there is a last captured event, see if the new event is different enough to treat it as a unique one.\n // If that's the case, emit the previous event and store locally the newly-captured DOM event.\n if (!isSimilarToLastCapturedEvent(event)) {\n const handlerData = { event, name, global: globalListener };\n handler(handlerData);\n lastCapturedEventType = event.type;\n lastCapturedEventTargetId = target ? target._sentryId : undefined;\n }\n\n // Start a new debounce timer that will prevent us from capturing multiple events that should be grouped together.\n clearTimeout(debounceTimerID);\n debounceTimerID = WINDOW.setTimeout(() => {\n lastCapturedEventTargetId = undefined;\n lastCapturedEventType = undefined;\n }, DEBOUNCE_DURATION);\n };\n}\n\nfunction getEventTarget(event) {\n try {\n return event.target ;\n } catch (e) {\n // just accessing `target` property can throw an exception in some rare circumstances\n // see: https://github.com/getsentry/sentry-javascript/issues/838\n return null;\n }\n}\n\nexport { addClickKeypressInstrumentationHandler, instrumentDOM };\n","import { isString } from '../is.js';\nimport { fill } from '../object.js';\nimport { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\nconst WINDOW = GLOBAL_OBJ ;\n\nconst SENTRY_XHR_DATA_KEY = '__sentry_xhr_v3__';\n\n/**\n * Add an instrumentation handler for when an XHR request happens.\n * The handler function is called once when the request starts and once when it ends,\n * which can be identified by checking if it has an `endTimestamp`.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addXhrInstrumentationHandler(handler) {\n const type = 'xhr';\n addHandler(type, handler);\n maybeInstrument(type, instrumentXHR);\n}\n\n/** Exported only for tests. */\nfunction instrumentXHR() {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n if (!(WINDOW ).XMLHttpRequest) {\n return;\n }\n\n const xhrproto = XMLHttpRequest.prototype;\n\n fill(xhrproto, 'open', function (originalOpen) {\n return function ( ...args) {\n const startTimestamp = Date.now();\n\n // open() should always be called with two or more arguments\n // But to be on the safe side, we actually validate this and bail out if we don't have a method & url\n const method = isString(args[0]) ? args[0].toUpperCase() : undefined;\n const url = parseUrl(args[1]);\n\n if (!method || !url) {\n return originalOpen.apply(this, args);\n }\n\n this[SENTRY_XHR_DATA_KEY] = {\n method,\n url,\n request_headers: {},\n };\n\n // if Sentry key appears in URL, don't capture it as a request\n if (method === 'POST' && url.match(/sentry_key/)) {\n this.__sentry_own_request__ = true;\n }\n\n const onreadystatechangeHandler = () => {\n // For whatever reason, this is not the same instance here as from the outer method\n const xhrInfo = this[SENTRY_XHR_DATA_KEY];\n\n if (!xhrInfo) {\n return;\n }\n\n if (this.readyState === 4) {\n try {\n // touching statusCode in some platforms throws\n // an exception\n xhrInfo.status_code = this.status;\n } catch (e) {\n /* do nothing */\n }\n\n const handlerData = {\n args: [method, url],\n endTimestamp: Date.now(),\n startTimestamp,\n xhr: this,\n };\n triggerHandlers('xhr', handlerData);\n }\n };\n\n if ('onreadystatechange' in this && typeof this.onreadystatechange === 'function') {\n fill(this, 'onreadystatechange', function (original) {\n return function ( ...readyStateArgs) {\n onreadystatechangeHandler();\n return original.apply(this, readyStateArgs);\n };\n });\n } else {\n this.addEventListener('readystatechange', onreadystatechangeHandler);\n }\n\n // Intercepting `setRequestHeader` to access the request headers of XHR instance.\n // This will only work for user/library defined headers, not for the default/browser-assigned headers.\n // Request cookies are also unavailable for XHR, as `Cookie` header can't be defined by `setRequestHeader`.\n fill(this, 'setRequestHeader', function (original) {\n return function ( ...setRequestHeaderArgs) {\n const [header, value] = setRequestHeaderArgs;\n\n const xhrInfo = this[SENTRY_XHR_DATA_KEY];\n\n if (xhrInfo && isString(header) && isString(value)) {\n xhrInfo.request_headers[header.toLowerCase()] = value;\n }\n\n return original.apply(this, setRequestHeaderArgs);\n };\n });\n\n return originalOpen.apply(this, args);\n };\n });\n\n fill(xhrproto, 'send', function (originalSend) {\n return function ( ...args) {\n const sentryXhrData = this[SENTRY_XHR_DATA_KEY];\n\n if (!sentryXhrData) {\n return originalSend.apply(this, args);\n }\n\n if (args[0] !== undefined) {\n sentryXhrData.body = args[0];\n }\n\n const handlerData = {\n args: [sentryXhrData.method, sentryXhrData.url],\n startTimestamp: Date.now(),\n xhr: this,\n };\n triggerHandlers('xhr', handlerData);\n\n return originalSend.apply(this, args);\n };\n });\n}\n\nfunction parseUrl(url) {\n if (isString(url)) {\n return url;\n }\n\n try {\n // url can be a string or URL\n // but since URL is not available in IE11, we do not check for it,\n // but simply assume it is an URL and return `toString()` from it (which returns the full URL)\n // If that fails, we just return undefined\n return (url ).toString();\n } catch (e2) {} // eslint-disable-line no-empty\n\n return undefined;\n}\n\nexport { SENTRY_XHR_DATA_KEY, addXhrInstrumentationHandler, instrumentXHR };\n","import { fill } from '../object.js';\nimport { supportsNativeFetch } from '../supports.js';\nimport { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\n/**\n * Add an instrumentation handler for when a fetch request happens.\n * The handler function is called once when the request starts and once when it ends,\n * which can be identified by checking if it has an `endTimestamp`.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addFetchInstrumentationHandler(handler) {\n const type = 'fetch';\n addHandler(type, handler);\n maybeInstrument(type, instrumentFetch);\n}\n\nfunction instrumentFetch() {\n if (!supportsNativeFetch()) {\n return;\n }\n\n fill(GLOBAL_OBJ, 'fetch', function (originalFetch) {\n return function (...args) {\n const { method, url } = parseFetchArgs(args);\n\n const handlerData = {\n args,\n fetchData: {\n method,\n url,\n },\n startTimestamp: Date.now(),\n };\n\n triggerHandlers('fetch', {\n ...handlerData,\n });\n\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return originalFetch.apply(GLOBAL_OBJ, args).then(\n (response) => {\n const finishedHandlerData = {\n ...handlerData,\n endTimestamp: Date.now(),\n response,\n };\n\n triggerHandlers('fetch', finishedHandlerData);\n return response;\n },\n (error) => {\n const erroredHandlerData = {\n ...handlerData,\n endTimestamp: Date.now(),\n error,\n };\n\n triggerHandlers('fetch', erroredHandlerData);\n // NOTE: If you are a Sentry user, and you are seeing this stack frame,\n // it means the sentry.javascript SDK caught an error invoking your application code.\n // This is expected behavior and NOT indicative of a bug with sentry.javascript.\n throw error;\n },\n );\n };\n });\n}\n\nfunction hasProp(obj, prop) {\n return !!obj && typeof obj === 'object' && !!(obj )[prop];\n}\n\nfunction getUrlFromResource(resource) {\n if (typeof resource === 'string') {\n return resource;\n }\n\n if (!resource) {\n return '';\n }\n\n if (hasProp(resource, 'url')) {\n return resource.url;\n }\n\n if (resource.toString) {\n return resource.toString();\n }\n\n return '';\n}\n\n/**\n * Parses the fetch arguments to find the used Http method and the url of the request.\n * Exported for tests only.\n */\nfunction parseFetchArgs(fetchArgs) {\n if (fetchArgs.length === 0) {\n return { method: 'GET', url: '' };\n }\n\n if (fetchArgs.length === 2) {\n const [url, options] = fetchArgs ;\n\n return {\n url: getUrlFromResource(url),\n method: hasProp(options, 'method') ? String(options.method).toUpperCase() : 'GET',\n };\n }\n\n const arg = fetchArgs[0];\n return {\n url: getUrlFromResource(arg ),\n method: hasProp(arg, 'method') ? String(arg.method).toUpperCase() : 'GET',\n };\n}\n\nexport { addFetchInstrumentationHandler, parseFetchArgs };\n","// Note: Ideally the `SeverityLevel` type would be derived from `validSeverityLevels`, but that would mean either\n//\n// a) moving `validSeverityLevels` to `@sentry/types`,\n// b) moving the`SeverityLevel` type here, or\n// c) importing `validSeverityLevels` from here into `@sentry/types`.\n//\n// Option A would make `@sentry/types` a runtime dependency of `@sentry/utils` (not good), and options B and C would\n// create a circular dependency between `@sentry/types` and `@sentry/utils` (also not good). So a TODO accompanying the\n// type, reminding anyone who changes it to change this list also, will have to do.\n\nconst validSeverityLevels = ['fatal', 'error', 'warning', 'log', 'info', 'debug'];\n\n/**\n * Converts a string-based level into a member of the deprecated {@link Severity} enum.\n *\n * @deprecated `severityFromString` is deprecated. Please use `severityLevelFromString` instead.\n *\n * @param level String representation of Severity\n * @returns Severity\n */\nfunction severityFromString(level) {\n return severityLevelFromString(level) ;\n}\n\n/**\n * Converts a string-based level into a `SeverityLevel`, normalizing it along the way.\n *\n * @param level String representation of desired `SeverityLevel`.\n * @returns The `SeverityLevel` corresponding to the given string, or 'log' if the string isn't a valid level.\n */\nfunction severityLevelFromString(level) {\n return (level === 'warn' ? 'warning' : validSeverityLevels.includes(level) ? level : 'log') ;\n}\n\nexport { severityFromString, severityLevelFromString, validSeverityLevels };\n","/**\n * Parses string form of URL into an object\n * // borrowed from https://tools.ietf.org/html/rfc3986#appendix-B\n * // intentionally using regex and not href parsing trick because React Native and other\n * // environments where DOM might not be available\n * @returns parsed URL object\n */\nfunction parseUrl(url) {\n if (!url) {\n return {};\n }\n\n const match = url.match(/^(([^:/?#]+):)?(\\/\\/([^/?#]*))?([^?#]*)(\\?([^#]*))?(#(.*))?$/);\n\n if (!match) {\n return {};\n }\n\n // coerce to undefined values to empty string so we don't get 'undefined'\n const query = match[6] || '';\n const fragment = match[8] || '';\n return {\n host: match[4],\n path: match[5],\n protocol: match[2],\n search: query,\n hash: fragment,\n relative: match[5] + query + fragment, // everything minus origin\n };\n}\n\n/**\n * Strip the query string and fragment off of a given URL or path (if present)\n *\n * @param urlPath Full URL or path, including possible query string and/or fragment\n * @returns URL or path without query string or fragment\n */\nfunction stripUrlQueryAndFragment(urlPath) {\n // eslint-disable-next-line no-useless-escape\n return urlPath.split(/[\\?#]/, 1)[0];\n}\n\n/**\n * Returns number of URL segments of a passed string URL.\n */\nfunction getNumberOfUrlSegments(url) {\n // split at '/' or at '\\/' to split regex urls correctly\n return url.split(/\\\\?\\//).filter(s => s.length > 0 && s !== ',').length;\n}\n\n/**\n * Takes a URL object and returns a sanitized string which is safe to use as span description\n * see: https://develop.sentry.dev/sdk/data-handling/#structuring-data\n */\nfunction getSanitizedUrlString(url) {\n const { protocol, host, path } = url;\n\n const filteredHost =\n (host &&\n host\n // Always filter out authority\n .replace(/^.*@/, '[filtered]:[filtered]@')\n // Don't show standard :80 (http) and :443 (https) ports to reduce the noise\n // TODO: Use new URL global if it exists\n .replace(/(:80)$/, '')\n .replace(/(:443)$/, '')) ||\n '';\n\n return `${protocol ? `${protocol}://` : ''}${filteredHost}${path}`;\n}\n\nexport { getNumberOfUrlSegments, getSanitizedUrlString, parseUrl, stripUrlQueryAndFragment };\n","import { defineIntegration, convertIntegrationFnToClass, getClient, addBreadcrumb } from '@sentry/core';\nimport { addConsoleInstrumentationHandler, addClickKeypressInstrumentationHandler, addXhrInstrumentationHandler, addFetchInstrumentationHandler, addHistoryInstrumentationHandler, getEventDescription, logger, htmlTreeAsString, getComponentName, severityLevelFromString, safeJoin, SENTRY_XHR_DATA_KEY, parseUrl } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { WINDOW } from '../helpers.js';\n\n/* eslint-disable max-lines */\n\n/** maxStringLength gets capped to prevent 100 breadcrumbs exceeding 1MB event payload size */\nconst MAX_ALLOWED_STRING_LENGTH = 1024;\n\nconst INTEGRATION_NAME = 'Breadcrumbs';\n\nconst _breadcrumbsIntegration = ((options = {}) => {\n const _options = {\n console: true,\n dom: true,\n fetch: true,\n history: true,\n sentry: true,\n xhr: true,\n ...options,\n };\n\n return {\n name: INTEGRATION_NAME,\n // TODO v8: Remove this\n setupOnce() {}, // eslint-disable-line @typescript-eslint/no-empty-function\n setup(client) {\n if (_options.console) {\n addConsoleInstrumentationHandler(_getConsoleBreadcrumbHandler(client));\n }\n if (_options.dom) {\n addClickKeypressInstrumentationHandler(_getDomBreadcrumbHandler(client, _options.dom));\n }\n if (_options.xhr) {\n addXhrInstrumentationHandler(_getXhrBreadcrumbHandler(client));\n }\n if (_options.fetch) {\n addFetchInstrumentationHandler(_getFetchBreadcrumbHandler(client));\n }\n if (_options.history) {\n addHistoryInstrumentationHandler(_getHistoryBreadcrumbHandler(client));\n }\n if (_options.sentry && client.on) {\n client.on('beforeSendEvent', _getSentryBreadcrumbHandler(client));\n }\n },\n };\n}) ;\n\nconst breadcrumbsIntegration = defineIntegration(_breadcrumbsIntegration);\n\n/**\n * Default Breadcrumbs instrumentations\n *\n * @deprecated Use `breadcrumbsIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst Breadcrumbs = convertIntegrationFnToClass(INTEGRATION_NAME, breadcrumbsIntegration)\n\n;\n\n/**\n * Adds a breadcrumb for Sentry events or transactions if this option is enabled.\n */\nfunction _getSentryBreadcrumbHandler(client) {\n return function addSentryBreadcrumb(event) {\n if (getClient() !== client) {\n return;\n }\n\n addBreadcrumb(\n {\n category: `sentry.${event.type === 'transaction' ? 'transaction' : 'event'}`,\n event_id: event.event_id,\n level: event.level,\n message: getEventDescription(event),\n },\n {\n event,\n },\n );\n };\n}\n\n/**\n * A HOC that creaes a function that creates breadcrumbs from DOM API calls.\n * This is a HOC so that we get access to dom options in the closure.\n */\nfunction _getDomBreadcrumbHandler(\n client,\n dom,\n) {\n return function _innerDomBreadcrumb(handlerData) {\n if (getClient() !== client) {\n return;\n }\n\n let target;\n let componentName;\n let keyAttrs = typeof dom === 'object' ? dom.serializeAttribute : undefined;\n\n let maxStringLength =\n typeof dom === 'object' && typeof dom.maxStringLength === 'number' ? dom.maxStringLength : undefined;\n if (maxStringLength && maxStringLength > MAX_ALLOWED_STRING_LENGTH) {\n DEBUG_BUILD &&\n logger.warn(\n `\\`dom.maxStringLength\\` cannot exceed ${MAX_ALLOWED_STRING_LENGTH}, but a value of ${maxStringLength} was configured. Sentry will use ${MAX_ALLOWED_STRING_LENGTH} instead.`,\n );\n maxStringLength = MAX_ALLOWED_STRING_LENGTH;\n }\n\n if (typeof keyAttrs === 'string') {\n keyAttrs = [keyAttrs];\n }\n\n // Accessing event.target can throw (see getsentry/raven-js#838, #768)\n try {\n const event = handlerData.event ;\n const element = _isEvent(event) ? event.target : event;\n\n target = htmlTreeAsString(element, { keyAttrs, maxStringLength });\n componentName = getComponentName(element);\n } catch (e) {\n target = '';\n }\n\n if (target.length === 0) {\n return;\n }\n\n const breadcrumb = {\n category: `ui.${handlerData.name}`,\n message: target,\n };\n\n if (componentName) {\n breadcrumb.data = { 'ui.component_name': componentName };\n }\n\n addBreadcrumb(breadcrumb, {\n event: handlerData.event,\n name: handlerData.name,\n global: handlerData.global,\n });\n };\n}\n\n/**\n * Creates breadcrumbs from console API calls\n */\nfunction _getConsoleBreadcrumbHandler(client) {\n return function _consoleBreadcrumb(handlerData) {\n if (getClient() !== client) {\n return;\n }\n\n const breadcrumb = {\n category: 'console',\n data: {\n arguments: handlerData.args,\n logger: 'console',\n },\n level: severityLevelFromString(handlerData.level),\n message: safeJoin(handlerData.args, ' '),\n };\n\n if (handlerData.level === 'assert') {\n if (handlerData.args[0] === false) {\n breadcrumb.message = `Assertion failed: ${safeJoin(handlerData.args.slice(1), ' ') || 'console.assert'}`;\n breadcrumb.data.arguments = handlerData.args.slice(1);\n } else {\n // Don't capture a breadcrumb for passed assertions\n return;\n }\n }\n\n addBreadcrumb(breadcrumb, {\n input: handlerData.args,\n level: handlerData.level,\n });\n };\n}\n\n/**\n * Creates breadcrumbs from XHR API calls\n */\nfunction _getXhrBreadcrumbHandler(client) {\n return function _xhrBreadcrumb(handlerData) {\n if (getClient() !== client) {\n return;\n }\n\n const { startTimestamp, endTimestamp } = handlerData;\n\n const sentryXhrData = handlerData.xhr[SENTRY_XHR_DATA_KEY];\n\n // We only capture complete, non-sentry requests\n if (!startTimestamp || !endTimestamp || !sentryXhrData) {\n return;\n }\n\n const { method, url, status_code, body } = sentryXhrData;\n\n const data = {\n method,\n url,\n status_code,\n };\n\n const hint = {\n xhr: handlerData.xhr,\n input: body,\n startTimestamp,\n endTimestamp,\n };\n\n addBreadcrumb(\n {\n category: 'xhr',\n data,\n type: 'http',\n },\n hint,\n );\n };\n}\n\n/**\n * Creates breadcrumbs from fetch API calls\n */\nfunction _getFetchBreadcrumbHandler(client) {\n return function _fetchBreadcrumb(handlerData) {\n if (getClient() !== client) {\n return;\n }\n\n const { startTimestamp, endTimestamp } = handlerData;\n\n // We only capture complete fetch requests\n if (!endTimestamp) {\n return;\n }\n\n if (handlerData.fetchData.url.match(/sentry_key/) && handlerData.fetchData.method === 'POST') {\n // We will not create breadcrumbs for fetch requests that contain `sentry_key` (internal sentry requests)\n return;\n }\n\n if (handlerData.error) {\n const data = handlerData.fetchData;\n const hint = {\n data: handlerData.error,\n input: handlerData.args,\n startTimestamp,\n endTimestamp,\n };\n\n addBreadcrumb(\n {\n category: 'fetch',\n data,\n level: 'error',\n type: 'http',\n },\n hint,\n );\n } else {\n const response = handlerData.response ;\n const data = {\n ...handlerData.fetchData,\n status_code: response && response.status,\n };\n const hint = {\n input: handlerData.args,\n response,\n startTimestamp,\n endTimestamp,\n };\n addBreadcrumb(\n {\n category: 'fetch',\n data,\n type: 'http',\n },\n hint,\n );\n }\n };\n}\n\n/**\n * Creates breadcrumbs from history API calls\n */\nfunction _getHistoryBreadcrumbHandler(client) {\n return function _historyBreadcrumb(handlerData) {\n if (getClient() !== client) {\n return;\n }\n\n let from = handlerData.from;\n let to = handlerData.to;\n const parsedLoc = parseUrl(WINDOW.location.href);\n let parsedFrom = from ? parseUrl(from) : undefined;\n const parsedTo = parseUrl(to);\n\n // Initial pushState doesn't provide `from` information\n if (!parsedFrom || !parsedFrom.path) {\n parsedFrom = parsedLoc;\n }\n\n // Use only the path component of the URL if the URL matches the current\n // document (almost all the time when using pushState)\n if (parsedLoc.protocol === parsedTo.protocol && parsedLoc.host === parsedTo.host) {\n to = parsedTo.relative;\n }\n if (parsedLoc.protocol === parsedFrom.protocol && parsedLoc.host === parsedFrom.host) {\n from = parsedFrom.relative;\n }\n\n addBreadcrumb({\n category: 'navigation',\n data: {\n from,\n to,\n },\n });\n };\n}\n\nfunction _isEvent(event) {\n return !!event && !!(event ).target;\n}\n\nexport { Breadcrumbs, breadcrumbsIntegration };\n","import { defineIntegration, convertIntegrationFnToClass } from '@sentry/core';\nimport { logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\n\nconst INTEGRATION_NAME = 'Dedupe';\n\nconst _dedupeIntegration = (() => {\n let previousEvent;\n\n return {\n name: INTEGRATION_NAME,\n // TODO v8: Remove this\n setupOnce() {}, // eslint-disable-line @typescript-eslint/no-empty-function\n processEvent(currentEvent) {\n // We want to ignore any non-error type events, e.g. transactions or replays\n // These should never be deduped, and also not be compared against as _previousEvent.\n if (currentEvent.type) {\n return currentEvent;\n }\n\n // Juuust in case something goes wrong\n try {\n if (_shouldDropEvent(currentEvent, previousEvent)) {\n DEBUG_BUILD && logger.warn('Event dropped due to being a duplicate of previously captured event.');\n return null;\n }\n } catch (_oO) {} // eslint-disable-line no-empty\n\n return (previousEvent = currentEvent);\n },\n };\n}) ;\n\nconst dedupeIntegration = defineIntegration(_dedupeIntegration);\n\n/**\n * Deduplication filter.\n * @deprecated Use `dedupeIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst Dedupe = convertIntegrationFnToClass(INTEGRATION_NAME, dedupeIntegration)\n\n;\n\nfunction _shouldDropEvent(currentEvent, previousEvent) {\n if (!previousEvent) {\n return false;\n }\n\n if (_isSameMessageEvent(currentEvent, previousEvent)) {\n return true;\n }\n\n if (_isSameExceptionEvent(currentEvent, previousEvent)) {\n return true;\n }\n\n return false;\n}\n\nfunction _isSameMessageEvent(currentEvent, previousEvent) {\n const currentMessage = currentEvent.message;\n const previousMessage = previousEvent.message;\n\n // If neither event has a message property, they were both exceptions, so bail out\n if (!currentMessage && !previousMessage) {\n return false;\n }\n\n // If only one event has a stacktrace, but not the other one, they are not the same\n if ((currentMessage && !previousMessage) || (!currentMessage && previousMessage)) {\n return false;\n }\n\n if (currentMessage !== previousMessage) {\n return false;\n }\n\n if (!_isSameFingerprint(currentEvent, previousEvent)) {\n return false;\n }\n\n if (!_isSameStacktrace(currentEvent, previousEvent)) {\n return false;\n }\n\n return true;\n}\n\nfunction _isSameExceptionEvent(currentEvent, previousEvent) {\n const previousException = _getExceptionFromEvent(previousEvent);\n const currentException = _getExceptionFromEvent(currentEvent);\n\n if (!previousException || !currentException) {\n return false;\n }\n\n if (previousException.type !== currentException.type || previousException.value !== currentException.value) {\n return false;\n }\n\n if (!_isSameFingerprint(currentEvent, previousEvent)) {\n return false;\n }\n\n if (!_isSameStacktrace(currentEvent, previousEvent)) {\n return false;\n }\n\n return true;\n}\n\nfunction _isSameStacktrace(currentEvent, previousEvent) {\n let currentFrames = _getFramesFromEvent(currentEvent);\n let previousFrames = _getFramesFromEvent(previousEvent);\n\n // If neither event has a stacktrace, they are assumed to be the same\n if (!currentFrames && !previousFrames) {\n return true;\n }\n\n // If only one event has a stacktrace, but not the other one, they are not the same\n if ((currentFrames && !previousFrames) || (!currentFrames && previousFrames)) {\n return false;\n }\n\n currentFrames = currentFrames ;\n previousFrames = previousFrames ;\n\n // If number of frames differ, they are not the same\n if (previousFrames.length !== currentFrames.length) {\n return false;\n }\n\n // Otherwise, compare the two\n for (let i = 0; i < previousFrames.length; i++) {\n const frameA = previousFrames[i];\n const frameB = currentFrames[i];\n\n if (\n frameA.filename !== frameB.filename ||\n frameA.lineno !== frameB.lineno ||\n frameA.colno !== frameB.colno ||\n frameA.function !== frameB.function\n ) {\n return false;\n }\n }\n\n return true;\n}\n\nfunction _isSameFingerprint(currentEvent, previousEvent) {\n let currentFingerprint = currentEvent.fingerprint;\n let previousFingerprint = previousEvent.fingerprint;\n\n // If neither event has a fingerprint, they are assumed to be the same\n if (!currentFingerprint && !previousFingerprint) {\n return true;\n }\n\n // If only one event has a fingerprint, but not the other one, they are not the same\n if ((currentFingerprint && !previousFingerprint) || (!currentFingerprint && previousFingerprint)) {\n return false;\n }\n\n currentFingerprint = currentFingerprint ;\n previousFingerprint = previousFingerprint ;\n\n // Otherwise, compare the two\n try {\n return !!(currentFingerprint.join('') === previousFingerprint.join(''));\n } catch (_oO) {\n return false;\n }\n}\n\nfunction _getExceptionFromEvent(event) {\n return event.exception && event.exception.values && event.exception.values[0];\n}\n\nfunction _getFramesFromEvent(event) {\n const exception = event.exception;\n\n if (exception) {\n try {\n // @ts-expect-error Object could be undefined\n return exception.values[0].stacktrace.frames;\n } catch (_oO) {\n return undefined;\n }\n }\n return undefined;\n}\n\nexport { Dedupe, dedupeIntegration };\n","import { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\nlet _oldOnErrorHandler = null;\n\n/**\n * Add an instrumentation handler for when an error is captured by the global error handler.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addGlobalErrorInstrumentationHandler(handler) {\n const type = 'error';\n addHandler(type, handler);\n maybeInstrument(type, instrumentError);\n}\n\nfunction instrumentError() {\n _oldOnErrorHandler = GLOBAL_OBJ.onerror;\n\n GLOBAL_OBJ.onerror = function (\n msg,\n url,\n line,\n column,\n error,\n ) {\n const handlerData = {\n column,\n error,\n line,\n msg,\n url,\n };\n triggerHandlers('error', handlerData);\n\n if (_oldOnErrorHandler && !_oldOnErrorHandler.__SENTRY_LOADER__) {\n // eslint-disable-next-line prefer-rest-params\n return _oldOnErrorHandler.apply(this, arguments);\n }\n\n return false;\n };\n\n GLOBAL_OBJ.onerror.__SENTRY_INSTRUMENTED__ = true;\n}\n\nexport { addGlobalErrorInstrumentationHandler };\n","import { GLOBAL_OBJ } from '../worldwide.js';\nimport { addHandler, maybeInstrument, triggerHandlers } from './_handlers.js';\n\nlet _oldOnUnhandledRejectionHandler = null;\n\n/**\n * Add an instrumentation handler for when an unhandled promise rejection is captured.\n *\n * Use at your own risk, this might break without changelog notice, only used internally.\n * @hidden\n */\nfunction addGlobalUnhandledRejectionInstrumentationHandler(\n handler,\n) {\n const type = 'unhandledrejection';\n addHandler(type, handler);\n maybeInstrument(type, instrumentUnhandledRejection);\n}\n\nfunction instrumentUnhandledRejection() {\n _oldOnUnhandledRejectionHandler = GLOBAL_OBJ.onunhandledrejection;\n\n GLOBAL_OBJ.onunhandledrejection = function (e) {\n const handlerData = e;\n triggerHandlers('unhandledrejection', handlerData);\n\n if (_oldOnUnhandledRejectionHandler && !_oldOnUnhandledRejectionHandler.__SENTRY_LOADER__) {\n // eslint-disable-next-line prefer-rest-params\n return _oldOnUnhandledRejectionHandler.apply(this, arguments);\n }\n\n return true;\n };\n\n GLOBAL_OBJ.onunhandledrejection.__SENTRY_INSTRUMENTED__ = true;\n}\n\nexport { addGlobalUnhandledRejectionInstrumentationHandler };\n","import { defineIntegration, convertIntegrationFnToClass, getClient, captureEvent } from '@sentry/core';\nimport { addGlobalErrorInstrumentationHandler, isString, addGlobalUnhandledRejectionInstrumentationHandler, isPrimitive, isErrorEvent, getLocationHref, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { eventFromUnknownInput } from '../eventbuilder.js';\nimport { shouldIgnoreOnError } from '../helpers.js';\n\n/* eslint-disable @typescript-eslint/no-unsafe-member-access */\n\nconst INTEGRATION_NAME = 'GlobalHandlers';\n\nconst _globalHandlersIntegration = ((options = {}) => {\n const _options = {\n onerror: true,\n onunhandledrejection: true,\n ...options,\n };\n\n return {\n name: INTEGRATION_NAME,\n setupOnce() {\n Error.stackTraceLimit = 50;\n },\n setup(client) {\n if (_options.onerror) {\n _installGlobalOnErrorHandler(client);\n globalHandlerLog('onerror');\n }\n if (_options.onunhandledrejection) {\n _installGlobalOnUnhandledRejectionHandler(client);\n globalHandlerLog('onunhandledrejection');\n }\n },\n };\n}) ;\n\nconst globalHandlersIntegration = defineIntegration(_globalHandlersIntegration);\n\n/**\n * Global handlers.\n * @deprecated Use `globalHandlersIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst GlobalHandlers = convertIntegrationFnToClass(\n INTEGRATION_NAME,\n globalHandlersIntegration,\n)\n\n;\n\nfunction _installGlobalOnErrorHandler(client) {\n addGlobalErrorInstrumentationHandler(data => {\n const { stackParser, attachStacktrace } = getOptions();\n\n if (getClient() !== client || shouldIgnoreOnError()) {\n return;\n }\n\n const { msg, url, line, column, error } = data;\n\n const event =\n error === undefined && isString(msg)\n ? _eventFromIncompleteOnError(msg, url, line, column)\n : _enhanceEventWithInitialFrame(\n eventFromUnknownInput(stackParser, error || msg, undefined, attachStacktrace, false),\n url,\n line,\n column,\n );\n\n event.level = 'error';\n\n captureEvent(event, {\n originalException: error,\n mechanism: {\n handled: false,\n type: 'onerror',\n },\n });\n });\n}\n\nfunction _installGlobalOnUnhandledRejectionHandler(client) {\n addGlobalUnhandledRejectionInstrumentationHandler(e => {\n const { stackParser, attachStacktrace } = getOptions();\n\n if (getClient() !== client || shouldIgnoreOnError()) {\n return;\n }\n\n const error = _getUnhandledRejectionError(e );\n\n const event = isPrimitive(error)\n ? _eventFromRejectionWithPrimitive(error)\n : eventFromUnknownInput(stackParser, error, undefined, attachStacktrace, true);\n\n event.level = 'error';\n\n captureEvent(event, {\n originalException: error,\n mechanism: {\n handled: false,\n type: 'onunhandledrejection',\n },\n });\n });\n}\n\nfunction _getUnhandledRejectionError(error) {\n if (isPrimitive(error)) {\n return error;\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const e = error ;\n\n // dig the object of the rejection out of known event types\n try {\n // PromiseRejectionEvents store the object of the rejection under 'reason'\n // see https://developer.mozilla.org/en-US/docs/Web/API/PromiseRejectionEvent\n if ('reason' in e) {\n return e.reason;\n }\n\n // something, somewhere, (likely a browser extension) effectively casts PromiseRejectionEvents\n // to CustomEvents, moving the `promise` and `reason` attributes of the PRE into\n // the CustomEvent's `detail` attribute, since they're not part of CustomEvent's spec\n // see https://developer.mozilla.org/en-US/docs/Web/API/CustomEvent and\n // https://github.com/getsentry/sentry-javascript/issues/2380\n else if ('detail' in e && 'reason' in e.detail) {\n return e.detail.reason;\n }\n } catch (e2) {} // eslint-disable-line no-empty\n\n return error;\n}\n\n/**\n * Create an event from a promise rejection where the `reason` is a primitive.\n *\n * @param reason: The `reason` property of the promise rejection\n * @returns An Event object with an appropriate `exception` value\n */\nfunction _eventFromRejectionWithPrimitive(reason) {\n return {\n exception: {\n values: [\n {\n type: 'UnhandledRejection',\n // String() is needed because the Primitive type includes symbols (which can't be automatically stringified)\n value: `Non-Error promise rejection captured with value: ${String(reason)}`,\n },\n ],\n },\n };\n}\n\n/**\n * This function creates a stack from an old, error-less onerror handler.\n */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction _eventFromIncompleteOnError(msg, url, line, column) {\n const ERROR_TYPES_RE =\n /^(?:[Uu]ncaught (?:exception: )?)?(?:((?:Eval|Internal|Range|Reference|Syntax|Type|URI|)Error): )?(.*)$/i;\n\n // If 'message' is ErrorEvent, get real message from inside\n let message = isErrorEvent(msg) ? msg.message : msg;\n let name = 'Error';\n\n const groups = message.match(ERROR_TYPES_RE);\n if (groups) {\n name = groups[1];\n message = groups[2];\n }\n\n const event = {\n exception: {\n values: [\n {\n type: name,\n value: message,\n },\n ],\n },\n };\n\n return _enhanceEventWithInitialFrame(event, url, line, column);\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction _enhanceEventWithInitialFrame(event, url, line, column) {\n // event.exception\n const e = (event.exception = event.exception || {});\n // event.exception.values\n const ev = (e.values = e.values || []);\n // event.exception.values[0]\n const ev0 = (ev[0] = ev[0] || {});\n // event.exception.values[0].stacktrace\n const ev0s = (ev0.stacktrace = ev0.stacktrace || {});\n // event.exception.values[0].stacktrace.frames\n const ev0sf = (ev0s.frames = ev0s.frames || []);\n\n const colno = isNaN(parseInt(column, 10)) ? undefined : column;\n const lineno = isNaN(parseInt(line, 10)) ? undefined : line;\n const filename = isString(url) && url.length > 0 ? url : getLocationHref();\n\n // event.exception.values[0].stacktrace.frames\n if (ev0sf.length === 0) {\n ev0sf.push({\n colno,\n filename,\n function: '?',\n in_app: true,\n lineno,\n });\n }\n\n return event;\n}\n\nfunction globalHandlerLog(type) {\n DEBUG_BUILD && logger.log(`Global Handler attached: ${type}`);\n}\n\nfunction getOptions() {\n const client = getClient();\n const options = (client && client.getOptions()) || {\n stackParser: () => [],\n attachStacktrace: false,\n };\n return options;\n}\n\nexport { GlobalHandlers, globalHandlersIntegration };\n","import { defineIntegration, convertIntegrationFnToClass } from '@sentry/core';\nimport { WINDOW } from '../helpers.js';\n\nconst INTEGRATION_NAME = 'HttpContext';\n\nconst _httpContextIntegration = (() => {\n return {\n name: INTEGRATION_NAME,\n // TODO v8: Remove this\n setupOnce() {}, // eslint-disable-line @typescript-eslint/no-empty-function\n preprocessEvent(event) {\n // if none of the information we want exists, don't bother\n if (!WINDOW.navigator && !WINDOW.location && !WINDOW.document) {\n return;\n }\n\n // grab as much info as exists and add it to the event\n const url = (event.request && event.request.url) || (WINDOW.location && WINDOW.location.href);\n const { referrer } = WINDOW.document || {};\n const { userAgent } = WINDOW.navigator || {};\n\n const headers = {\n ...(event.request && event.request.headers),\n ...(referrer && { Referer: referrer }),\n ...(userAgent && { 'User-Agent': userAgent }),\n };\n const request = { ...event.request, ...(url && { url }), headers };\n\n event.request = request;\n },\n };\n}) ;\n\nconst httpContextIntegration = defineIntegration(_httpContextIntegration);\n\n/**\n * HttpContext integration collects information about HTTP request headers.\n * @deprecated Use `httpContextIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst HttpContext = convertIntegrationFnToClass(INTEGRATION_NAME, httpContextIntegration)\n\n;\n\nexport { HttpContext, httpContextIntegration };\n","import { isInstanceOf } from './is.js';\nimport { truncate } from './string.js';\n\n/**\n * Creates exceptions inside `event.exception.values` for errors that are nested on properties based on the `key` parameter.\n */\nfunction applyAggregateErrorsToEvent(\n exceptionFromErrorImplementation,\n parser,\n maxValueLimit = 250,\n key,\n limit,\n event,\n hint,\n) {\n if (!event.exception || !event.exception.values || !hint || !isInstanceOf(hint.originalException, Error)) {\n return;\n }\n\n // Generally speaking the last item in `event.exception.values` is the exception originating from the original Error\n const originalException =\n event.exception.values.length > 0 ? event.exception.values[event.exception.values.length - 1] : undefined;\n\n // We only create exception grouping if there is an exception in the event.\n if (originalException) {\n event.exception.values = truncateAggregateExceptions(\n aggregateExceptionsFromError(\n exceptionFromErrorImplementation,\n parser,\n limit,\n hint.originalException ,\n key,\n event.exception.values,\n originalException,\n 0,\n ),\n maxValueLimit,\n );\n }\n}\n\nfunction aggregateExceptionsFromError(\n exceptionFromErrorImplementation,\n parser,\n limit,\n error,\n key,\n prevExceptions,\n exception,\n exceptionId,\n) {\n if (prevExceptions.length >= limit + 1) {\n return prevExceptions;\n }\n\n let newExceptions = [...prevExceptions];\n\n // Recursively call this function in order to walk down a chain of errors\n if (isInstanceOf(error[key], Error)) {\n applyExceptionGroupFieldsForParentException(exception, exceptionId);\n const newException = exceptionFromErrorImplementation(parser, error[key]);\n const newExceptionId = newExceptions.length;\n applyExceptionGroupFieldsForChildException(newException, key, newExceptionId, exceptionId);\n newExceptions = aggregateExceptionsFromError(\n exceptionFromErrorImplementation,\n parser,\n limit,\n error[key],\n key,\n [newException, ...newExceptions],\n newException,\n newExceptionId,\n );\n }\n\n // This will create exception grouping for AggregateErrors\n // https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/AggregateError\n if (Array.isArray(error.errors)) {\n error.errors.forEach((childError, i) => {\n if (isInstanceOf(childError, Error)) {\n applyExceptionGroupFieldsForParentException(exception, exceptionId);\n const newException = exceptionFromErrorImplementation(parser, childError);\n const newExceptionId = newExceptions.length;\n applyExceptionGroupFieldsForChildException(newException, `errors[${i}]`, newExceptionId, exceptionId);\n newExceptions = aggregateExceptionsFromError(\n exceptionFromErrorImplementation,\n parser,\n limit,\n childError,\n key,\n [newException, ...newExceptions],\n newException,\n newExceptionId,\n );\n }\n });\n }\n\n return newExceptions;\n}\n\nfunction applyExceptionGroupFieldsForParentException(exception, exceptionId) {\n // Don't know if this default makes sense. The protocol requires us to set these values so we pick *some* default.\n exception.mechanism = exception.mechanism || { type: 'generic', handled: true };\n\n exception.mechanism = {\n ...exception.mechanism,\n ...(exception.type === 'AggregateError' && { is_exception_group: true }),\n exception_id: exceptionId,\n };\n}\n\nfunction applyExceptionGroupFieldsForChildException(\n exception,\n source,\n exceptionId,\n parentId,\n) {\n // Don't know if this default makes sense. The protocol requires us to set these values so we pick *some* default.\n exception.mechanism = exception.mechanism || { type: 'generic', handled: true };\n\n exception.mechanism = {\n ...exception.mechanism,\n type: 'chained',\n source,\n exception_id: exceptionId,\n parent_id: parentId,\n };\n}\n\n/**\n * Truncate the message (exception.value) of all exceptions in the event.\n * Because this event processor is ran after `applyClientOptions`,\n * we need to truncate the message of the added exceptions here.\n */\nfunction truncateAggregateExceptions(exceptions, maxValueLength) {\n return exceptions.map(exception => {\n if (exception.value) {\n exception.value = truncate(exception.value, maxValueLength);\n }\n return exception;\n });\n}\n\nexport { applyAggregateErrorsToEvent };\n","import { defineIntegration, convertIntegrationFnToClass } from '@sentry/core';\nimport { applyAggregateErrorsToEvent } from '@sentry/utils';\nimport { exceptionFromError } from '../eventbuilder.js';\n\nconst DEFAULT_KEY = 'cause';\nconst DEFAULT_LIMIT = 5;\n\nconst INTEGRATION_NAME = 'LinkedErrors';\n\nconst _linkedErrorsIntegration = ((options = {}) => {\n const limit = options.limit || DEFAULT_LIMIT;\n const key = options.key || DEFAULT_KEY;\n\n return {\n name: INTEGRATION_NAME,\n // TODO v8: Remove this\n setupOnce() {}, // eslint-disable-line @typescript-eslint/no-empty-function\n preprocessEvent(event, hint, client) {\n const options = client.getOptions();\n\n applyAggregateErrorsToEvent(\n // This differs from the LinkedErrors integration in core by using a different exceptionFromError function\n exceptionFromError,\n options.stackParser,\n options.maxValueLength,\n key,\n limit,\n event,\n hint,\n );\n },\n };\n}) ;\n\nconst linkedErrorsIntegration = defineIntegration(_linkedErrorsIntegration);\n\n/**\n * Aggregrate linked errors in an event.\n * @deprecated Use `linkedErrorsIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst LinkedErrors = convertIntegrationFnToClass(INTEGRATION_NAME, linkedErrorsIntegration)\n\n;\n\nexport { LinkedErrors, linkedErrorsIntegration };\n","import { defineIntegration, convertIntegrationFnToClass } from '@sentry/core';\nimport { fill, getFunctionName, getOriginalFunction } from '@sentry/utils';\nimport { WINDOW, wrap } from '../helpers.js';\n\nconst DEFAULT_EVENT_TARGET = [\n 'EventTarget',\n 'Window',\n 'Node',\n 'ApplicationCache',\n 'AudioTrackList',\n 'BroadcastChannel',\n 'ChannelMergerNode',\n 'CryptoOperation',\n 'EventSource',\n 'FileReader',\n 'HTMLUnknownElement',\n 'IDBDatabase',\n 'IDBRequest',\n 'IDBTransaction',\n 'KeyOperation',\n 'MediaController',\n 'MessagePort',\n 'ModalWindow',\n 'Notification',\n 'SVGElementInstance',\n 'Screen',\n 'SharedWorker',\n 'TextTrack',\n 'TextTrackCue',\n 'TextTrackList',\n 'WebSocket',\n 'WebSocketWorker',\n 'Worker',\n 'XMLHttpRequest',\n 'XMLHttpRequestEventTarget',\n 'XMLHttpRequestUpload',\n];\n\nconst INTEGRATION_NAME = 'TryCatch';\n\nconst _browserApiErrorsIntegration = ((options = {}) => {\n const _options = {\n XMLHttpRequest: true,\n eventTarget: true,\n requestAnimationFrame: true,\n setInterval: true,\n setTimeout: true,\n ...options,\n };\n\n return {\n name: INTEGRATION_NAME,\n // TODO: This currently only works for the first client this is setup\n // We may want to adjust this to check for client etc.\n setupOnce() {\n if (_options.setTimeout) {\n fill(WINDOW, 'setTimeout', _wrapTimeFunction);\n }\n\n if (_options.setInterval) {\n fill(WINDOW, 'setInterval', _wrapTimeFunction);\n }\n\n if (_options.requestAnimationFrame) {\n fill(WINDOW, 'requestAnimationFrame', _wrapRAF);\n }\n\n if (_options.XMLHttpRequest && 'XMLHttpRequest' in WINDOW) {\n fill(XMLHttpRequest.prototype, 'send', _wrapXHR);\n }\n\n const eventTargetOption = _options.eventTarget;\n if (eventTargetOption) {\n const eventTarget = Array.isArray(eventTargetOption) ? eventTargetOption : DEFAULT_EVENT_TARGET;\n eventTarget.forEach(_wrapEventTarget);\n }\n },\n };\n}) ;\n\nconst browserApiErrorsIntegration = defineIntegration(_browserApiErrorsIntegration);\n\n/**\n * Wrap timer functions and event targets to catch errors and provide better meta data.\n * @deprecated Use `browserApiErrorsIntegration()` instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst TryCatch = convertIntegrationFnToClass(\n INTEGRATION_NAME,\n browserApiErrorsIntegration,\n)\n\n;\n\nfunction _wrapTimeFunction(original) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return function ( ...args) {\n const originalCallback = args[0];\n args[0] = wrap(originalCallback, {\n mechanism: {\n data: { function: getFunctionName(original) },\n handled: false,\n type: 'instrument',\n },\n });\n return original.apply(this, args);\n };\n}\n\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction _wrapRAF(original) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return function ( callback) {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return original.apply(this, [\n wrap(callback, {\n mechanism: {\n data: {\n function: 'requestAnimationFrame',\n handler: getFunctionName(original),\n },\n handled: false,\n type: 'instrument',\n },\n }),\n ]);\n };\n}\n\nfunction _wrapXHR(originalSend) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n return function ( ...args) {\n // eslint-disable-next-line @typescript-eslint/no-this-alias\n const xhr = this;\n const xmlHttpRequestProps = ['onload', 'onerror', 'onprogress', 'onreadystatechange'];\n\n xmlHttpRequestProps.forEach(prop => {\n if (prop in xhr && typeof xhr[prop] === 'function') {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n fill(xhr, prop, function (original) {\n const wrapOptions = {\n mechanism: {\n data: {\n function: prop,\n handler: getFunctionName(original),\n },\n handled: false,\n type: 'instrument',\n },\n };\n\n // If Instrument integration has been called before TryCatch, get the name of original function\n const originalFunction = getOriginalFunction(original);\n if (originalFunction) {\n wrapOptions.mechanism.data.handler = getFunctionName(originalFunction);\n }\n\n // Otherwise wrap directly\n return wrap(original, wrapOptions);\n });\n }\n });\n\n return originalSend.apply(this, args);\n };\n}\n\nfunction _wrapEventTarget(target) {\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const globalObject = WINDOW ;\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n const proto = globalObject[target] && globalObject[target].prototype;\n\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access, no-prototype-builtins\n if (!proto || !proto.hasOwnProperty || !proto.hasOwnProperty('addEventListener')) {\n return;\n }\n\n fill(proto, 'addEventListener', function (original,)\n\n {\n return function (\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n\n eventName,\n fn,\n options,\n ) {\n try {\n if (typeof fn.handleEvent === 'function') {\n // ESlint disable explanation:\n // First, it is generally safe to call `wrap` with an unbound function. Furthermore, using `.bind()` would\n // introduce a bug here, because bind returns a new function that doesn't have our\n // flags(like __sentry_original__) attached. `wrap` checks for those flags to avoid unnecessary wrapping.\n // Without those flags, every call to addEventListener wraps the function again, causing a memory leak.\n // eslint-disable-next-line @typescript-eslint/unbound-method\n fn.handleEvent = wrap(fn.handleEvent, {\n mechanism: {\n data: {\n function: 'handleEvent',\n handler: getFunctionName(fn),\n target,\n },\n handled: false,\n type: 'instrument',\n },\n });\n }\n } catch (err) {\n // can sometimes get 'Permission denied to access property \"handle Event'\n }\n\n return original.apply(this, [\n eventName,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n wrap(fn , {\n mechanism: {\n data: {\n function: 'addEventListener',\n handler: getFunctionName(fn),\n target,\n },\n handled: false,\n type: 'instrument',\n },\n }),\n options,\n ]);\n };\n });\n\n fill(\n proto,\n 'removeEventListener',\n function (\n originalRemoveEventListener,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n ) {\n return function (\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n\n eventName,\n fn,\n options,\n ) {\n /**\n * There are 2 possible scenarios here:\n *\n * 1. Someone passes a callback, which was attached prior to Sentry initialization, or by using unmodified\n * method, eg. `document.addEventListener.call(el, name, handler). In this case, we treat this function\n * as a pass-through, and call original `removeEventListener` with it.\n *\n * 2. Someone passes a callback, which was attached after Sentry was initialized, which means that it was using\n * our wrapped version of `addEventListener`, which internally calls `wrap` helper.\n * This helper \"wraps\" whole callback inside a try/catch statement, and attached appropriate metadata to it,\n * in order for us to make a distinction between wrapped/non-wrapped functions possible.\n * If a function was wrapped, it has additional property of `__sentry_wrapped__`, holding the handler.\n *\n * When someone adds a handler prior to initialization, and then do it again, but after,\n * then we have to detach both of them. Otherwise, if we'd detach only wrapped one, it'd be impossible\n * to get rid of the initial handler and it'd stick there forever.\n */\n const wrappedEventHandler = fn ;\n try {\n const originalEventHandler = wrappedEventHandler && wrappedEventHandler.__sentry_wrapped__;\n if (originalEventHandler) {\n originalRemoveEventListener.call(this, eventName, originalEventHandler, options);\n }\n } catch (e) {\n // ignore, accessing __sentry_wrapped__ will throw in some Selenium environments\n }\n return originalRemoveEventListener.call(this, eventName, wrappedEventHandler, options);\n };\n },\n );\n}\n\nexport { TryCatch, browserApiErrorsIntegration };\n","import { createStackParser } from '@sentry/utils';\n\n// global reference to slice\nconst UNKNOWN_FUNCTION = '?';\n\nconst OPERA10_PRIORITY = 10;\nconst OPERA11_PRIORITY = 20;\nconst CHROME_PRIORITY = 30;\nconst WINJS_PRIORITY = 40;\nconst GECKO_PRIORITY = 50;\n\nfunction createFrame(filename, func, lineno, colno) {\n const frame = {\n filename,\n function: func,\n in_app: true, // All browser frames are considered in_app\n };\n\n if (lineno !== undefined) {\n frame.lineno = lineno;\n }\n\n if (colno !== undefined) {\n frame.colno = colno;\n }\n\n return frame;\n}\n\n// Chromium based browsers: Chrome, Brave, new Opera, new Edge\nconst chromeRegex =\n /^\\s*at (?:(.+?\\)(?: \\[.+\\])?|.*?) ?\\((?:address at )?)?(?:async )?((?:|[-a-z]+:|.*bundle|\\/)?.*?)(?::(\\d+))?(?::(\\d+))?\\)?\\s*$/i;\nconst chromeEvalRegex = /\\((\\S*)(?::(\\d+))(?::(\\d+))\\)/;\n\n// We cannot call this variable `chrome` because it can conflict with global `chrome` variable in certain environments\n// See: https://github.com/getsentry/sentry-javascript/issues/6880\nconst chromeStackParserFn = line => {\n const parts = chromeRegex.exec(line);\n\n if (parts) {\n const isEval = parts[2] && parts[2].indexOf('eval') === 0; // start of line\n\n if (isEval) {\n const subMatch = chromeEvalRegex.exec(parts[2]);\n\n if (subMatch) {\n // throw out eval line/column and use top-most line/column number\n parts[2] = subMatch[1]; // url\n parts[3] = subMatch[2]; // line\n parts[4] = subMatch[3]; // column\n }\n }\n\n // Kamil: One more hack won't hurt us right? Understanding and adding more rules on top of these regexps right now\n // would be way too time consuming. (TODO: Rewrite whole RegExp to be more readable)\n const [func, filename] = extractSafariExtensionDetails(parts[1] || UNKNOWN_FUNCTION, parts[2]);\n\n return createFrame(filename, func, parts[3] ? +parts[3] : undefined, parts[4] ? +parts[4] : undefined);\n }\n\n return;\n};\n\nconst chromeStackLineParser = [CHROME_PRIORITY, chromeStackParserFn];\n\n// gecko regex: `(?:bundle|\\d+\\.js)`: `bundle` is for react native, `\\d+\\.js` also but specifically for ram bundles because it\n// generates filenames without a prefix like `file://` the filenames in the stacktrace are just 42.js\n// We need this specific case for now because we want no other regex to match.\nconst geckoREgex =\n /^\\s*(.*?)(?:\\((.*?)\\))?(?:^|@)?((?:[-a-z]+)?:\\/.*?|\\[native code\\]|[^@]*(?:bundle|\\d+\\.js)|\\/[\\w\\-. /=]+)(?::(\\d+))?(?::(\\d+))?\\s*$/i;\nconst geckoEvalRegex = /(\\S+) line (\\d+)(?: > eval line \\d+)* > eval/i;\n\nconst gecko = line => {\n const parts = geckoREgex.exec(line);\n\n if (parts) {\n const isEval = parts[3] && parts[3].indexOf(' > eval') > -1;\n if (isEval) {\n const subMatch = geckoEvalRegex.exec(parts[3]);\n\n if (subMatch) {\n // throw out eval line/column and use top-most line number\n parts[1] = parts[1] || 'eval';\n parts[3] = subMatch[1];\n parts[4] = subMatch[2];\n parts[5] = ''; // no column when eval\n }\n }\n\n let filename = parts[3];\n let func = parts[1] || UNKNOWN_FUNCTION;\n [func, filename] = extractSafariExtensionDetails(func, filename);\n\n return createFrame(filename, func, parts[4] ? +parts[4] : undefined, parts[5] ? +parts[5] : undefined);\n }\n\n return;\n};\n\nconst geckoStackLineParser = [GECKO_PRIORITY, gecko];\n\nconst winjsRegex = /^\\s*at (?:((?:\\[object object\\])?.+) )?\\(?((?:[-a-z]+):.*?):(\\d+)(?::(\\d+))?\\)?\\s*$/i;\n\nconst winjs = line => {\n const parts = winjsRegex.exec(line);\n\n return parts\n ? createFrame(parts[2], parts[1] || UNKNOWN_FUNCTION, +parts[3], parts[4] ? +parts[4] : undefined)\n : undefined;\n};\n\nconst winjsStackLineParser = [WINJS_PRIORITY, winjs];\n\nconst opera10Regex = / line (\\d+).*script (?:in )?(\\S+)(?:: in function (\\S+))?$/i;\n\nconst opera10 = line => {\n const parts = opera10Regex.exec(line);\n return parts ? createFrame(parts[2], parts[3] || UNKNOWN_FUNCTION, +parts[1]) : undefined;\n};\n\nconst opera10StackLineParser = [OPERA10_PRIORITY, opera10];\n\nconst opera11Regex =\n / line (\\d+), column (\\d+)\\s*(?:in (?:]+)>|([^)]+))\\(.*\\))? in (.*):\\s*$/i;\n\nconst opera11 = line => {\n const parts = opera11Regex.exec(line);\n return parts ? createFrame(parts[5], parts[3] || parts[4] || UNKNOWN_FUNCTION, +parts[1], +parts[2]) : undefined;\n};\n\nconst opera11StackLineParser = [OPERA11_PRIORITY, opera11];\n\nconst defaultStackLineParsers = [chromeStackLineParser, geckoStackLineParser, winjsStackLineParser];\n\nconst defaultStackParser = createStackParser(...defaultStackLineParsers);\n\n/**\n * Safari web extensions, starting version unknown, can produce \"frames-only\" stacktraces.\n * What it means, is that instead of format like:\n *\n * Error: wat\n * at function@url:row:col\n * at function@url:row:col\n * at function@url:row:col\n *\n * it produces something like:\n *\n * function@url:row:col\n * function@url:row:col\n * function@url:row:col\n *\n * Because of that, it won't be captured by `chrome` RegExp and will fall into `Gecko` branch.\n * This function is extracted so that we can use it in both places without duplicating the logic.\n * Unfortunately \"just\" changing RegExp is too complicated now and making it pass all tests\n * and fix this case seems like an impossible, or at least way too time-consuming task.\n */\nconst extractSafariExtensionDetails = (func, filename) => {\n const isSafariExtension = func.indexOf('safari-extension') !== -1;\n const isSafariWebExtension = func.indexOf('safari-web-extension') !== -1;\n\n return isSafariExtension || isSafariWebExtension\n ? [\n func.indexOf('@') !== -1 ? func.split('@')[0] : UNKNOWN_FUNCTION,\n isSafariExtension ? `safari-extension:${filename}` : `safari-web-extension:${filename}`,\n ]\n : [func, filename];\n};\n\nexport { chromeStackLineParser, defaultStackLineParsers, defaultStackParser, geckoStackLineParser, opera10StackLineParser, opera11StackLineParser, winjsStackLineParser };\n","import { makePromiseBuffer, forEachEnvelopeItem, envelopeItemTypeToDataCategory, isRateLimited, resolvedSyncPromise, createEnvelope, SentryError, logger, serializeEnvelope, updateRateLimits } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\n\nconst DEFAULT_TRANSPORT_BUFFER_SIZE = 30;\n\n/**\n * Creates an instance of a Sentry `Transport`\n *\n * @param options\n * @param makeRequest\n */\nfunction createTransport(\n options,\n makeRequest,\n buffer = makePromiseBuffer(\n options.bufferSize || DEFAULT_TRANSPORT_BUFFER_SIZE,\n ),\n) {\n let rateLimits = {};\n const flush = (timeout) => buffer.drain(timeout);\n\n function send(envelope) {\n const filteredEnvelopeItems = [];\n\n // Drop rate limited items from envelope\n forEachEnvelopeItem(envelope, (item, type) => {\n const dataCategory = envelopeItemTypeToDataCategory(type);\n if (isRateLimited(rateLimits, dataCategory)) {\n const event = getEventForEnvelopeItem(item, type);\n options.recordDroppedEvent('ratelimit_backoff', dataCategory, event);\n } else {\n filteredEnvelopeItems.push(item);\n }\n });\n\n // Skip sending if envelope is empty after filtering out rate limited events\n if (filteredEnvelopeItems.length === 0) {\n return resolvedSyncPromise();\n }\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const filteredEnvelope = createEnvelope(envelope[0], filteredEnvelopeItems );\n\n // Creates client report for each item in an envelope\n const recordEnvelopeLoss = (reason) => {\n forEachEnvelopeItem(filteredEnvelope, (item, type) => {\n const event = getEventForEnvelopeItem(item, type);\n options.recordDroppedEvent(reason, envelopeItemTypeToDataCategory(type), event);\n });\n };\n\n const requestTask = () =>\n makeRequest({ body: serializeEnvelope(filteredEnvelope, options.textEncoder) }).then(\n response => {\n // We don't want to throw on NOK responses, but we want to at least log them\n if (response.statusCode !== undefined && (response.statusCode < 200 || response.statusCode >= 300)) {\n DEBUG_BUILD && logger.warn(`Sentry responded with status code ${response.statusCode} to sent event.`);\n }\n\n rateLimits = updateRateLimits(rateLimits, response);\n return response;\n },\n error => {\n recordEnvelopeLoss('network_error');\n throw error;\n },\n );\n\n return buffer.add(requestTask).then(\n result => result,\n error => {\n if (error instanceof SentryError) {\n DEBUG_BUILD && logger.error('Skipped sending event because buffer is full.');\n recordEnvelopeLoss('queue_overflow');\n return resolvedSyncPromise();\n } else {\n throw error;\n }\n },\n );\n }\n\n // We use this to identifify if the transport is the base transport\n // TODO (v8): Remove this again as we'll no longer need it\n send.__sentry__baseTransport__ = true;\n\n return {\n send,\n flush,\n };\n}\n\nfunction getEventForEnvelopeItem(item, type) {\n if (type !== 'event' && type !== 'transaction') {\n return undefined;\n }\n\n return Array.isArray(item) ? (item )[1] : undefined;\n}\n\nexport { DEFAULT_TRANSPORT_BUFFER_SIZE, createTransport };\n","import { SentryError } from './error.js';\nimport { rejectedSyncPromise, SyncPromise, resolvedSyncPromise } from './syncpromise.js';\n\n/**\n * Creates an new PromiseBuffer object with the specified limit\n * @param limit max number of promises that can be stored in the buffer\n */\nfunction makePromiseBuffer(limit) {\n const buffer = [];\n\n function isReady() {\n return limit === undefined || buffer.length < limit;\n }\n\n /**\n * Remove a promise from the queue.\n *\n * @param task Can be any PromiseLike\n * @returns Removed promise.\n */\n function remove(task) {\n return buffer.splice(buffer.indexOf(task), 1)[0];\n }\n\n /**\n * Add a promise (representing an in-flight action) to the queue, and set it to remove itself on fulfillment.\n *\n * @param taskProducer A function producing any PromiseLike; In previous versions this used to be `task:\n * PromiseLike`, but under that model, Promises were instantly created on the call-site and their executor\n * functions therefore ran immediately. Thus, even if the buffer was full, the action still happened. By\n * requiring the promise to be wrapped in a function, we can defer promise creation until after the buffer\n * limit check.\n * @returns The original promise.\n */\n function add(taskProducer) {\n if (!isReady()) {\n return rejectedSyncPromise(new SentryError('Not adding Promise because buffer limit was reached.'));\n }\n\n // start the task and add its promise to the queue\n const task = taskProducer();\n if (buffer.indexOf(task) === -1) {\n buffer.push(task);\n }\n void task\n .then(() => remove(task))\n // Use `then(null, rejectionHandler)` rather than `catch(rejectionHandler)` so that we can use `PromiseLike`\n // rather than `Promise`. `PromiseLike` doesn't have a `.catch` method, making its polyfill smaller. (ES5 didn't\n // have promises, so TS has to polyfill when down-compiling.)\n .then(null, () =>\n remove(task).then(null, () => {\n // We have to add another catch here because `remove()` starts a new promise chain.\n }),\n );\n return task;\n }\n\n /**\n * Wait for all promises in the queue to resolve or for timeout to expire, whichever comes first.\n *\n * @param timeout The time, in ms, after which to resolve to `false` if the queue is still non-empty. Passing `0` (or\n * not passing anything) will make the promise wait as long as it takes for the queue to drain before resolving to\n * `true`.\n * @returns A promise which will resolve to `true` if the queue is already empty or drains before the timeout, and\n * `false` otherwise\n */\n function drain(timeout) {\n return new SyncPromise((resolve, reject) => {\n let counter = buffer.length;\n\n if (!counter) {\n return resolve(true);\n }\n\n // wait for `timeout` ms and then resolve to `false` (if not cancelled first)\n const capturedSetTimeout = setTimeout(() => {\n if (timeout && timeout > 0) {\n resolve(false);\n }\n }, timeout);\n\n // if all promises resolve in time, cancel the timer and resolve to `true`\n buffer.forEach(item => {\n void resolvedSyncPromise(item).then(() => {\n if (!--counter) {\n clearTimeout(capturedSetTimeout);\n resolve(true);\n }\n }, reject);\n });\n });\n }\n\n return {\n $: buffer,\n add,\n drain,\n };\n}\n\nexport { makePromiseBuffer };\n","// Intentionally keeping the key broad, as we don't know for sure what rate limit headers get returned from backend\n\nconst DEFAULT_RETRY_AFTER = 60 * 1000; // 60 seconds\n\n/**\n * Extracts Retry-After value from the request header or returns default value\n * @param header string representation of 'Retry-After' header\n * @param now current unix timestamp\n *\n */\nfunction parseRetryAfterHeader(header, now = Date.now()) {\n const headerDelay = parseInt(`${header}`, 10);\n if (!isNaN(headerDelay)) {\n return headerDelay * 1000;\n }\n\n const headerDate = Date.parse(`${header}`);\n if (!isNaN(headerDate)) {\n return headerDate - now;\n }\n\n return DEFAULT_RETRY_AFTER;\n}\n\n/**\n * Gets the time that the given category is disabled until for rate limiting.\n * In case no category-specific limit is set but a general rate limit across all categories is active,\n * that time is returned.\n *\n * @return the time in ms that the category is disabled until or 0 if there's no active rate limit.\n */\nfunction disabledUntil(limits, dataCategory) {\n return limits[dataCategory] || limits.all || 0;\n}\n\n/**\n * Checks if a category is rate limited\n */\nfunction isRateLimited(limits, dataCategory, now = Date.now()) {\n return disabledUntil(limits, dataCategory) > now;\n}\n\n/**\n * Update ratelimits from incoming headers.\n *\n * @return the updated RateLimits object.\n */\nfunction updateRateLimits(\n limits,\n { statusCode, headers },\n now = Date.now(),\n) {\n const updatedRateLimits = {\n ...limits,\n };\n\n // \"The name is case-insensitive.\"\n // https://developer.mozilla.org/en-US/docs/Web/API/Headers/get\n const rateLimitHeader = headers && headers['x-sentry-rate-limits'];\n const retryAfterHeader = headers && headers['retry-after'];\n\n if (rateLimitHeader) {\n /**\n * rate limit headers are of the form\n *
,
,..\n * where each
is of the form\n * : : : : \n * where\n * is a delay in seconds\n * is the event type(s) (error, transaction, etc) being rate limited and is of the form\n * ;;...\n * is what's being limited (org, project, or key) - ignored by SDK\n * is an arbitrary string like \"org_quota\" - ignored by SDK\n * Semicolon-separated list of metric namespace identifiers. Defines which namespace(s) will be affected.\n * Only present if rate limit applies to the metric_bucket data category.\n */\n for (const limit of rateLimitHeader.trim().split(',')) {\n const [retryAfter, categories, , , namespaces] = limit.split(':', 5);\n const headerDelay = parseInt(retryAfter, 10);\n const delay = (!isNaN(headerDelay) ? headerDelay : 60) * 1000; // 60sec default\n if (!categories) {\n updatedRateLimits.all = now + delay;\n } else {\n for (const category of categories.split(';')) {\n if (category === 'metric_bucket') {\n // namespaces will be present when category === 'metric_bucket'\n if (!namespaces || namespaces.split(';').includes('custom')) {\n updatedRateLimits[category] = now + delay;\n }\n } else {\n updatedRateLimits[category] = now + delay;\n }\n }\n }\n }\n } else if (retryAfterHeader) {\n updatedRateLimits.all = now + parseRetryAfterHeader(retryAfterHeader, now);\n } else if (statusCode === 429) {\n updatedRateLimits.all = now + 60 * 1000;\n }\n\n return updatedRateLimits;\n}\n\nexport { DEFAULT_RETRY_AFTER, disabledUntil, isRateLimited, parseRetryAfterHeader, updateRateLimits };\n","import { isNativeFetch, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { WINDOW } from '../helpers.js';\n\nlet cachedFetchImpl = undefined;\n\n/**\n * A special usecase for incorrectly wrapped Fetch APIs in conjunction with ad-blockers.\n * Whenever someone wraps the Fetch API and returns the wrong promise chain,\n * this chain becomes orphaned and there is no possible way to capture it's rejections\n * other than allowing it bubble up to this very handler. eg.\n *\n * const f = window.fetch;\n * window.fetch = function () {\n * const p = f.apply(this, arguments);\n *\n * p.then(function() {\n * console.log('hi.');\n * });\n *\n * return p;\n * }\n *\n * `p.then(function () { ... })` is producing a completely separate promise chain,\n * however, what's returned is `p` - the result of original `fetch` call.\n *\n * This mean, that whenever we use the Fetch API to send our own requests, _and_\n * some ad-blocker blocks it, this orphaned chain will _always_ reject,\n * effectively causing another event to be captured.\n * This makes a whole process become an infinite loop, which we need to somehow\n * deal with, and break it in one way or another.\n *\n * To deal with this issue, we are making sure that we _always_ use the real\n * browser Fetch API, instead of relying on what `window.fetch` exposes.\n * The only downside to this would be missing our own requests as breadcrumbs,\n * but because we are already not doing this, it should be just fine.\n *\n * Possible failed fetch error messages per-browser:\n *\n * Chrome: Failed to fetch\n * Edge: Failed to Fetch\n * Firefox: NetworkError when attempting to fetch resource\n * Safari: resource blocked by content blocker\n */\nfunction getNativeFetchImplementation() {\n if (cachedFetchImpl) {\n return cachedFetchImpl;\n }\n\n /* eslint-disable @typescript-eslint/unbound-method */\n\n // Fast path to avoid DOM I/O\n if (isNativeFetch(WINDOW.fetch)) {\n return (cachedFetchImpl = WINDOW.fetch.bind(WINDOW));\n }\n\n const document = WINDOW.document;\n let fetchImpl = WINDOW.fetch;\n // eslint-disable-next-line deprecation/deprecation\n if (document && typeof document.createElement === 'function') {\n try {\n const sandbox = document.createElement('iframe');\n sandbox.hidden = true;\n document.head.appendChild(sandbox);\n const contentWindow = sandbox.contentWindow;\n if (contentWindow && contentWindow.fetch) {\n fetchImpl = contentWindow.fetch;\n }\n document.head.removeChild(sandbox);\n } catch (e) {\n DEBUG_BUILD && logger.warn('Could not create sandbox iframe for pure fetch check, bailing to window.fetch: ', e);\n }\n }\n\n return (cachedFetchImpl = fetchImpl.bind(WINDOW));\n /* eslint-enable @typescript-eslint/unbound-method */\n}\n\n/** Clears cached fetch impl */\nfunction clearCachedFetchImplementation() {\n cachedFetchImpl = undefined;\n}\n\nexport { clearCachedFetchImplementation, getNativeFetchImplementation };\n","import { createTransport } from '@sentry/core';\nimport { rejectedSyncPromise } from '@sentry/utils';\nimport { getNativeFetchImplementation, clearCachedFetchImplementation } from './utils.js';\n\n/**\n * Creates a Transport that uses the Fetch API to send events to Sentry.\n */\nfunction makeFetchTransport(\n options,\n nativeFetch = getNativeFetchImplementation(),\n) {\n let pendingBodySize = 0;\n let pendingCount = 0;\n\n function makeRequest(request) {\n const requestSize = request.body.length;\n pendingBodySize += requestSize;\n pendingCount++;\n\n const requestOptions = {\n body: request.body,\n method: 'POST',\n referrerPolicy: 'origin',\n headers: options.headers,\n // Outgoing requests are usually cancelled when navigating to a different page, causing a \"TypeError: Failed to\n // fetch\" error and sending a \"network_error\" client-outcome - in Chrome, the request status shows \"(cancelled)\".\n // The `keepalive` flag keeps outgoing requests alive, even when switching pages. We want this since we're\n // frequently sending events right before the user is switching pages (eg. whenfinishing navigation transactions).\n // Gotchas:\n // - `keepalive` isn't supported by Firefox\n // - As per spec (https://fetch.spec.whatwg.org/#http-network-or-cache-fetch):\n // If the sum of contentLength and inflightKeepaliveBytes is greater than 64 kibibytes, then return a network error.\n // We will therefore only activate the flag when we're below that limit.\n // There is also a limit of requests that can be open at the same time, so we also limit this to 15\n // See https://github.com/getsentry/sentry-javascript/pull/7553 for details\n keepalive: pendingBodySize <= 60000 && pendingCount < 15,\n ...options.fetchOptions,\n };\n\n try {\n return nativeFetch(options.url, requestOptions).then(response => {\n pendingBodySize -= requestSize;\n pendingCount--;\n return {\n statusCode: response.status,\n headers: {\n 'x-sentry-rate-limits': response.headers.get('X-Sentry-Rate-Limits'),\n 'retry-after': response.headers.get('Retry-After'),\n },\n };\n });\n } catch (e) {\n clearCachedFetchImplementation();\n pendingBodySize -= requestSize;\n pendingCount--;\n return rejectedSyncPromise(e);\n }\n }\n\n return createTransport(options, makeRequest);\n}\n\nexport { makeFetchTransport };\n","import { createTransport } from '@sentry/core';\nimport { SyncPromise } from '@sentry/utils';\n\n/**\n * The DONE ready state for XmlHttpRequest\n *\n * Defining it here as a constant b/c XMLHttpRequest.DONE is not always defined\n * (e.g. during testing, it is `undefined`)\n *\n * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest/readyState}\n */\nconst XHR_READYSTATE_DONE = 4;\n\n/**\n * Creates a Transport that uses the XMLHttpRequest API to send events to Sentry.\n */\nfunction makeXHRTransport(options) {\n function makeRequest(request) {\n return new SyncPromise((resolve, reject) => {\n const xhr = new XMLHttpRequest();\n\n xhr.onerror = reject;\n\n xhr.onreadystatechange = () => {\n if (xhr.readyState === XHR_READYSTATE_DONE) {\n resolve({\n statusCode: xhr.status,\n headers: {\n 'x-sentry-rate-limits': xhr.getResponseHeader('X-Sentry-Rate-Limits'),\n 'retry-after': xhr.getResponseHeader('Retry-After'),\n },\n });\n }\n };\n\n xhr.open('POST', options.url);\n\n for (const header in options.headers) {\n if (Object.prototype.hasOwnProperty.call(options.headers, header)) {\n xhr.setRequestHeader(header, options.headers[header]);\n }\n }\n\n xhr.send(request.body);\n });\n }\n\n return createTransport(options, makeRequest);\n}\n\nexport { makeXHRTransport };\n","import { inboundFiltersIntegration, functionToStringIntegration, getIntegrationsToSetup, initAndBind, getReportDialogEndpoint, getCurrentHub, startSession, captureSession, getClient } from '@sentry/core';\nimport { stackParserFromStackParserOptions, supportsFetch, logger, addHistoryInstrumentationHandler } from '@sentry/utils';\nimport { BrowserClient } from './client.js';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { WINDOW, wrap as wrap$1 } from './helpers.js';\nimport { breadcrumbsIntegration } from './integrations/breadcrumbs.js';\nimport { dedupeIntegration } from './integrations/dedupe.js';\nimport { globalHandlersIntegration } from './integrations/globalhandlers.js';\nimport { httpContextIntegration } from './integrations/httpcontext.js';\nimport { linkedErrorsIntegration } from './integrations/linkederrors.js';\nimport { browserApiErrorsIntegration } from './integrations/trycatch.js';\nimport { defaultStackParser } from './stack-parsers.js';\nimport { makeFetchTransport } from './transports/fetch.js';\nimport { makeXHRTransport } from './transports/xhr.js';\n\n/** @deprecated Use `getDefaultIntegrations(options)` instead. */\nconst defaultIntegrations = [\n inboundFiltersIntegration(),\n functionToStringIntegration(),\n browserApiErrorsIntegration(),\n breadcrumbsIntegration(),\n globalHandlersIntegration(),\n linkedErrorsIntegration(),\n dedupeIntegration(),\n httpContextIntegration(),\n];\n\n/** Get the default integrations for the browser SDK. */\nfunction getDefaultIntegrations(_options) {\n // We return a copy of the defaultIntegrations here to avoid mutating this\n return [\n // eslint-disable-next-line deprecation/deprecation\n ...defaultIntegrations,\n ];\n}\n\n/**\n * A magic string that build tooling can leverage in order to inject a release value into the SDK.\n */\n\n/**\n * The Sentry Browser SDK Client.\n *\n * To use this SDK, call the {@link init} function as early as possible when\n * loading the web page. To set context information or send manual events, use\n * the provided methods.\n *\n * @example\n *\n * ```\n *\n * import { init } from '@sentry/browser';\n *\n * init({\n * dsn: '__DSN__',\n * // ...\n * });\n * ```\n *\n * @example\n * ```\n *\n * import { configureScope } from '@sentry/browser';\n * configureScope((scope: Scope) => {\n * scope.setExtra({ battery: 0.7 });\n * scope.setTag({ user_mode: 'admin' });\n * scope.setUser({ id: '4711' });\n * });\n * ```\n *\n * @example\n * ```\n *\n * import { addBreadcrumb } from '@sentry/browser';\n * addBreadcrumb({\n * message: 'My Breadcrumb',\n * // ...\n * });\n * ```\n *\n * @example\n *\n * ```\n *\n * import * as Sentry from '@sentry/browser';\n * Sentry.captureMessage('Hello, world!');\n * Sentry.captureException(new Error('Good bye'));\n * Sentry.captureEvent({\n * message: 'Manual',\n * stacktrace: [\n * // ...\n * ],\n * });\n * ```\n *\n * @see {@link BrowserOptions} for documentation on configuration options.\n */\nfunction init(options = {}) {\n if (options.defaultIntegrations === undefined) {\n options.defaultIntegrations = getDefaultIntegrations();\n }\n if (options.release === undefined) {\n // This allows build tooling to find-and-replace __SENTRY_RELEASE__ to inject a release value\n if (typeof __SENTRY_RELEASE__ === 'string') {\n options.release = __SENTRY_RELEASE__;\n }\n\n // This supports the variable that sentry-webpack-plugin injects\n if (WINDOW.SENTRY_RELEASE && WINDOW.SENTRY_RELEASE.id) {\n options.release = WINDOW.SENTRY_RELEASE.id;\n }\n }\n if (options.autoSessionTracking === undefined) {\n options.autoSessionTracking = true;\n }\n if (options.sendClientReports === undefined) {\n options.sendClientReports = true;\n }\n\n const clientOptions = {\n ...options,\n stackParser: stackParserFromStackParserOptions(options.stackParser || defaultStackParser),\n integrations: getIntegrationsToSetup(options),\n transport: options.transport || (supportsFetch() ? makeFetchTransport : makeXHRTransport),\n };\n\n initAndBind(BrowserClient, clientOptions);\n\n if (options.autoSessionTracking) {\n startSessionTracking();\n }\n}\n\nconst showReportDialog = (\n // eslint-disable-next-line deprecation/deprecation\n options = {},\n // eslint-disable-next-line deprecation/deprecation\n hub = getCurrentHub(),\n) => {\n // doesn't work without a document (React Native)\n if (!WINDOW.document) {\n DEBUG_BUILD && logger.error('Global document not defined in showReportDialog call');\n return;\n }\n\n // eslint-disable-next-line deprecation/deprecation\n const { client, scope } = hub.getStackTop();\n const dsn = options.dsn || (client && client.getDsn());\n if (!dsn) {\n DEBUG_BUILD && logger.error('DSN not configured for showReportDialog call');\n return;\n }\n\n if (scope) {\n options.user = {\n ...scope.getUser(),\n ...options.user,\n };\n }\n\n if (!options.eventId) {\n // eslint-disable-next-line deprecation/deprecation\n options.eventId = hub.lastEventId();\n }\n\n const script = WINDOW.document.createElement('script');\n script.async = true;\n script.crossOrigin = 'anonymous';\n script.src = getReportDialogEndpoint(dsn, options);\n\n if (options.onLoad) {\n script.onload = options.onLoad;\n }\n\n const { onClose } = options;\n if (onClose) {\n const reportDialogClosedMessageHandler = (event) => {\n if (event.data === '__sentry_reportdialog_closed__') {\n try {\n onClose();\n } finally {\n WINDOW.removeEventListener('message', reportDialogClosedMessageHandler);\n }\n }\n };\n WINDOW.addEventListener('message', reportDialogClosedMessageHandler);\n }\n\n const injectionPoint = WINDOW.document.head || WINDOW.document.body;\n if (injectionPoint) {\n injectionPoint.appendChild(script);\n } else {\n DEBUG_BUILD && logger.error('Not injecting report dialog. No injection point found in HTML');\n }\n};\n\n/**\n * This function is here to be API compatible with the loader.\n * @hidden\n */\nfunction forceLoad() {\n // Noop\n}\n\n/**\n * This function is here to be API compatible with the loader.\n * @hidden\n */\nfunction onLoad(callback) {\n callback();\n}\n\n/**\n * Wrap code within a try/catch block so the SDK is able to capture errors.\n *\n * @deprecated This function will be removed in v8.\n * It is not part of Sentry's official API and it's easily replaceable by using a try/catch block\n * and calling Sentry.captureException.\n *\n * @param fn A function to wrap.\n *\n * @returns The result of wrapped function call.\n */\n// TODO(v8): Remove this function\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction wrap(fn) {\n return wrap$1(fn)();\n}\n\n/**\n * Enable automatic Session Tracking for the initial page load.\n */\nfunction startSessionTracking() {\n if (typeof WINDOW.document === 'undefined') {\n DEBUG_BUILD && logger.warn('Session tracking in non-browser environment with @sentry/browser is not supported.');\n return;\n }\n\n // The session duration for browser sessions does not track a meaningful\n // concept that can be used as a metric.\n // Automatically captured sessions are akin to page views, and thus we\n // discard their duration.\n startSession({ ignoreDuration: true });\n captureSession();\n\n // We want to create a session for every navigation as well\n addHistoryInstrumentationHandler(({ from, to }) => {\n // Don't create an additional session for the initial route or if the location did not change\n if (from !== undefined && from !== to) {\n startSession({ ignoreDuration: true });\n captureSession();\n }\n });\n}\n\n/**\n * Captures user feedback and sends it to Sentry.\n */\nfunction captureUserFeedback(feedback) {\n const client = getClient();\n if (client) {\n client.captureUserFeedback(feedback);\n }\n}\n\nexport { captureUserFeedback, defaultIntegrations, forceLoad, getDefaultIntegrations, init, onLoad, showReportDialog, wrap };\n","import { logger, consoleSandbox } from '@sentry/utils';\nimport { DEBUG_BUILD } from './debug-build.js';\nimport { getCurrentScope } from './exports.js';\nimport { getCurrentHub } from './hub.js';\n\n/** A class object that can instantiate Client objects. */\n\n/**\n * Internal function to create a new SDK client instance. The client is\n * installed and then bound to the current scope.\n *\n * @param clientClass The client class to instantiate.\n * @param options Options to pass to the client.\n */\nfunction initAndBind(\n clientClass,\n options,\n) {\n if (options.debug === true) {\n if (DEBUG_BUILD) {\n logger.enable();\n } else {\n // use `console.warn` rather than `logger.warn` since by non-debug bundles have all `logger.x` statements stripped\n consoleSandbox(() => {\n // eslint-disable-next-line no-console\n console.warn('[Sentry] Cannot initialize SDK with `debug` option using a non-debug bundle.');\n });\n }\n }\n const scope = getCurrentScope();\n scope.update(options.initialScope);\n\n const client = new clientClass(options);\n setCurrentClient(client);\n initializeClient(client);\n}\n\n/**\n * Make the given client the current client.\n */\nfunction setCurrentClient(client) {\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n // eslint-disable-next-line deprecation/deprecation\n const top = hub.getStackTop();\n top.client = client;\n top.scope.setClient(client);\n}\n\n/**\n * Initialize the client for the current scope.\n * Make sure to call this after `setCurrentClient()`.\n */\nfunction initializeClient(client) {\n if (client.init) {\n client.init();\n // TODO v8: Remove this fallback\n // eslint-disable-next-line deprecation/deprecation\n } else if (client.setupIntegrations) {\n // eslint-disable-next-line deprecation/deprecation\n client.setupIntegrations();\n }\n}\n\nexport { initAndBind, setCurrentClient };\n","/**\n * Use this attribute to represent the source of a span.\n * Should be one of: custom, url, route, view, component, task, unknown\n *\n */\nconst SEMANTIC_ATTRIBUTE_SENTRY_SOURCE = 'sentry.source';\n\n/**\n * Use this attribute to represent the sample rate used for a span.\n */\nconst SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE = 'sentry.sample_rate';\n\n/**\n * Use this attribute to represent the operation of a span.\n */\nconst SEMANTIC_ATTRIBUTE_SENTRY_OP = 'sentry.op';\n\n/**\n * Use this attribute to represent the origin of a span.\n */\nconst SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN = 'sentry.origin';\n\n/**\n * The id of the profile that this span occured in.\n */\nconst SEMANTIC_ATTRIBUTE_PROFILE_ID = 'profile_id';\n\nexport { SEMANTIC_ATTRIBUTE_PROFILE_ID, SEMANTIC_ATTRIBUTE_SENTRY_OP, SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE, SEMANTIC_ATTRIBUTE_SENTRY_SOURCE };\n","/** The status of an Span.\n *\n * @deprecated Use string literals - if you require type casting, cast to SpanStatusType type\n */\nvar SpanStatus; (function (SpanStatus) {\n /** The operation completed successfully. */\n const Ok = 'ok'; SpanStatus[\"Ok\"] = Ok;\n /** Deadline expired before operation could complete. */\n const DeadlineExceeded = 'deadline_exceeded'; SpanStatus[\"DeadlineExceeded\"] = DeadlineExceeded;\n /** 401 Unauthorized (actually does mean unauthenticated according to RFC 7235) */\n const Unauthenticated = 'unauthenticated'; SpanStatus[\"Unauthenticated\"] = Unauthenticated;\n /** 403 Forbidden */\n const PermissionDenied = 'permission_denied'; SpanStatus[\"PermissionDenied\"] = PermissionDenied;\n /** 404 Not Found. Some requested entity (file or directory) was not found. */\n const NotFound = 'not_found'; SpanStatus[\"NotFound\"] = NotFound;\n /** 429 Too Many Requests */\n const ResourceExhausted = 'resource_exhausted'; SpanStatus[\"ResourceExhausted\"] = ResourceExhausted;\n /** Client specified an invalid argument. 4xx. */\n const InvalidArgument = 'invalid_argument'; SpanStatus[\"InvalidArgument\"] = InvalidArgument;\n /** 501 Not Implemented */\n const Unimplemented = 'unimplemented'; SpanStatus[\"Unimplemented\"] = Unimplemented;\n /** 503 Service Unavailable */\n const Unavailable = 'unavailable'; SpanStatus[\"Unavailable\"] = Unavailable;\n /** Other/generic 5xx. */\n const InternalError = 'internal_error'; SpanStatus[\"InternalError\"] = InternalError;\n /** Unknown. Any non-standard HTTP status code. */\n const UnknownError = 'unknown_error'; SpanStatus[\"UnknownError\"] = UnknownError;\n /** The operation was cancelled (typically by the user). */\n const Cancelled = 'cancelled'; SpanStatus[\"Cancelled\"] = Cancelled;\n /** Already exists (409) */\n const AlreadyExists = 'already_exists'; SpanStatus[\"AlreadyExists\"] = AlreadyExists;\n /** Operation was rejected because the system is not in a state required for the operation's */\n const FailedPrecondition = 'failed_precondition'; SpanStatus[\"FailedPrecondition\"] = FailedPrecondition;\n /** The operation was aborted, typically due to a concurrency issue. */\n const Aborted = 'aborted'; SpanStatus[\"Aborted\"] = Aborted;\n /** Operation was attempted past the valid range. */\n const OutOfRange = 'out_of_range'; SpanStatus[\"OutOfRange\"] = OutOfRange;\n /** Unrecoverable data loss or corruption */\n const DataLoss = 'data_loss'; SpanStatus[\"DataLoss\"] = DataLoss;\n})(SpanStatus || (SpanStatus = {}));\n\n/**\n * Converts a HTTP status code into a {@link SpanStatusType}.\n *\n * @param httpStatus The HTTP response status code.\n * @returns The span status or unknown_error.\n */\nfunction getSpanStatusFromHttpCode(httpStatus) {\n if (httpStatus < 400 && httpStatus >= 100) {\n return 'ok';\n }\n\n if (httpStatus >= 400 && httpStatus < 500) {\n switch (httpStatus) {\n case 401:\n return 'unauthenticated';\n case 403:\n return 'permission_denied';\n case 404:\n return 'not_found';\n case 409:\n return 'already_exists';\n case 413:\n return 'failed_precondition';\n case 429:\n return 'resource_exhausted';\n default:\n return 'invalid_argument';\n }\n }\n\n if (httpStatus >= 500 && httpStatus < 600) {\n switch (httpStatus) {\n case 501:\n return 'unimplemented';\n case 503:\n return 'unavailable';\n case 504:\n return 'deadline_exceeded';\n default:\n return 'internal_error';\n }\n }\n\n return 'unknown_error';\n}\n\n/**\n * Converts a HTTP status code into a {@link SpanStatusType}.\n *\n * @deprecated Use {@link spanStatusFromHttpCode} instead.\n * This export will be removed in v8 as the signature contains a typo.\n *\n * @param httpStatus The HTTP response status code.\n * @returns The span status or unknown_error.\n */\nconst spanStatusfromHttpCode = getSpanStatusFromHttpCode;\n\n/**\n * Sets the Http status attributes on the current span based on the http code.\n * Additionally, the span's status is updated, depending on the http code.\n */\nfunction setHttpStatus(span, httpStatus) {\n // TODO (v8): Remove these calls\n // Relay does not require us to send the status code as a tag\n // For now, just because users might expect it to land as a tag we keep sending it.\n // Same with data.\n // In v8, we replace both, simply with\n // span.setAttribute('http.response.status_code', httpStatus);\n\n // eslint-disable-next-line deprecation/deprecation\n span.setTag('http.status_code', String(httpStatus));\n // eslint-disable-next-line deprecation/deprecation\n span.setData('http.response.status_code', httpStatus);\n\n const spanStatus = getSpanStatusFromHttpCode(httpStatus);\n if (spanStatus !== 'unknown_error') {\n span.setStatus(spanStatus);\n }\n}\n\nexport { SpanStatus, getSpanStatusFromHttpCode, setHttpStatus, spanStatusfromHttpCode };\n","import { uuid4, timestampInSeconds, logger, dropUndefinedKeys } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { getMetricSummaryJsonForSpan } from '../metrics/metric-summary.js';\nimport { SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, SEMANTIC_ATTRIBUTE_SENTRY_OP, SEMANTIC_ATTRIBUTE_PROFILE_ID } from '../semanticAttributes.js';\nimport { getRootSpan } from '../utils/getRootSpan.js';\nimport { TRACE_FLAG_SAMPLED, TRACE_FLAG_NONE, spanToJSON, spanTimeInputToSeconds, spanToTraceHeader, spanToTraceContext } from '../utils/spanUtils.js';\nimport { setHttpStatus } from './spanstatus.js';\n\n/**\n * Keeps track of finished spans for a given transaction\n * @internal\n * @hideconstructor\n * @hidden\n */\nclass SpanRecorder {\n\n constructor(maxlen = 1000) {\n this._maxlen = maxlen;\n this.spans = [];\n }\n\n /**\n * This is just so that we don't run out of memory while recording a lot\n * of spans. At some point we just stop and flush out the start of the\n * trace tree (i.e.the first n spans with the smallest\n * start_timestamp).\n */\n add(span) {\n if (this.spans.length > this._maxlen) {\n // eslint-disable-next-line deprecation/deprecation\n span.spanRecorder = undefined;\n } else {\n this.spans.push(span);\n }\n }\n}\n\n/**\n * Span contains all data about a span\n */\nclass Span {\n /**\n * Tags for the span.\n * @deprecated Use `spanToJSON(span).atttributes` instead.\n */\n\n /**\n * Data for the span.\n * @deprecated Use `spanToJSON(span).atttributes` instead.\n */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n\n /**\n * List of spans that were finalized\n *\n * @deprecated This property will no longer be public. Span recording will be handled internally.\n */\n\n /**\n * @inheritDoc\n * @deprecated Use top level `Sentry.getRootSpan()` instead\n */\n\n /**\n * The instrumenter that created this span.\n *\n * TODO (v8): This can probably be replaced by an `instanceOf` check of the span class.\n * the instrumenter can only be sentry or otel so we can check the span instance\n * to verify which one it is and remove this field entirely.\n *\n * @deprecated This field will be removed.\n */\n\n /** Epoch timestamp in seconds when the span started. */\n\n /** Epoch timestamp in seconds when the span ended. */\n\n /** Internal keeper of the status */\n\n /**\n * You should never call the constructor manually, always use `Sentry.startTransaction()`\n * or call `startChild()` on an existing span.\n * @internal\n * @hideconstructor\n * @hidden\n */\n constructor(spanContext = {}) {\n this._traceId = spanContext.traceId || uuid4();\n this._spanId = spanContext.spanId || uuid4().substring(16);\n this._startTime = spanContext.startTimestamp || timestampInSeconds();\n // eslint-disable-next-line deprecation/deprecation\n this.tags = spanContext.tags ? { ...spanContext.tags } : {};\n // eslint-disable-next-line deprecation/deprecation\n this.data = spanContext.data ? { ...spanContext.data } : {};\n // eslint-disable-next-line deprecation/deprecation\n this.instrumenter = spanContext.instrumenter || 'sentry';\n\n this._attributes = {};\n this.setAttributes({\n [SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN]: spanContext.origin || 'manual',\n [SEMANTIC_ATTRIBUTE_SENTRY_OP]: spanContext.op,\n ...spanContext.attributes,\n });\n\n // eslint-disable-next-line deprecation/deprecation\n this._name = spanContext.name || spanContext.description;\n\n if (spanContext.parentSpanId) {\n this._parentSpanId = spanContext.parentSpanId;\n }\n // We want to include booleans as well here\n if ('sampled' in spanContext) {\n this._sampled = spanContext.sampled;\n }\n if (spanContext.status) {\n this._status = spanContext.status;\n }\n if (spanContext.endTimestamp) {\n this._endTime = spanContext.endTimestamp;\n }\n if (spanContext.exclusiveTime !== undefined) {\n this._exclusiveTime = spanContext.exclusiveTime;\n }\n this._measurements = spanContext.measurements ? { ...spanContext.measurements } : {};\n }\n\n // This rule conflicts with another eslint rule :(\n /* eslint-disable @typescript-eslint/member-ordering */\n\n /**\n * An alias for `description` of the Span.\n * @deprecated Use `spanToJSON(span).description` instead.\n */\n get name() {\n return this._name || '';\n }\n\n /**\n * Update the name of the span.\n * @deprecated Use `spanToJSON(span).description` instead.\n */\n set name(name) {\n this.updateName(name);\n }\n\n /**\n * Get the description of the Span.\n * @deprecated Use `spanToJSON(span).description` instead.\n */\n get description() {\n return this._name;\n }\n\n /**\n * Get the description of the Span.\n * @deprecated Use `spanToJSON(span).description` instead.\n */\n set description(description) {\n this._name = description;\n }\n\n /**\n * The ID of the trace.\n * @deprecated Use `spanContext().traceId` instead.\n */\n get traceId() {\n return this._traceId;\n }\n\n /**\n * The ID of the trace.\n * @deprecated You cannot update the traceId of a span after span creation.\n */\n set traceId(traceId) {\n this._traceId = traceId;\n }\n\n /**\n * The ID of the span.\n * @deprecated Use `spanContext().spanId` instead.\n */\n get spanId() {\n return this._spanId;\n }\n\n /**\n * The ID of the span.\n * @deprecated You cannot update the spanId of a span after span creation.\n */\n set spanId(spanId) {\n this._spanId = spanId;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `startSpan` functions instead.\n */\n set parentSpanId(string) {\n this._parentSpanId = string;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `spanToJSON(span).parent_span_id` instead.\n */\n get parentSpanId() {\n return this._parentSpanId;\n }\n\n /**\n * Was this span chosen to be sent as part of the sample?\n * @deprecated Use `isRecording()` instead.\n */\n get sampled() {\n return this._sampled;\n }\n\n /**\n * Was this span chosen to be sent as part of the sample?\n * @deprecated You cannot update the sampling decision of a span after span creation.\n */\n set sampled(sampled) {\n this._sampled = sampled;\n }\n\n /**\n * Attributes for the span.\n * @deprecated Use `spanToJSON(span).atttributes` instead.\n */\n get attributes() {\n return this._attributes;\n }\n\n /**\n * Attributes for the span.\n * @deprecated Use `setAttributes()` instead.\n */\n set attributes(attributes) {\n this._attributes = attributes;\n }\n\n /**\n * Timestamp in seconds (epoch time) indicating when the span started.\n * @deprecated Use `spanToJSON()` instead.\n */\n get startTimestamp() {\n return this._startTime;\n }\n\n /**\n * Timestamp in seconds (epoch time) indicating when the span started.\n * @deprecated In v8, you will not be able to update the span start time after creation.\n */\n set startTimestamp(startTime) {\n this._startTime = startTime;\n }\n\n /**\n * Timestamp in seconds when the span ended.\n * @deprecated Use `spanToJSON()` instead.\n */\n get endTimestamp() {\n return this._endTime;\n }\n\n /**\n * Timestamp in seconds when the span ended.\n * @deprecated Set the end time via `span.end()` instead.\n */\n set endTimestamp(endTime) {\n this._endTime = endTime;\n }\n\n /**\n * The status of the span.\n *\n * @deprecated Use `spanToJSON().status` instead to get the status.\n */\n get status() {\n return this._status;\n }\n\n /**\n * The status of the span.\n *\n * @deprecated Use `.setStatus()` instead to set or update the status.\n */\n set status(status) {\n this._status = status;\n }\n\n /**\n * Operation of the span\n *\n * @deprecated Use `spanToJSON().op` to read the op instead.\n */\n get op() {\n return this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_OP] ;\n }\n\n /**\n * Operation of the span\n *\n * @deprecated Use `startSpan()` functions to set or `span.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_OP, 'op')\n * to update the span instead.\n */\n set op(op) {\n this.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_OP, op);\n }\n\n /**\n * The origin of the span, giving context about what created the span.\n *\n * @deprecated Use `spanToJSON().origin` to read the origin instead.\n */\n get origin() {\n return this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN] ;\n }\n\n /**\n * The origin of the span, giving context about what created the span.\n *\n * @deprecated Use `startSpan()` functions to set the origin instead.\n */\n set origin(origin) {\n this.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, origin);\n }\n\n /* eslint-enable @typescript-eslint/member-ordering */\n\n /** @inheritdoc */\n spanContext() {\n const { _spanId: spanId, _traceId: traceId, _sampled: sampled } = this;\n return {\n spanId,\n traceId,\n traceFlags: sampled ? TRACE_FLAG_SAMPLED : TRACE_FLAG_NONE,\n };\n }\n\n /**\n * Creates a new `Span` while setting the current `Span.id` as `parentSpanId`.\n * Also the `sampled` decision will be inherited.\n *\n * @deprecated Use `startSpan()`, `startSpanManual()` or `startInactiveSpan()` instead.\n */\n startChild(\n spanContext,\n ) {\n const childSpan = new Span({\n ...spanContext,\n parentSpanId: this._spanId,\n sampled: this._sampled,\n traceId: this._traceId,\n });\n\n // eslint-disable-next-line deprecation/deprecation\n childSpan.spanRecorder = this.spanRecorder;\n // eslint-disable-next-line deprecation/deprecation\n if (childSpan.spanRecorder) {\n // eslint-disable-next-line deprecation/deprecation\n childSpan.spanRecorder.add(childSpan);\n }\n\n const rootSpan = getRootSpan(this);\n // TODO: still set span.transaction here until we have a more permanent solution\n // Probably similarly to the weakmap we hold in node-experimental\n // eslint-disable-next-line deprecation/deprecation\n childSpan.transaction = rootSpan ;\n\n if (DEBUG_BUILD && rootSpan) {\n const opStr = (spanContext && spanContext.op) || '< unknown op >';\n const nameStr = spanToJSON(childSpan).description || '< unknown name >';\n const idStr = rootSpan.spanContext().spanId;\n\n const logMessage = `[Tracing] Starting '${opStr}' span on transaction '${nameStr}' (${idStr}).`;\n logger.log(logMessage);\n this._logMessage = logMessage;\n }\n\n return childSpan;\n }\n\n /**\n * Sets the tag attribute on the current span.\n *\n * Can also be used to unset a tag, by passing `undefined`.\n *\n * @param key Tag key\n * @param value Tag value\n * @deprecated Use `setAttribute()` instead.\n */\n setTag(key, value) {\n // eslint-disable-next-line deprecation/deprecation\n this.tags = { ...this.tags, [key]: value };\n return this;\n }\n\n /**\n * Sets the data attribute on the current span\n * @param key Data key\n * @param value Data value\n * @deprecated Use `setAttribute()` instead.\n */\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n setData(key, value) {\n // eslint-disable-next-line deprecation/deprecation\n this.data = { ...this.data, [key]: value };\n return this;\n }\n\n /** @inheritdoc */\n setAttribute(key, value) {\n if (value === undefined) {\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete this._attributes[key];\n } else {\n this._attributes[key] = value;\n }\n }\n\n /** @inheritdoc */\n setAttributes(attributes) {\n Object.keys(attributes).forEach(key => this.setAttribute(key, attributes[key]));\n }\n\n /**\n * @inheritDoc\n */\n setStatus(value) {\n this._status = value;\n return this;\n }\n\n /**\n * @inheritDoc\n * @deprecated Use top-level `setHttpStatus()` instead.\n */\n setHttpStatus(httpStatus) {\n setHttpStatus(this, httpStatus);\n return this;\n }\n\n /**\n * @inheritdoc\n *\n * @deprecated Use `.updateName()` instead.\n */\n setName(name) {\n this.updateName(name);\n }\n\n /**\n * @inheritDoc\n */\n updateName(name) {\n this._name = name;\n return this;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `spanToJSON(span).status === 'ok'` instead.\n */\n isSuccess() {\n return this._status === 'ok';\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `.end()` instead.\n */\n finish(endTimestamp) {\n return this.end(endTimestamp);\n }\n\n /** @inheritdoc */\n end(endTimestamp) {\n // If already ended, skip\n if (this._endTime) {\n return;\n }\n const rootSpan = getRootSpan(this);\n if (\n DEBUG_BUILD &&\n // Don't call this for transactions\n rootSpan &&\n rootSpan.spanContext().spanId !== this._spanId\n ) {\n const logMessage = this._logMessage;\n if (logMessage) {\n logger.log((logMessage ).replace('Starting', 'Finishing'));\n }\n }\n\n this._endTime = spanTimeInputToSeconds(endTimestamp);\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `spanToTraceHeader()` instead.\n */\n toTraceparent() {\n return spanToTraceHeader(this);\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `spanToJSON()` or access the fields directly instead.\n */\n toContext() {\n return dropUndefinedKeys({\n data: this._getData(),\n description: this._name,\n endTimestamp: this._endTime,\n // eslint-disable-next-line deprecation/deprecation\n op: this.op,\n parentSpanId: this._parentSpanId,\n sampled: this._sampled,\n spanId: this._spanId,\n startTimestamp: this._startTime,\n status: this._status,\n // eslint-disable-next-line deprecation/deprecation\n tags: this.tags,\n traceId: this._traceId,\n });\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Update the fields directly instead.\n */\n updateWithContext(spanContext) {\n // eslint-disable-next-line deprecation/deprecation\n this.data = spanContext.data || {};\n // eslint-disable-next-line deprecation/deprecation\n this._name = spanContext.name || spanContext.description;\n this._endTime = spanContext.endTimestamp;\n // eslint-disable-next-line deprecation/deprecation\n this.op = spanContext.op;\n this._parentSpanId = spanContext.parentSpanId;\n this._sampled = spanContext.sampled;\n this._spanId = spanContext.spanId || this._spanId;\n this._startTime = spanContext.startTimestamp || this._startTime;\n this._status = spanContext.status;\n // eslint-disable-next-line deprecation/deprecation\n this.tags = spanContext.tags || {};\n this._traceId = spanContext.traceId || this._traceId;\n\n return this;\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use `spanToTraceContext()` util function instead.\n */\n getTraceContext() {\n return spanToTraceContext(this);\n }\n\n /**\n * Get JSON representation of this span.\n *\n * @hidden\n * @internal This method is purely for internal purposes and should not be used outside\n * of SDK code. If you need to get a JSON representation of a span,\n * use `spanToJSON(span)` instead.\n */\n getSpanJSON() {\n return dropUndefinedKeys({\n data: this._getData(),\n description: this._name,\n op: this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_OP] ,\n parent_span_id: this._parentSpanId,\n span_id: this._spanId,\n start_timestamp: this._startTime,\n status: this._status,\n // eslint-disable-next-line deprecation/deprecation\n tags: Object.keys(this.tags).length > 0 ? this.tags : undefined,\n timestamp: this._endTime,\n trace_id: this._traceId,\n origin: this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN] ,\n _metrics_summary: getMetricSummaryJsonForSpan(this),\n profile_id: this._attributes[SEMANTIC_ATTRIBUTE_PROFILE_ID] ,\n exclusive_time: this._exclusiveTime,\n measurements: Object.keys(this._measurements).length > 0 ? this._measurements : undefined,\n });\n }\n\n /** @inheritdoc */\n isRecording() {\n return !this._endTime && !!this._sampled;\n }\n\n /**\n * Convert the object to JSON.\n * @deprecated Use `spanToJSON(span)` instead.\n */\n toJSON() {\n return this.getSpanJSON();\n }\n\n /**\n * Get the merged data for this span.\n * For now, this combines `data` and `attributes` together,\n * until eventually we can ingest `attributes` directly.\n */\n _getData()\n\n {\n // eslint-disable-next-line deprecation/deprecation\n const { data, _attributes: attributes } = this;\n\n const hasData = Object.keys(data).length > 0;\n const hasAttributes = Object.keys(attributes).length > 0;\n\n if (!hasData && !hasAttributes) {\n return undefined;\n }\n\n if (hasData && hasAttributes) {\n return {\n ...data,\n ...attributes,\n };\n }\n\n return hasData ? data : attributes;\n }\n}\n\nexport { Span, SpanRecorder };\n","import { getClient } from '../exports.js';\n\n// Treeshakable guard to remove all code related to tracing\n\n/**\n * Determines if tracing is currently enabled.\n *\n * Tracing is enabled when at least one of `tracesSampleRate` and `tracesSampler` is defined in the SDK config.\n */\nfunction hasTracingEnabled(\n maybeOptions,\n) {\n if (typeof __SENTRY_TRACING__ === 'boolean' && !__SENTRY_TRACING__) {\n return false;\n }\n\n const client = getClient();\n const options = maybeOptions || (client && client.getOptions());\n return !!options && (options.enableTracing || 'tracesSampleRate' in options || 'tracesSampler' in options);\n}\n\nexport { hasTracingEnabled };\n","import { tracingContextFromHeaders, logger, dropUndefinedKeys, addNonEnumerableProperty } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { getCurrentHub, runWithAsyncContext, getIsolationScope } from '../hub.js';\nimport { spanToJSON, spanIsSampled, spanTimeInputToSeconds } from '../utils/spanUtils.js';\nimport './errors.js';\nimport './spanstatus.js';\nimport { getDynamicSamplingContextFromSpan } from './dynamicSamplingContext.js';\nimport { getCurrentScope, withScope } from '../exports.js';\nimport { handleCallbackErrors } from '../utils/handleCallbackErrors.js';\nimport { hasTracingEnabled } from '../utils/hasTracingEnabled.js';\n\n/**\n * Wraps a function with a transaction/span and finishes the span after the function is done.\n *\n * Note that if you have not enabled tracing extensions via `addTracingExtensions`\n * or you didn't set `tracesSampleRate`, this function will not generate spans\n * and the `span` returned from the callback will be undefined.\n *\n * This function is meant to be used internally and may break at any time. Use at your own risk.\n *\n * @internal\n * @private\n *\n * @deprecated Use `startSpan` instead.\n */\nfunction trace(\n context,\n callback,\n // eslint-disable-next-line @typescript-eslint/no-empty-function\n onError = () => {},\n // eslint-disable-next-line @typescript-eslint/no-empty-function\n afterFinish = () => {},\n) {\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n const scope = getCurrentScope();\n // eslint-disable-next-line deprecation/deprecation\n const parentSpan = scope.getSpan();\n\n const spanContext = normalizeContext(context);\n const activeSpan = createChildSpanOrTransaction(hub, {\n parentSpan,\n spanContext,\n forceTransaction: false,\n scope,\n });\n\n // eslint-disable-next-line deprecation/deprecation\n scope.setSpan(activeSpan);\n\n return handleCallbackErrors(\n () => callback(activeSpan),\n error => {\n activeSpan && activeSpan.setStatus('internal_error');\n onError(error, activeSpan);\n },\n () => {\n activeSpan && activeSpan.end();\n // eslint-disable-next-line deprecation/deprecation\n scope.setSpan(parentSpan);\n afterFinish();\n },\n );\n}\n\n/**\n * Wraps a function with a transaction/span and finishes the span after the function is done.\n * The created span is the active span and will be used as parent by other spans created inside the function\n * and can be accessed via `Sentry.getSpan()`, as long as the function is executed while the scope is active.\n *\n * If you want to create a span that is not set as active, use {@link startInactiveSpan}.\n *\n * Note that if you have not enabled tracing extensions via `addTracingExtensions`\n * or you didn't set `tracesSampleRate`, this function will not generate spans\n * and the `span` returned from the callback will be undefined.\n */\nfunction startSpan(context, callback) {\n const spanContext = normalizeContext(context);\n\n return runWithAsyncContext(() => {\n return withScope(context.scope, scope => {\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n // eslint-disable-next-line deprecation/deprecation\n const parentSpan = scope.getSpan();\n\n const shouldSkipSpan = context.onlyIfParent && !parentSpan;\n const activeSpan = shouldSkipSpan\n ? undefined\n : createChildSpanOrTransaction(hub, {\n parentSpan,\n spanContext,\n forceTransaction: context.forceTransaction,\n scope,\n });\n\n return handleCallbackErrors(\n () => callback(activeSpan),\n () => {\n // Only update the span status if it hasn't been changed yet\n if (activeSpan) {\n const { status } = spanToJSON(activeSpan);\n if (!status || status === 'ok') {\n activeSpan.setStatus('internal_error');\n }\n }\n },\n () => activeSpan && activeSpan.end(),\n );\n });\n });\n}\n\n/**\n * @deprecated Use {@link startSpan} instead.\n */\nconst startActiveSpan = startSpan;\n\n/**\n * Similar to `Sentry.startSpan`. Wraps a function with a transaction/span, but does not finish the span\n * after the function is done automatically. You'll have to call `span.end()` manually.\n *\n * The created span is the active span and will be used as parent by other spans created inside the function\n * and can be accessed via `Sentry.getActiveSpan()`, as long as the function is executed while the scope is active.\n *\n * Note that if you have not enabled tracing extensions via `addTracingExtensions`\n * or you didn't set `tracesSampleRate`, this function will not generate spans\n * and the `span` returned from the callback will be undefined.\n */\nfunction startSpanManual(\n context,\n callback,\n) {\n const spanContext = normalizeContext(context);\n\n return runWithAsyncContext(() => {\n return withScope(context.scope, scope => {\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n // eslint-disable-next-line deprecation/deprecation\n const parentSpan = scope.getSpan();\n\n const shouldSkipSpan = context.onlyIfParent && !parentSpan;\n const activeSpan = shouldSkipSpan\n ? undefined\n : createChildSpanOrTransaction(hub, {\n parentSpan,\n spanContext,\n forceTransaction: context.forceTransaction,\n scope,\n });\n\n function finishAndSetSpan() {\n activeSpan && activeSpan.end();\n }\n\n return handleCallbackErrors(\n () => callback(activeSpan, finishAndSetSpan),\n () => {\n // Only update the span status if it hasn't been changed yet, and the span is not yet finished\n if (activeSpan && activeSpan.isRecording()) {\n const { status } = spanToJSON(activeSpan);\n if (!status || status === 'ok') {\n activeSpan.setStatus('internal_error');\n }\n }\n },\n );\n });\n });\n}\n\n/**\n * Creates a span. This span is not set as active, so will not get automatic instrumentation spans\n * as children or be able to be accessed via `Sentry.getSpan()`.\n *\n * If you want to create a span that is set as active, use {@link startSpan}.\n *\n * Note that if you have not enabled tracing extensions via `addTracingExtensions`\n * or you didn't set `tracesSampleRate` or `tracesSampler`, this function will not generate spans\n * and the `span` returned from the callback will be undefined.\n */\nfunction startInactiveSpan(context) {\n if (!hasTracingEnabled()) {\n return undefined;\n }\n\n const spanContext = normalizeContext(context);\n // eslint-disable-next-line deprecation/deprecation\n const hub = getCurrentHub();\n const parentSpan = context.scope\n ? // eslint-disable-next-line deprecation/deprecation\n context.scope.getSpan()\n : getActiveSpan();\n\n const shouldSkipSpan = context.onlyIfParent && !parentSpan;\n\n if (shouldSkipSpan) {\n return undefined;\n }\n\n const scope = context.scope || getCurrentScope();\n\n // Even though we don't actually want to make this span active on the current scope,\n // we need to make it active on a temporary scope that we use for event processing\n // as otherwise, it won't pick the correct span for the event when processing it\n const temporaryScope = (scope ).clone();\n\n return createChildSpanOrTransaction(hub, {\n parentSpan,\n spanContext,\n forceTransaction: context.forceTransaction,\n scope: temporaryScope,\n });\n}\n\n/**\n * Returns the currently active span.\n */\nfunction getActiveSpan() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentScope().getSpan();\n}\n\nconst continueTrace = (\n {\n sentryTrace,\n baggage,\n }\n\n,\n callback,\n) => {\n // TODO(v8): Change this function so it doesn't do anything besides setting the propagation context on the current scope:\n /*\n return withScope((scope) => {\n const propagationContext = propagationContextFromHeaders(sentryTrace, baggage);\n scope.setPropagationContext(propagationContext);\n return callback();\n })\n */\n\n const currentScope = getCurrentScope();\n\n // eslint-disable-next-line deprecation/deprecation\n const { traceparentData, dynamicSamplingContext, propagationContext } = tracingContextFromHeaders(\n sentryTrace,\n baggage,\n );\n\n currentScope.setPropagationContext(propagationContext);\n\n if (DEBUG_BUILD && traceparentData) {\n logger.log(`[Tracing] Continuing trace ${traceparentData.traceId}.`);\n }\n\n const transactionContext = {\n ...traceparentData,\n metadata: dropUndefinedKeys({\n dynamicSamplingContext,\n }),\n };\n\n if (!callback) {\n return transactionContext;\n }\n\n return runWithAsyncContext(() => {\n return callback(transactionContext);\n });\n};\n\nfunction createChildSpanOrTransaction(\n // eslint-disable-next-line deprecation/deprecation\n hub,\n {\n parentSpan,\n spanContext,\n forceTransaction,\n scope,\n }\n\n,\n) {\n if (!hasTracingEnabled()) {\n return undefined;\n }\n\n const isolationScope = getIsolationScope();\n\n let span;\n if (parentSpan && !forceTransaction) {\n // eslint-disable-next-line deprecation/deprecation\n span = parentSpan.startChild(spanContext);\n } else if (parentSpan) {\n // If we forced a transaction but have a parent span, make sure to continue from the parent span, not the scope\n const dsc = getDynamicSamplingContextFromSpan(parentSpan);\n const { traceId, spanId: parentSpanId } = parentSpan.spanContext();\n const sampled = spanIsSampled(parentSpan);\n\n // eslint-disable-next-line deprecation/deprecation\n span = hub.startTransaction({\n traceId,\n parentSpanId,\n parentSampled: sampled,\n ...spanContext,\n metadata: {\n dynamicSamplingContext: dsc,\n // eslint-disable-next-line deprecation/deprecation\n ...spanContext.metadata,\n },\n });\n } else {\n const { traceId, dsc, parentSpanId, sampled } = {\n ...isolationScope.getPropagationContext(),\n ...scope.getPropagationContext(),\n };\n\n // eslint-disable-next-line deprecation/deprecation\n span = hub.startTransaction({\n traceId,\n parentSpanId,\n parentSampled: sampled,\n ...spanContext,\n metadata: {\n dynamicSamplingContext: dsc,\n // eslint-disable-next-line deprecation/deprecation\n ...spanContext.metadata,\n },\n });\n }\n\n // We always set this as active span on the scope\n // In the case of this being an inactive span, we ensure to pass a detached scope in here in the first place\n // But by having this here, we can ensure that the lookup through `getCapturedScopesOnSpan` results in the correct scope & span combo\n // eslint-disable-next-line deprecation/deprecation\n scope.setSpan(span);\n\n setCapturedScopesOnSpan(span, scope, isolationScope);\n\n return span;\n}\n\n/**\n * This converts StartSpanOptions to TransactionContext.\n * For the most part (for now) we accept the same options,\n * but some of them need to be transformed.\n *\n * Eventually the StartSpanOptions will be more aligned with OpenTelemetry.\n */\nfunction normalizeContext(context) {\n if (context.startTime) {\n const ctx = { ...context };\n ctx.startTimestamp = spanTimeInputToSeconds(context.startTime);\n delete ctx.startTime;\n return ctx;\n }\n\n return context;\n}\n\nconst SCOPE_ON_START_SPAN_FIELD = '_sentryScope';\nconst ISOLATION_SCOPE_ON_START_SPAN_FIELD = '_sentryIsolationScope';\n\nfunction setCapturedScopesOnSpan(span, scope, isolationScope) {\n if (span) {\n addNonEnumerableProperty(span, ISOLATION_SCOPE_ON_START_SPAN_FIELD, isolationScope);\n addNonEnumerableProperty(span, SCOPE_ON_START_SPAN_FIELD, scope);\n }\n}\n\n/**\n * Grabs the scope and isolation scope off a span that were active when the span was started.\n */\nfunction getCapturedScopesOnSpan(span) {\n return {\n scope: (span )[SCOPE_ON_START_SPAN_FIELD],\n isolationScope: (span )[ISOLATION_SCOPE_ON_START_SPAN_FIELD],\n };\n}\n\nexport { continueTrace, getActiveSpan, getCapturedScopesOnSpan, startActiveSpan, startInactiveSpan, startSpan, startSpanManual, trace };\n","import { dropUndefinedKeys, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { getCurrentHub } from '../hub.js';\nimport { getMetricSummaryJsonForSpan } from '../metrics/metric-summary.js';\nimport { SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE } from '../semanticAttributes.js';\nimport { spanTimeInputToSeconds, spanToJSON, spanToTraceContext } from '../utils/spanUtils.js';\nimport { getDynamicSamplingContextFromSpan } from './dynamicSamplingContext.js';\nimport { Span, SpanRecorder } from './span.js';\nimport { getCapturedScopesOnSpan } from './trace.js';\n\n/** JSDoc */\nclass Transaction extends Span {\n /**\n * The reference to the current hub.\n */\n // eslint-disable-next-line deprecation/deprecation\n\n // DO NOT yet remove this property, it is used in a hack for v7 backwards compatibility.\n\n /**\n * This constructor should never be called manually. Those instrumenting tracing should use\n * `Sentry.startTransaction()`, and internal methods should use `hub.startTransaction()`.\n * @internal\n * @hideconstructor\n * @hidden\n *\n * @deprecated Transactions will be removed in v8. Use spans instead.\n */\n // eslint-disable-next-line deprecation/deprecation\n constructor(transactionContext, hub) {\n super(transactionContext);\n this._contexts = {};\n\n // eslint-disable-next-line deprecation/deprecation\n this._hub = hub || getCurrentHub();\n\n this._name = transactionContext.name || '';\n\n this._metadata = {\n // eslint-disable-next-line deprecation/deprecation\n ...transactionContext.metadata,\n };\n\n this._trimEnd = transactionContext.trimEnd;\n\n // this is because transactions are also spans, and spans have a transaction pointer\n // TODO (v8): Replace this with another way to set the root span\n // eslint-disable-next-line deprecation/deprecation\n this.transaction = this;\n\n // If Dynamic Sampling Context is provided during the creation of the transaction, we freeze it as it usually means\n // there is incoming Dynamic Sampling Context. (Either through an incoming request, a baggage meta-tag, or other means)\n const incomingDynamicSamplingContext = this._metadata.dynamicSamplingContext;\n if (incomingDynamicSamplingContext) {\n // We shallow copy this in case anything writes to the original reference of the passed in `dynamicSamplingContext`\n this._frozenDynamicSamplingContext = { ...incomingDynamicSamplingContext };\n }\n }\n\n // This sadly conflicts with the getter/setter ordering :(\n /* eslint-disable @typescript-eslint/member-ordering */\n\n /**\n * Getter for `name` property.\n * @deprecated Use `spanToJSON(span).description` instead.\n */\n get name() {\n return this._name;\n }\n\n /**\n * Setter for `name` property, which also sets `source` as custom.\n * @deprecated Use `updateName()` and `setMetadata()` instead.\n */\n set name(newName) {\n // eslint-disable-next-line deprecation/deprecation\n this.setName(newName);\n }\n\n /**\n * Get the metadata for this transaction.\n * @deprecated Use `spanGetMetadata(transaction)` instead.\n */\n get metadata() {\n // We merge attributes in for backwards compatibility\n return {\n // Defaults\n // eslint-disable-next-line deprecation/deprecation\n source: 'custom',\n spanMetadata: {},\n\n // Legacy metadata\n ...this._metadata,\n\n // From attributes\n ...(this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE] && {\n source: this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE] ,\n }),\n ...(this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE] && {\n sampleRate: this._attributes[SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE] ,\n }),\n };\n }\n\n /**\n * Update the metadata for this transaction.\n * @deprecated Use `spanGetMetadata(transaction)` instead.\n */\n set metadata(metadata) {\n this._metadata = metadata;\n }\n\n /* eslint-enable @typescript-eslint/member-ordering */\n\n /**\n * Setter for `name` property, which also sets `source` on the metadata.\n *\n * @deprecated Use `.updateName()` and `.setAttribute()` instead.\n */\n setName(name, source = 'custom') {\n this._name = name;\n this.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, source);\n }\n\n /** @inheritdoc */\n updateName(name) {\n this._name = name;\n return this;\n }\n\n /**\n * Attaches SpanRecorder to the span itself\n * @param maxlen maximum number of spans that can be recorded\n */\n initSpanRecorder(maxlen = 1000) {\n // eslint-disable-next-line deprecation/deprecation\n if (!this.spanRecorder) {\n // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder = new SpanRecorder(maxlen);\n }\n // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder.add(this);\n }\n\n /**\n * Set the context of a transaction event.\n * @deprecated Use either `.setAttribute()`, or set the context on the scope before creating the transaction.\n */\n setContext(key, context) {\n if (context === null) {\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete this._contexts[key];\n } else {\n this._contexts[key] = context;\n }\n }\n\n /**\n * @inheritDoc\n *\n * @deprecated Use top-level `setMeasurement()` instead.\n */\n setMeasurement(name, value, unit = '') {\n this._measurements[name] = { value, unit };\n }\n\n /**\n * Store metadata on this transaction.\n * @deprecated Use attributes or store data on the scope instead.\n */\n setMetadata(newMetadata) {\n this._metadata = { ...this._metadata, ...newMetadata };\n }\n\n /**\n * @inheritDoc\n */\n end(endTimestamp) {\n const timestampInS = spanTimeInputToSeconds(endTimestamp);\n const transaction = this._finishTransaction(timestampInS);\n if (!transaction) {\n return undefined;\n }\n // eslint-disable-next-line deprecation/deprecation\n return this._hub.captureEvent(transaction);\n }\n\n /**\n * @inheritDoc\n */\n toContext() {\n // eslint-disable-next-line deprecation/deprecation\n const spanContext = super.toContext();\n\n return dropUndefinedKeys({\n ...spanContext,\n name: this._name,\n trimEnd: this._trimEnd,\n });\n }\n\n /**\n * @inheritDoc\n */\n updateWithContext(transactionContext) {\n // eslint-disable-next-line deprecation/deprecation\n super.updateWithContext(transactionContext);\n\n this._name = transactionContext.name || '';\n this._trimEnd = transactionContext.trimEnd;\n\n return this;\n }\n\n /**\n * @inheritdoc\n *\n * @experimental\n *\n * @deprecated Use top-level `getDynamicSamplingContextFromSpan` instead.\n */\n getDynamicSamplingContext() {\n return getDynamicSamplingContextFromSpan(this);\n }\n\n /**\n * Override the current hub with a new one.\n * Used if you want another hub to finish the transaction.\n *\n * @internal\n */\n // eslint-disable-next-line deprecation/deprecation\n setHub(hub) {\n this._hub = hub;\n }\n\n /**\n * Get the profile id of the transaction.\n */\n getProfileId() {\n if (this._contexts !== undefined && this._contexts['profile'] !== undefined) {\n return this._contexts['profile'].profile_id ;\n }\n return undefined;\n }\n\n /**\n * Finish the transaction & prepare the event to send to Sentry.\n */\n _finishTransaction(endTimestamp) {\n // This transaction is already finished, so we should not flush it again.\n if (this._endTime !== undefined) {\n return undefined;\n }\n\n if (!this._name) {\n DEBUG_BUILD && logger.warn('Transaction has no name, falling back to ``.');\n this._name = '';\n }\n\n // just sets the end timestamp\n super.end(endTimestamp);\n\n // eslint-disable-next-line deprecation/deprecation\n const client = this._hub.getClient();\n if (client && client.emit) {\n client.emit('finishTransaction', this);\n }\n\n if (this._sampled !== true) {\n // At this point if `sampled !== true` we want to discard the transaction.\n DEBUG_BUILD && logger.log('[Tracing] Discarding transaction because its trace was not chosen to be sampled.');\n\n if (client) {\n client.recordDroppedEvent('sample_rate', 'transaction');\n }\n\n return undefined;\n }\n\n // eslint-disable-next-line deprecation/deprecation\n const finishedSpans = this.spanRecorder\n ? // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder.spans.filter(span => span !== this && spanToJSON(span).timestamp)\n : [];\n\n if (this._trimEnd && finishedSpans.length > 0) {\n const endTimes = finishedSpans.map(span => spanToJSON(span).timestamp).filter(Boolean) ;\n this._endTime = endTimes.reduce((prev, current) => {\n return prev > current ? prev : current;\n });\n }\n\n const { scope: capturedSpanScope, isolationScope: capturedSpanIsolationScope } = getCapturedScopesOnSpan(this);\n\n // eslint-disable-next-line deprecation/deprecation\n const { metadata } = this;\n // eslint-disable-next-line deprecation/deprecation\n const { source } = metadata;\n\n const transaction = {\n contexts: {\n ...this._contexts,\n // We don't want to override trace context\n trace: spanToTraceContext(this),\n },\n // TODO: Pass spans serialized via `spanToJSON()` here instead in v8.\n spans: finishedSpans,\n start_timestamp: this._startTime,\n // eslint-disable-next-line deprecation/deprecation\n tags: this.tags,\n timestamp: this._endTime,\n transaction: this._name,\n type: 'transaction',\n sdkProcessingMetadata: {\n ...metadata,\n capturedSpanScope,\n capturedSpanIsolationScope,\n ...dropUndefinedKeys({\n dynamicSamplingContext: getDynamicSamplingContextFromSpan(this),\n }),\n },\n _metrics_summary: getMetricSummaryJsonForSpan(this),\n ...(source && {\n transaction_info: {\n source,\n },\n }),\n };\n\n const hasMeasurements = Object.keys(this._measurements).length > 0;\n\n if (hasMeasurements) {\n DEBUG_BUILD &&\n logger.log(\n '[Measurements] Adding measurements to transaction',\n JSON.stringify(this._measurements, undefined, 2),\n );\n transaction.measurements = this._measurements;\n }\n\n // eslint-disable-next-line deprecation/deprecation\n DEBUG_BUILD && logger.log(`[Tracing] Finishing ${this.op} transaction: ${this._name}.`);\n\n return transaction;\n }\n}\n\nexport { Transaction };\n","import { logger, timestampInSeconds } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { spanTimeInputToSeconds, spanToJSON } from '../utils/spanUtils.js';\nimport { SpanRecorder } from './span.js';\nimport { Transaction } from './transaction.js';\n\nconst TRACING_DEFAULTS = {\n idleTimeout: 1000,\n finalTimeout: 30000,\n heartbeatInterval: 5000,\n};\n\nconst FINISH_REASON_TAG = 'finishReason';\n\nconst IDLE_TRANSACTION_FINISH_REASONS = [\n 'heartbeatFailed',\n 'idleTimeout',\n 'documentHidden',\n 'finalTimeout',\n 'externalFinish',\n 'cancelled',\n];\n\n/**\n * @inheritDoc\n */\nclass IdleTransactionSpanRecorder extends SpanRecorder {\n constructor(\n _pushActivity,\n _popActivity,\n transactionSpanId,\n maxlen,\n ) {\n super(maxlen);this._pushActivity = _pushActivity;this._popActivity = _popActivity;this.transactionSpanId = transactionSpanId; }\n\n /**\n * @inheritDoc\n */\n add(span) {\n // We should make sure we do not push and pop activities for\n // the transaction that this span recorder belongs to.\n if (span.spanContext().spanId !== this.transactionSpanId) {\n // We patch span.end() to pop an activity after setting an endTimestamp.\n // eslint-disable-next-line @typescript-eslint/unbound-method\n const originalEnd = span.end;\n span.end = (...rest) => {\n this._popActivity(span.spanContext().spanId);\n return originalEnd.apply(span, rest);\n };\n\n // We should only push new activities if the span does not have an end timestamp.\n if (spanToJSON(span).timestamp === undefined) {\n this._pushActivity(span.spanContext().spanId);\n }\n }\n\n super.add(span);\n }\n}\n\n/**\n * An IdleTransaction is a transaction that automatically finishes. It does this by tracking child spans as activities.\n * You can have multiple IdleTransactions active, but if the `onScope` option is specified, the idle transaction will\n * put itself on the scope on creation.\n */\nclass IdleTransaction extends Transaction {\n // Activities store a list of active spans\n\n // Track state of activities in previous heartbeat\n\n // Amount of times heartbeat has counted. Will cause transaction to finish after 3 beats.\n\n // We should not use heartbeat if we finished a transaction\n\n // Idle timeout was canceled and we should finish the transaction with the last span end.\n\n /**\n * Timer that tracks Transaction idleTimeout\n */\n\n /**\n * @deprecated Transactions will be removed in v8. Use spans instead.\n */\n constructor(\n transactionContext,\n // eslint-disable-next-line deprecation/deprecation\n _idleHub,\n /**\n * The time to wait in ms until the idle transaction will be finished. This timer is started each time\n * there are no active spans on this transaction.\n */\n _idleTimeout = TRACING_DEFAULTS.idleTimeout,\n /**\n * The final value in ms that a transaction cannot exceed\n */\n _finalTimeout = TRACING_DEFAULTS.finalTimeout,\n _heartbeatInterval = TRACING_DEFAULTS.heartbeatInterval,\n // Whether or not the transaction should put itself on the scope when it starts and pop itself off when it ends\n _onScope = false,\n /**\n * When set to `true`, will disable the idle timeout (`_idleTimeout` option) and heartbeat mechanisms (`_heartbeatInterval`\n * option) until the `sendAutoFinishSignal()` method is called. The final timeout mechanism (`_finalTimeout` option)\n * will not be affected by this option, meaning the transaction will definitely be finished when the final timeout is\n * reached, no matter what this option is configured to.\n *\n * Defaults to `false`.\n */\n delayAutoFinishUntilSignal = false,\n ) {\n super(transactionContext, _idleHub);this._idleHub = _idleHub;this._idleTimeout = _idleTimeout;this._finalTimeout = _finalTimeout;this._heartbeatInterval = _heartbeatInterval;this._onScope = _onScope;\n this.activities = {};\n this._heartbeatCounter = 0;\n this._finished = false;\n this._idleTimeoutCanceledPermanently = false;\n this._beforeFinishCallbacks = [];\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[4];\n this._autoFinishAllowed = !delayAutoFinishUntilSignal;\n\n if (_onScope) {\n // We set the transaction here on the scope so error events pick up the trace\n // context and attach it to the error.\n DEBUG_BUILD && logger.log(`Setting idle transaction on scope. Span ID: ${this.spanContext().spanId}`);\n // eslint-disable-next-line deprecation/deprecation\n _idleHub.getScope().setSpan(this);\n }\n\n if (!delayAutoFinishUntilSignal) {\n this._restartIdleTimeout();\n }\n\n setTimeout(() => {\n if (!this._finished) {\n this.setStatus('deadline_exceeded');\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[3];\n this.end();\n }\n }, this._finalTimeout);\n }\n\n /** {@inheritDoc} */\n end(endTimestamp) {\n const endTimestampInS = spanTimeInputToSeconds(endTimestamp);\n\n this._finished = true;\n this.activities = {};\n\n // eslint-disable-next-line deprecation/deprecation\n if (this.op === 'ui.action.click') {\n this.setAttribute(FINISH_REASON_TAG, this._finishReason);\n }\n\n // eslint-disable-next-line deprecation/deprecation\n if (this.spanRecorder) {\n DEBUG_BUILD &&\n // eslint-disable-next-line deprecation/deprecation\n logger.log('[Tracing] finishing IdleTransaction', new Date(endTimestampInS * 1000).toISOString(), this.op);\n\n for (const callback of this._beforeFinishCallbacks) {\n callback(this, endTimestampInS);\n }\n\n // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder.spans = this.spanRecorder.spans.filter((span) => {\n // If we are dealing with the transaction itself, we just return it\n if (span.spanContext().spanId === this.spanContext().spanId) {\n return true;\n }\n\n // We cancel all pending spans with status \"cancelled\" to indicate the idle transaction was finished early\n if (!spanToJSON(span).timestamp) {\n span.setStatus('cancelled');\n span.end(endTimestampInS);\n DEBUG_BUILD &&\n logger.log('[Tracing] cancelling span since transaction ended early', JSON.stringify(span, undefined, 2));\n }\n\n const { start_timestamp: startTime, timestamp: endTime } = spanToJSON(span);\n const spanStartedBeforeTransactionFinish = startTime && startTime < endTimestampInS;\n\n // Add a delta with idle timeout so that we prevent false positives\n const timeoutWithMarginOfError = (this._finalTimeout + this._idleTimeout) / 1000;\n const spanEndedBeforeFinalTimeout = endTime && startTime && endTime - startTime < timeoutWithMarginOfError;\n\n if (DEBUG_BUILD) {\n const stringifiedSpan = JSON.stringify(span, undefined, 2);\n if (!spanStartedBeforeTransactionFinish) {\n logger.log('[Tracing] discarding Span since it happened after Transaction was finished', stringifiedSpan);\n } else if (!spanEndedBeforeFinalTimeout) {\n logger.log('[Tracing] discarding Span since it finished after Transaction final timeout', stringifiedSpan);\n }\n }\n\n return spanStartedBeforeTransactionFinish && spanEndedBeforeFinalTimeout;\n });\n\n DEBUG_BUILD && logger.log('[Tracing] flushing IdleTransaction');\n } else {\n DEBUG_BUILD && logger.log('[Tracing] No active IdleTransaction');\n }\n\n // if `this._onScope` is `true`, the transaction put itself on the scope when it started\n if (this._onScope) {\n // eslint-disable-next-line deprecation/deprecation\n const scope = this._idleHub.getScope();\n // eslint-disable-next-line deprecation/deprecation\n if (scope.getTransaction() === this) {\n // eslint-disable-next-line deprecation/deprecation\n scope.setSpan(undefined);\n }\n }\n\n return super.end(endTimestamp);\n }\n\n /**\n * Register a callback function that gets executed before the transaction finishes.\n * Useful for cleanup or if you want to add any additional spans based on current context.\n *\n * This is exposed because users have no other way of running something before an idle transaction\n * finishes.\n */\n registerBeforeFinishCallback(callback) {\n this._beforeFinishCallbacks.push(callback);\n }\n\n /**\n * @inheritDoc\n */\n initSpanRecorder(maxlen) {\n // eslint-disable-next-line deprecation/deprecation\n if (!this.spanRecorder) {\n const pushActivity = (id) => {\n if (this._finished) {\n return;\n }\n this._pushActivity(id);\n };\n const popActivity = (id) => {\n if (this._finished) {\n return;\n }\n this._popActivity(id);\n };\n\n // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder = new IdleTransactionSpanRecorder(pushActivity, popActivity, this.spanContext().spanId, maxlen);\n\n // Start heartbeat so that transactions do not run forever.\n DEBUG_BUILD && logger.log('Starting heartbeat');\n this._pingHeartbeat();\n }\n // eslint-disable-next-line deprecation/deprecation\n this.spanRecorder.add(this);\n }\n\n /**\n * Cancels the existing idle timeout, if there is one.\n * @param restartOnChildSpanChange Default is `true`.\n * If set to false the transaction will end\n * with the last child span.\n */\n cancelIdleTimeout(\n endTimestamp,\n {\n restartOnChildSpanChange,\n }\n\n = {\n restartOnChildSpanChange: true,\n },\n ) {\n this._idleTimeoutCanceledPermanently = restartOnChildSpanChange === false;\n if (this._idleTimeoutID) {\n clearTimeout(this._idleTimeoutID);\n this._idleTimeoutID = undefined;\n\n if (Object.keys(this.activities).length === 0 && this._idleTimeoutCanceledPermanently) {\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[5];\n this.end(endTimestamp);\n }\n }\n }\n\n /**\n * Temporary method used to externally set the transaction's `finishReason`\n *\n * ** WARNING**\n * This is for the purpose of experimentation only and will be removed in the near future, do not use!\n *\n * @internal\n *\n */\n setFinishReason(reason) {\n this._finishReason = reason;\n }\n\n /**\n * Permits the IdleTransaction to automatically end itself via the idle timeout and heartbeat mechanisms when the `delayAutoFinishUntilSignal` option was set to `true`.\n */\n sendAutoFinishSignal() {\n if (!this._autoFinishAllowed) {\n DEBUG_BUILD && logger.log('[Tracing] Received finish signal for idle transaction.');\n this._restartIdleTimeout();\n this._autoFinishAllowed = true;\n }\n }\n\n /**\n * Restarts idle timeout, if there is no running idle timeout it will start one.\n */\n _restartIdleTimeout(endTimestamp) {\n this.cancelIdleTimeout();\n this._idleTimeoutID = setTimeout(() => {\n if (!this._finished && Object.keys(this.activities).length === 0) {\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[1];\n this.end(endTimestamp);\n }\n }, this._idleTimeout);\n }\n\n /**\n * Start tracking a specific activity.\n * @param spanId The span id that represents the activity\n */\n _pushActivity(spanId) {\n this.cancelIdleTimeout(undefined, { restartOnChildSpanChange: !this._idleTimeoutCanceledPermanently });\n DEBUG_BUILD && logger.log(`[Tracing] pushActivity: ${spanId}`);\n this.activities[spanId] = true;\n DEBUG_BUILD && logger.log('[Tracing] new activities count', Object.keys(this.activities).length);\n }\n\n /**\n * Remove an activity from usage\n * @param spanId The span id that represents the activity\n */\n _popActivity(spanId) {\n if (this.activities[spanId]) {\n DEBUG_BUILD && logger.log(`[Tracing] popActivity ${spanId}`);\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete this.activities[spanId];\n DEBUG_BUILD && logger.log('[Tracing] new activities count', Object.keys(this.activities).length);\n }\n\n if (Object.keys(this.activities).length === 0) {\n const endTimestamp = timestampInSeconds();\n if (this._idleTimeoutCanceledPermanently) {\n if (this._autoFinishAllowed) {\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[5];\n this.end(endTimestamp);\n }\n } else {\n // We need to add the timeout here to have the real endtimestamp of the transaction\n // Remember timestampInSeconds is in seconds, timeout is in ms\n this._restartIdleTimeout(endTimestamp + this._idleTimeout / 1000);\n }\n }\n }\n\n /**\n * Checks when entries of this.activities are not changing for 3 beats.\n * If this occurs we finish the transaction.\n */\n _beat() {\n // We should not be running heartbeat if the idle transaction is finished.\n if (this._finished) {\n return;\n }\n\n const heartbeatString = Object.keys(this.activities).join('');\n\n if (heartbeatString === this._prevHeartbeatString) {\n this._heartbeatCounter++;\n } else {\n this._heartbeatCounter = 1;\n }\n\n this._prevHeartbeatString = heartbeatString;\n\n if (this._heartbeatCounter >= 3) {\n if (this._autoFinishAllowed) {\n DEBUG_BUILD && logger.log('[Tracing] Transaction finished because of no change for 3 heart beats');\n this.setStatus('deadline_exceeded');\n this._finishReason = IDLE_TRANSACTION_FINISH_REASONS[0];\n this.end();\n }\n } else {\n this._pingHeartbeat();\n }\n }\n\n /**\n * Pings the heartbeat\n */\n _pingHeartbeat() {\n DEBUG_BUILD && logger.log(`pinging Heartbeat -> current counter: ${this._heartbeatCounter}`);\n setTimeout(() => {\n this._beat();\n }, this._heartbeatInterval);\n }\n}\n\nexport { IdleTransaction, IdleTransactionSpanRecorder, TRACING_DEFAULTS };\n","/**\n * This serves as a build time flag that will be true by default, but false in non-debug builds or if users replace `__SENTRY_DEBUG__` in their generated code.\n *\n * ATTENTION: This constant must never cross package boundaries (i.e. be exported) to guarantee that it can be used for tree shaking.\n */\nconst DEBUG_BUILD = (typeof __SENTRY_DEBUG__ === 'undefined' || __SENTRY_DEBUG__);\n\nexport { DEBUG_BUILD };\n","const bindReporter = (\n callback,\n metric,\n reportAllChanges,\n) => {\n let prevValue;\n let delta;\n return (forceReport) => {\n if (metric.value >= 0) {\n if (forceReport || reportAllChanges) {\n delta = metric.value - (prevValue || 0);\n\n // Report the metric if there's a non-zero delta or if no previous\n // value exists (which can happen in the case of the document becoming\n // hidden when the metric value is 0).\n // See: https://github.com/GoogleChrome/web-vitals/issues/14\n if (delta || prevValue === undefined) {\n prevValue = metric.value;\n metric.delta = delta;\n callback(metric);\n }\n }\n }\n };\n};\n\nexport { bindReporter };\n","import { GLOBAL_OBJ } from '@sentry/utils';\n\nconst WINDOW = GLOBAL_OBJ\n\n;\n\nexport { WINDOW };\n","import { WINDOW } from '../../types.js';\n\n/*\n * Copyright 2022 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst getNavigationEntryFromPerformanceTiming = () => {\n // eslint-disable-next-line deprecation/deprecation\n const timing = WINDOW.performance.timing;\n // eslint-disable-next-line deprecation/deprecation\n const type = WINDOW.performance.navigation.type;\n\n const navigationEntry = {\n entryType: 'navigation',\n startTime: 0,\n type: type == 2 ? 'back_forward' : type === 1 ? 'reload' : 'navigate',\n };\n\n for (const key in timing) {\n if (key !== 'navigationStart' && key !== 'toJSON') {\n // eslint-disable-next-line deprecation/deprecation\n navigationEntry[key] = Math.max((timing[key ] ) - timing.navigationStart, 0);\n }\n }\n return navigationEntry ;\n};\n\nconst getNavigationEntry = () => {\n if (WINDOW.__WEB_VITALS_POLYFILL__) {\n return (\n WINDOW.performance &&\n ((performance.getEntriesByType && performance.getEntriesByType('navigation')[0]) ||\n getNavigationEntryFromPerformanceTiming())\n );\n } else {\n return WINDOW.performance && performance.getEntriesByType && performance.getEntriesByType('navigation')[0];\n }\n};\n\nexport { getNavigationEntry };\n","import { getNavigationEntry } from './getNavigationEntry.js';\n\n/*\n * Copyright 2022 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst getActivationStart = () => {\n const navEntry = getNavigationEntry();\n return (navEntry && navEntry.activationStart) || 0;\n};\n\nexport { getActivationStart };\n","import { WINDOW } from '../../types.js';\nimport { generateUniqueID } from './generateUniqueID.js';\nimport { getActivationStart } from './getActivationStart.js';\nimport { getNavigationEntry } from './getNavigationEntry.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst initMetric = (name, value) => {\n const navEntry = getNavigationEntry();\n let navigationType = 'navigate';\n\n if (navEntry) {\n if ((WINDOW.document && WINDOW.document.prerendering) || getActivationStart() > 0) {\n navigationType = 'prerender';\n } else {\n navigationType = navEntry.type.replace(/_/g, '-') ;\n }\n }\n\n return {\n name,\n value: typeof value === 'undefined' ? -1 : value,\n rating: 'good', // Will be updated if the value changes.\n delta: 0,\n entries: [],\n id: generateUniqueID(),\n navigationType,\n };\n};\n\nexport { initMetric };\n","/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Performantly generate a unique, 30-char string by combining a version\n * number, the current timestamp with a 13-digit number integer.\n * @return {string}\n */\nconst generateUniqueID = () => {\n return `v3-${Date.now()}-${Math.floor(Math.random() * (9e12 - 1)) + 1e12}`;\n};\n\nexport { generateUniqueID };\n","/**\n * Takes a performance entry type and a callback function, and creates a\n * `PerformanceObserver` instance that will observe the specified entry type\n * with buffering enabled and call the callback _for each entry_.\n *\n * This function also feature-detects entry support and wraps the logic in a\n * try/catch to avoid errors in unsupporting browsers.\n */\nconst observe = (\n type,\n callback,\n opts,\n) => {\n try {\n if (PerformanceObserver.supportedEntryTypes.includes(type)) {\n const po = new PerformanceObserver(list => {\n callback(list.getEntries() );\n });\n po.observe(\n Object.assign(\n {\n type,\n buffered: true,\n },\n opts || {},\n ) ,\n );\n return po;\n }\n } catch (e) {\n // Do nothing.\n }\n return;\n};\n\nexport { observe };\n","import { WINDOW } from '../../types.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst onHidden = (cb, once) => {\n const onHiddenOrPageHide = (event) => {\n if (event.type === 'pagehide' || WINDOW.document.visibilityState === 'hidden') {\n cb(event);\n if (once) {\n removeEventListener('visibilitychange', onHiddenOrPageHide, true);\n removeEventListener('pagehide', onHiddenOrPageHide, true);\n }\n }\n };\n\n if (WINDOW.document) {\n addEventListener('visibilitychange', onHiddenOrPageHide, true);\n // Some browsers have buggy implementations of visibilitychange,\n // so we use pagehide in addition, just to be safe.\n addEventListener('pagehide', onHiddenOrPageHide, true);\n }\n};\n\nexport { onHidden };\n","import { WINDOW } from '../../types.js';\nimport { onHidden } from './onHidden.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nlet firstHiddenTime = -1;\n\nconst initHiddenTime = () => {\n // If the document is hidden and not prerendering, assume it was always\n // hidden and the page was loaded in the background.\n if (WINDOW.document && WINDOW.document.visibilityState) {\n firstHiddenTime = WINDOW.document.visibilityState === 'hidden' && !WINDOW.document.prerendering ? 0 : Infinity;\n }\n};\n\nconst trackChanges = () => {\n // Update the time if/when the document becomes hidden.\n onHidden(({ timeStamp }) => {\n firstHiddenTime = timeStamp;\n }, true);\n};\n\nconst getVisibilityWatcher = (\n\n) => {\n if (firstHiddenTime < 0) {\n // If the document is hidden when this code runs, assume it was hidden\n // since navigation start. This isn't a perfect heuristic, but it's the\n // best we can do until an API is available to support querying past\n // visibilityState.\n initHiddenTime();\n trackChanges();\n }\n return {\n get firstHiddenTime() {\n return firstHiddenTime;\n },\n };\n};\n\nexport { getVisibilityWatcher };\n","import { observe } from '../observe.js';\n\nlet interactionCountEstimate = 0;\nlet minKnownInteractionId = Infinity;\nlet maxKnownInteractionId = 0;\n\nconst updateEstimate = (entries) => {\n (entries ).forEach(e => {\n if (e.interactionId) {\n minKnownInteractionId = Math.min(minKnownInteractionId, e.interactionId);\n maxKnownInteractionId = Math.max(maxKnownInteractionId, e.interactionId);\n\n interactionCountEstimate = maxKnownInteractionId ? (maxKnownInteractionId - minKnownInteractionId) / 7 + 1 : 0;\n }\n });\n};\n\nlet po;\n\n/**\n * Returns the `interactionCount` value using the native API (if available)\n * or the polyfill estimate in this module.\n */\nconst getInteractionCount = () => {\n return po ? interactionCountEstimate : performance.interactionCount || 0;\n};\n\n/**\n * Feature detects native support or initializes the polyfill if needed.\n */\nconst initInteractionCountPolyfill = () => {\n if ('interactionCount' in performance || po) return;\n\n po = observe('event', updateEstimate, {\n type: 'event',\n buffered: true,\n durationThreshold: 0,\n } );\n};\n\nexport { getInteractionCount, initInteractionCountPolyfill };\n","import { bindReporter } from './lib/bindReporter.js';\nimport { initMetric } from './lib/initMetric.js';\nimport { observe } from './lib/observe.js';\nimport { onHidden } from './lib/onHidden.js';\nimport { initInteractionCountPolyfill, getInteractionCount } from './lib/polyfills/interactionCountPolyfill.js';\n\n/*\n * Copyright 2022 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Returns the interaction count since the last bfcache restore (or for the\n * full page lifecycle if there were no bfcache restores).\n */\nconst getInteractionCountForNavigation = () => {\n return getInteractionCount();\n};\n\n// To prevent unnecessary memory usage on pages with lots of interactions,\n// store at most 10 of the longest interactions to consider as INP candidates.\nconst MAX_INTERACTIONS_TO_CONSIDER = 10;\n\n// A list of longest interactions on the page (by latency) sorted so the\n// longest one is first. The list is as most MAX_INTERACTIONS_TO_CONSIDER long.\nconst longestInteractionList = [];\n\n// A mapping of longest interactions by their interaction ID.\n// This is used for faster lookup.\nconst longestInteractionMap = {};\n\n/**\n * Takes a performance entry and adds it to the list of worst interactions\n * if its duration is long enough to make it among the worst. If the\n * entry is part of an existing interaction, it is merged and the latency\n * and entries list is updated as needed.\n */\nconst processEntry = (entry) => {\n // The least-long of the 10 longest interactions.\n const minLongestInteraction = longestInteractionList[longestInteractionList.length - 1];\n\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n const existingInteraction = longestInteractionMap[entry.interactionId];\n\n // Only process the entry if it's possibly one of the ten longest,\n // or if it's part of an existing interaction.\n if (\n existingInteraction ||\n longestInteractionList.length < MAX_INTERACTIONS_TO_CONSIDER ||\n entry.duration > minLongestInteraction.latency\n ) {\n // If the interaction already exists, update it. Otherwise create one.\n if (existingInteraction) {\n existingInteraction.entries.push(entry);\n existingInteraction.latency = Math.max(existingInteraction.latency, entry.duration);\n } else {\n const interaction = {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n id: entry.interactionId,\n latency: entry.duration,\n entries: [entry],\n };\n longestInteractionMap[interaction.id] = interaction;\n longestInteractionList.push(interaction);\n }\n\n // Sort the entries by latency (descending) and keep only the top ten.\n longestInteractionList.sort((a, b) => b.latency - a.latency);\n longestInteractionList.splice(MAX_INTERACTIONS_TO_CONSIDER).forEach(i => {\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete longestInteractionMap[i.id];\n });\n }\n};\n\n/**\n * Returns the estimated p98 longest interaction based on the stored\n * interaction candidates and the interaction count for the current page.\n */\nconst estimateP98LongestInteraction = () => {\n const candidateInteractionIndex = Math.min(\n longestInteractionList.length - 1,\n Math.floor(getInteractionCountForNavigation() / 50),\n );\n\n return longestInteractionList[candidateInteractionIndex];\n};\n\n/**\n * Calculates the [INP](https://web.dev/responsiveness/) value for the current\n * page and calls the `callback` function once the value is ready, along with\n * the `event` performance entries reported for that interaction. The reported\n * value is a `DOMHighResTimeStamp`.\n *\n * A custom `durationThreshold` configuration option can optionally be passed to\n * control what `event-timing` entries are considered for INP reporting. The\n * default threshold is `40`, which means INP scores of less than 40 are\n * reported as 0. Note that this will not affect your 75th percentile INP value\n * unless that value is also less than 40 (well below the recommended\n * [good](https://web.dev/inp/#what-is-a-good-inp-score) threshold).\n *\n * If the `reportAllChanges` configuration option is set to `true`, the\n * `callback` function will be called as soon as the value is initially\n * determined as well as any time the value changes throughout the page\n * lifespan.\n *\n * _**Important:** INP should be continually monitored for changes throughout\n * the entire lifespan of a page—including if the user returns to the page after\n * it's been hidden/backgrounded. However, since browsers often [will not fire\n * additional callbacks once the user has backgrounded a\n * page](https://developer.chrome.com/blog/page-lifecycle-api/#advice-hidden),\n * `callback` is always called when the page's visibility state changes to\n * hidden. As a result, the `callback` function might be called multiple times\n * during the same page load._\n */\nconst onINP = (onReport, opts) => {\n // Set defaults\n // eslint-disable-next-line no-param-reassign\n opts = opts || {};\n\n // https://web.dev/inp/#what's-a-%22good%22-inp-value\n // const thresholds = [200, 500];\n\n // TODO(philipwalton): remove once the polyfill is no longer needed.\n initInteractionCountPolyfill();\n\n const metric = initMetric('INP');\n // eslint-disable-next-line prefer-const\n let report;\n\n const handleEntries = (entries) => {\n entries.forEach(entry => {\n if (entry.interactionId) {\n processEntry(entry);\n }\n\n // Entries of type `first-input` don't currently have an `interactionId`,\n // so to consider them in INP we have to first check that an existing\n // entry doesn't match the `duration` and `startTime`.\n // Note that this logic assumes that `event` entries are dispatched\n // before `first-input` entries. This is true in Chrome but it is not\n // true in Firefox; however, Firefox doesn't support interactionId, so\n // it's not an issue at the moment.\n // TODO(philipwalton): remove once crbug.com/1325826 is fixed.\n if (entry.entryType === 'first-input') {\n const noMatchingEntry = !longestInteractionList.some(interaction => {\n return interaction.entries.some(prevEntry => {\n return entry.duration === prevEntry.duration && entry.startTime === prevEntry.startTime;\n });\n });\n if (noMatchingEntry) {\n processEntry(entry);\n }\n }\n });\n\n const inp = estimateP98LongestInteraction();\n\n if (inp && inp.latency !== metric.value) {\n metric.value = inp.latency;\n metric.entries = inp.entries;\n report();\n }\n };\n\n const po = observe('event', handleEntries, {\n // Event Timing entries have their durations rounded to the nearest 8ms,\n // so a duration of 40ms would be any event that spans 2.5 or more frames\n // at 60Hz. This threshold is chosen to strike a balance between usefulness\n // and performance. Running this callback for any interaction that spans\n // just one or two frames is likely not worth the insight that could be\n // gained.\n durationThreshold: opts.durationThreshold || 40,\n } );\n\n report = bindReporter(onReport, metric, opts.reportAllChanges);\n\n if (po) {\n // Also observe entries of type `first-input`. This is useful in cases\n // where the first interaction is less than the `durationThreshold`.\n po.observe({ type: 'first-input', buffered: true });\n\n onHidden(() => {\n handleEntries(po.takeRecords() );\n\n // If the interaction count shows that there were interactions but\n // none were captured by the PerformanceObserver, report a latency of 0.\n if (metric.value < 0 && getInteractionCountForNavigation() > 0) {\n metric.value = 0;\n metric.entries = [];\n }\n\n report(true);\n });\n }\n};\n\nexport { onINP };\n","import { WINDOW } from '../types.js';\nimport { bindReporter } from './lib/bindReporter.js';\nimport { getActivationStart } from './lib/getActivationStart.js';\nimport { getVisibilityWatcher } from './lib/getVisibilityWatcher.js';\nimport { initMetric } from './lib/initMetric.js';\nimport { observe } from './lib/observe.js';\nimport { onHidden } from './lib/onHidden.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\nconst reportedMetricIDs = {};\n\n/**\n * Calculates the [LCP](https://web.dev/lcp/) value for the current page and\n * calls the `callback` function once the value is ready (along with the\n * relevant `largest-contentful-paint` performance entry used to determine the\n * value). The reported value is a `DOMHighResTimeStamp`.\n */\nconst onLCP = (onReport) => {\n const visibilityWatcher = getVisibilityWatcher();\n const metric = initMetric('LCP');\n let report;\n\n const handleEntries = (entries) => {\n const lastEntry = entries[entries.length - 1] ;\n if (lastEntry) {\n // The startTime attribute returns the value of the renderTime if it is\n // not 0, and the value of the loadTime otherwise. The activationStart\n // reference is used because LCP should be relative to page activation\n // rather than navigation start if the page was prerendered.\n const value = Math.max(lastEntry.startTime - getActivationStart(), 0);\n\n // Only report if the page wasn't hidden prior to LCP.\n if (value < visibilityWatcher.firstHiddenTime) {\n metric.value = value;\n metric.entries = [lastEntry];\n report();\n }\n }\n };\n\n const po = observe('largest-contentful-paint', handleEntries);\n\n if (po) {\n report = bindReporter(onReport, metric);\n\n const stopListening = () => {\n if (!reportedMetricIDs[metric.id]) {\n handleEntries(po.takeRecords() );\n po.disconnect();\n reportedMetricIDs[metric.id] = true;\n report(true);\n }\n };\n\n // Stop listening after input. Note: while scrolling is an input that\n // stop LCP observation, it's unreliable since it can be programmatically\n // generated. See: https://github.com/GoogleChrome/web-vitals/issues/75\n ['keydown', 'click'].forEach(type => {\n if (WINDOW.document) {\n addEventListener(type, stopListening, { once: true, capture: true });\n }\n });\n\n onHidden(stopListening, true);\n\n return stopListening;\n }\n\n return;\n};\n\nexport { onLCP };\n","import { WINDOW } from '../types.js';\nimport { bindReporter } from './lib/bindReporter.js';\nimport { getActivationStart } from './lib/getActivationStart.js';\nimport { getNavigationEntry } from './lib/getNavigationEntry.js';\nimport { initMetric } from './lib/initMetric.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Runs in the next task after the page is done loading and/or prerendering.\n * @param callback\n */\nconst whenReady = (callback) => {\n if (!WINDOW.document) {\n return;\n }\n\n if (WINDOW.document.prerendering) {\n addEventListener('prerenderingchange', () => whenReady(callback), true);\n } else if (WINDOW.document.readyState !== 'complete') {\n addEventListener('load', () => whenReady(callback), true);\n } else {\n // Queue a task so the callback runs after `loadEventEnd`.\n setTimeout(callback, 0);\n }\n};\n\n/**\n * Calculates the [TTFB](https://web.dev/time-to-first-byte/) value for the\n * current page and calls the `callback` function once the page has loaded,\n * along with the relevant `navigation` performance entry used to determine the\n * value. The reported value is a `DOMHighResTimeStamp`.\n *\n * Note, this function waits until after the page is loaded to call `callback`\n * in order to ensure all properties of the `navigation` entry are populated.\n * This is useful if you want to report on other metrics exposed by the\n * [Navigation Timing API](https://w3c.github.io/navigation-timing/). For\n * example, the TTFB metric starts from the page's [time\n * origin](https://www.w3.org/TR/hr-time-2/#sec-time-origin), which means it\n * includes time spent on DNS lookup, connection negotiation, network latency,\n * and server processing time.\n */\nconst onTTFB = (onReport, opts) => {\n // Set defaults\n // eslint-disable-next-line no-param-reassign\n opts = opts || {};\n\n // https://web.dev/ttfb/#what-is-a-good-ttfb-score\n // const thresholds = [800, 1800];\n\n const metric = initMetric('TTFB');\n const report = bindReporter(onReport, metric, opts.reportAllChanges);\n\n whenReady(() => {\n const navEntry = getNavigationEntry() ;\n\n if (navEntry) {\n // The activationStart reference is used because TTFB should be\n // relative to page activation rather than navigation start if the\n // page was prerendered. But in cases where `activationStart` occurs\n // after the first byte is received, this time should be clamped at 0.\n metric.value = Math.max(navEntry.responseStart - getActivationStart(), 0);\n\n // In some cases the value reported is negative or is larger\n // than the current page time. Ignore these cases:\n // https://github.com/GoogleChrome/web-vitals/issues/137\n // https://github.com/GoogleChrome/web-vitals/issues/162\n if (metric.value < 0 || metric.value > performance.now()) return;\n\n metric.entries = [navEntry];\n\n report(true);\n }\n });\n};\n\nexport { onTTFB };\n","import { logger, getFunctionName } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../common/debug-build.js';\nimport { onCLS } from './web-vitals/getCLS.js';\nimport { onFID } from './web-vitals/getFID.js';\nimport { onINP } from './web-vitals/getINP.js';\nimport { onLCP } from './web-vitals/getLCP.js';\nimport { observe } from './web-vitals/lib/observe.js';\nimport { onTTFB } from './web-vitals/onTTFB.js';\n\nconst handlers = {};\nconst instrumented = {};\n\nlet _previousCls;\nlet _previousFid;\nlet _previousLcp;\nlet _previousTtfb;\nlet _previousInp;\n\n/**\n * Add a callback that will be triggered when a CLS metric is available.\n * Returns a cleanup callback which can be called to remove the instrumentation handler.\n *\n * Pass `stopOnCallback = true` to stop listening for CLS when the cleanup callback is called.\n * This will lead to the CLS being finalized and frozen.\n */\nfunction addClsInstrumentationHandler(\n callback,\n stopOnCallback = false,\n) {\n return addMetricObserver('cls', callback, instrumentCls, _previousCls, stopOnCallback);\n}\n\n/**\n * Add a callback that will be triggered when a LCP metric is available.\n * Returns a cleanup callback which can be called to remove the instrumentation handler.\n *\n * Pass `stopOnCallback = true` to stop listening for LCP when the cleanup callback is called.\n * This will lead to the LCP being finalized and frozen.\n */\nfunction addLcpInstrumentationHandler(\n callback,\n stopOnCallback = false,\n) {\n return addMetricObserver('lcp', callback, instrumentLcp, _previousLcp, stopOnCallback);\n}\n\n/**\n * Add a callback that will be triggered when a FID metric is available.\n */\nfunction addTtfbInstrumentationHandler(callback) {\n return addMetricObserver('ttfb', callback, instrumentTtfb, _previousTtfb);\n}\n\n/**\n * Add a callback that will be triggered when a FID metric is available.\n * Returns a cleanup callback which can be called to remove the instrumentation handler.\n */\nfunction addFidInstrumentationHandler(callback) {\n return addMetricObserver('fid', callback, instrumentFid, _previousFid);\n}\n\n/**\n * Add a callback that will be triggered when a INP metric is available.\n * Returns a cleanup callback which can be called to remove the instrumentation handler.\n */\nfunction addInpInstrumentationHandler(\n callback,\n) {\n return addMetricObserver('inp', callback, instrumentInp, _previousInp);\n}\n\n/**\n * Add a callback that will be triggered when a performance observer is triggered,\n * and receives the entries of the observer.\n * Returns a cleanup callback which can be called to remove the instrumentation handler.\n */\nfunction addPerformanceInstrumentationHandler(\n type,\n callback,\n) {\n addHandler(type, callback);\n\n if (!instrumented[type]) {\n instrumentPerformanceObserver(type);\n instrumented[type] = true;\n }\n\n return getCleanupCallback(type, callback);\n}\n\n/** Trigger all handlers of a given type. */\nfunction triggerHandlers(type, data) {\n const typeHandlers = handlers[type];\n\n if (!typeHandlers || !typeHandlers.length) {\n return;\n }\n\n for (const handler of typeHandlers) {\n try {\n handler(data);\n } catch (e) {\n DEBUG_BUILD &&\n logger.error(\n `Error while triggering instrumentation handler.\\nType: ${type}\\nName: ${getFunctionName(handler)}\\nError:`,\n e,\n );\n }\n }\n}\n\nfunction instrumentCls() {\n return onCLS(\n metric => {\n triggerHandlers('cls', {\n metric,\n });\n _previousCls = metric;\n },\n { reportAllChanges: true },\n );\n}\n\nfunction instrumentFid() {\n return onFID(metric => {\n triggerHandlers('fid', {\n metric,\n });\n _previousFid = metric;\n });\n}\n\nfunction instrumentLcp() {\n return onLCP(metric => {\n triggerHandlers('lcp', {\n metric,\n });\n _previousLcp = metric;\n });\n}\n\nfunction instrumentTtfb() {\n return onTTFB(metric => {\n triggerHandlers('ttfb', {\n metric,\n });\n _previousTtfb = metric;\n });\n}\n\nfunction instrumentInp() {\n return onINP(metric => {\n triggerHandlers('inp', {\n metric,\n });\n _previousInp = metric;\n });\n}\n\nfunction addMetricObserver(\n type,\n callback,\n instrumentFn,\n previousValue,\n stopOnCallback = false,\n) {\n addHandler(type, callback);\n\n let stopListening;\n\n if (!instrumented[type]) {\n stopListening = instrumentFn();\n instrumented[type] = true;\n }\n\n if (previousValue) {\n callback({ metric: previousValue });\n }\n\n return getCleanupCallback(type, callback, stopOnCallback ? stopListening : undefined);\n}\n\nfunction instrumentPerformanceObserver(type) {\n const options = {};\n\n // Special per-type options we want to use\n if (type === 'event') {\n options.durationThreshold = 0;\n }\n\n observe(\n type,\n entries => {\n triggerHandlers(type, { entries });\n },\n options,\n );\n}\n\nfunction addHandler(type, handler) {\n handlers[type] = handlers[type] || [];\n (handlers[type] ).push(handler);\n}\n\n// Get a callback which can be called to remove the instrumentation handler\nfunction getCleanupCallback(\n type,\n callback,\n stopListening,\n) {\n return () => {\n if (stopListening) {\n stopListening();\n }\n\n const typeHandlers = handlers[type];\n\n if (!typeHandlers) {\n return;\n }\n\n const index = typeHandlers.indexOf(callback);\n if (index !== -1) {\n typeHandlers.splice(index, 1);\n }\n };\n}\n\nexport { addClsInstrumentationHandler, addFidInstrumentationHandler, addInpInstrumentationHandler, addLcpInstrumentationHandler, addPerformanceInstrumentationHandler, addTtfbInstrumentationHandler };\n","import { bindReporter } from './lib/bindReporter.js';\nimport { initMetric } from './lib/initMetric.js';\nimport { observe } from './lib/observe.js';\nimport { onHidden } from './lib/onHidden.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Calculates the [CLS](https://web.dev/cls/) value for the current page and\n * calls the `callback` function once the value is ready to be reported, along\n * with all `layout-shift` performance entries that were used in the metric\n * value calculation. The reported value is a `double` (corresponding to a\n * [layout shift score](https://web.dev/cls/#layout-shift-score)).\n *\n * If the `reportAllChanges` configuration option is set to `true`, the\n * `callback` function will be called as soon as the value is initially\n * determined as well as any time the value changes throughout the page\n * lifespan.\n *\n * _**Important:** CLS should be continually monitored for changes throughout\n * the entire lifespan of a page—including if the user returns to the page after\n * it's been hidden/backgrounded. However, since browsers often [will not fire\n * additional callbacks once the user has backgrounded a\n * page](https://developer.chrome.com/blog/page-lifecycle-api/#advice-hidden),\n * `callback` is always called when the page's visibility state changes to\n * hidden. As a result, the `callback` function might be called multiple times\n * during the same page load._\n */\nconst onCLS = (\n onReport,\n options = {},\n) => {\n const metric = initMetric('CLS', 0);\n let report;\n\n let sessionValue = 0;\n let sessionEntries = [];\n\n // const handleEntries = (entries: Metric['entries']) => {\n const handleEntries = (entries) => {\n entries.forEach(entry => {\n // Only count layout shifts without recent user input.\n if (!entry.hadRecentInput) {\n const firstSessionEntry = sessionEntries[0];\n const lastSessionEntry = sessionEntries[sessionEntries.length - 1];\n\n // If the entry occurred less than 1 second after the previous entry and\n // less than 5 seconds after the first entry in the session, include the\n // entry in the current session. Otherwise, start a new session.\n if (\n sessionValue &&\n sessionEntries.length !== 0 &&\n entry.startTime - lastSessionEntry.startTime < 1000 &&\n entry.startTime - firstSessionEntry.startTime < 5000\n ) {\n sessionValue += entry.value;\n sessionEntries.push(entry);\n } else {\n sessionValue = entry.value;\n sessionEntries = [entry];\n }\n\n // If the current session value is larger than the current CLS value,\n // update CLS and the entries contributing to it.\n if (sessionValue > metric.value) {\n metric.value = sessionValue;\n metric.entries = sessionEntries;\n if (report) {\n report();\n }\n }\n }\n });\n };\n\n const po = observe('layout-shift', handleEntries);\n if (po) {\n report = bindReporter(onReport, metric, options.reportAllChanges);\n\n const stopListening = () => {\n handleEntries(po.takeRecords() );\n report(true);\n };\n\n onHidden(stopListening);\n\n return stopListening;\n }\n\n return;\n};\n\nexport { onCLS };\n","import { bindReporter } from './lib/bindReporter.js';\nimport { getVisibilityWatcher } from './lib/getVisibilityWatcher.js';\nimport { initMetric } from './lib/initMetric.js';\nimport { observe } from './lib/observe.js';\nimport { onHidden } from './lib/onHidden.js';\n\n/*\n * Copyright 2020 Google LLC\n *\n * Licensed under the Apache License, Version 2.0 (the \"License\");\n * you may not use this file except in compliance with the License.\n * You may obtain a copy of the License at\n *\n * https://www.apache.org/licenses/LICENSE-2.0\n *\n * Unless required by applicable law or agreed to in writing, software\n * distributed under the License is distributed on an \"AS IS\" BASIS,\n * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\n * See the License for the specific language governing permissions and\n * limitations under the License.\n */\n\n/**\n * Calculates the [FID](https://web.dev/fid/) value for the current page and\n * calls the `callback` function once the value is ready, along with the\n * relevant `first-input` performance entry used to determine the value. The\n * reported value is a `DOMHighResTimeStamp`.\n *\n * _**Important:** since FID is only reported after the user interacts with the\n * page, it's possible that it will not be reported for some page loads._\n */\nconst onFID = (onReport) => {\n const visibilityWatcher = getVisibilityWatcher();\n const metric = initMetric('FID');\n // eslint-disable-next-line prefer-const\n let report;\n\n const handleEntry = (entry) => {\n // Only report if the page wasn't hidden prior to the first input.\n if (entry.startTime < visibilityWatcher.firstHiddenTime) {\n metric.value = entry.processingStart - entry.startTime;\n metric.entries.push(entry);\n report(true);\n }\n };\n\n const handleEntries = (entries) => {\n (entries ).forEach(handleEntry);\n };\n\n const po = observe('first-input', handleEntries);\n report = bindReporter(onReport, metric);\n\n if (po) {\n onHidden(() => {\n handleEntries(po.takeRecords() );\n po.disconnect();\n }, true);\n }\n};\n\nexport { onFID };\n","import { spanToJSON, hasTracingEnabled, setHttpStatus, getCurrentScope, getIsolationScope, startInactiveSpan, SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, getClient, spanToTraceHeader, getDynamicSamplingContextFromSpan, getDynamicSamplingContextFromClient } from '@sentry/core';\nimport { addFetchInstrumentationHandler, parseUrl, addXhrInstrumentationHandler, SENTRY_XHR_DATA_KEY, generateSentryTraceHeader, dynamicSamplingContextToSentryBaggageHeader, BAGGAGE_HEADER_NAME, browserPerformanceTimeOrigin, stringMatchesSomePattern } from '@sentry/utils';\nimport { instrumentFetchRequest } from '../common/fetch.js';\nimport { addPerformanceInstrumentationHandler } from './instrument.js';\nimport { WINDOW } from './types.js';\n\n/* eslint-disable max-lines */\n\nconst DEFAULT_TRACE_PROPAGATION_TARGETS = ['localhost', /^\\/(?!\\/)/];\n\n/** Options for Request Instrumentation */\n\nconst defaultRequestInstrumentationOptions = {\n traceFetch: true,\n traceXHR: true,\n enableHTTPTimings: true,\n // TODO (v8): Remove this property\n tracingOrigins: DEFAULT_TRACE_PROPAGATION_TARGETS,\n tracePropagationTargets: DEFAULT_TRACE_PROPAGATION_TARGETS,\n};\n\n/** Registers span creators for xhr and fetch requests */\nfunction instrumentOutgoingRequests(_options) {\n const {\n traceFetch,\n traceXHR,\n // eslint-disable-next-line deprecation/deprecation\n tracePropagationTargets,\n // eslint-disable-next-line deprecation/deprecation\n tracingOrigins,\n shouldCreateSpanForRequest,\n enableHTTPTimings,\n } = {\n traceFetch: defaultRequestInstrumentationOptions.traceFetch,\n traceXHR: defaultRequestInstrumentationOptions.traceXHR,\n ..._options,\n };\n\n const shouldCreateSpan =\n typeof shouldCreateSpanForRequest === 'function' ? shouldCreateSpanForRequest : (_) => true;\n\n // TODO(v8) Remove tracingOrigins here\n // The only reason we're passing it in here is because this instrumentOutgoingRequests function is publicly exported\n // and we don't want to break the API. We can remove it in v8.\n const shouldAttachHeadersWithTargets = (url) =>\n shouldAttachHeaders(url, tracePropagationTargets || tracingOrigins);\n\n const spans = {};\n\n if (traceFetch) {\n addFetchInstrumentationHandler(handlerData => {\n const createdSpan = instrumentFetchRequest(handlerData, shouldCreateSpan, shouldAttachHeadersWithTargets, spans);\n // We cannot use `window.location` in the generic fetch instrumentation,\n // but we need it for reliable `server.address` attribute.\n // so we extend this in here\n if (createdSpan) {\n const fullUrl = getFullURL(handlerData.fetchData.url);\n const host = fullUrl ? parseUrl(fullUrl).host : undefined;\n createdSpan.setAttributes({\n 'http.url': fullUrl,\n 'server.address': host,\n });\n }\n\n if (enableHTTPTimings && createdSpan) {\n addHTTPTimings(createdSpan);\n }\n });\n }\n\n if (traceXHR) {\n addXhrInstrumentationHandler(handlerData => {\n const createdSpan = xhrCallback(handlerData, shouldCreateSpan, shouldAttachHeadersWithTargets, spans);\n if (enableHTTPTimings && createdSpan) {\n addHTTPTimings(createdSpan);\n }\n });\n }\n}\n\nfunction isPerformanceResourceTiming(entry) {\n return (\n entry.entryType === 'resource' &&\n 'initiatorType' in entry &&\n typeof (entry ).nextHopProtocol === 'string' &&\n (entry.initiatorType === 'fetch' || entry.initiatorType === 'xmlhttprequest')\n );\n}\n\n/**\n * Creates a temporary observer to listen to the next fetch/xhr resourcing timings,\n * so that when timings hit their per-browser limit they don't need to be removed.\n *\n * @param span A span that has yet to be finished, must contain `url` on data.\n */\nfunction addHTTPTimings(span) {\n const { url } = spanToJSON(span).data || {};\n\n if (!url || typeof url !== 'string') {\n return;\n }\n\n const cleanup = addPerformanceInstrumentationHandler('resource', ({ entries }) => {\n entries.forEach(entry => {\n if (isPerformanceResourceTiming(entry) && entry.name.endsWith(url)) {\n const spanData = resourceTimingEntryToSpanData(entry);\n spanData.forEach(data => span.setAttribute(...data));\n // In the next tick, clean this handler up\n // We have to wait here because otherwise this cleans itself up before it is fully done\n setTimeout(cleanup);\n }\n });\n });\n}\n\n/**\n * Converts ALPN protocol ids to name and version.\n *\n * (https://www.iana.org/assignments/tls-extensiontype-values/tls-extensiontype-values.xhtml#alpn-protocol-ids)\n * @param nextHopProtocol PerformanceResourceTiming.nextHopProtocol\n */\nfunction extractNetworkProtocol(nextHopProtocol) {\n let name = 'unknown';\n let version = 'unknown';\n let _name = '';\n for (const char of nextHopProtocol) {\n // http/1.1 etc.\n if (char === '/') {\n [name, version] = nextHopProtocol.split('/');\n break;\n }\n // h2, h3 etc.\n if (!isNaN(Number(char))) {\n name = _name === 'h' ? 'http' : _name;\n version = nextHopProtocol.split(_name)[1];\n break;\n }\n _name += char;\n }\n if (_name === nextHopProtocol) {\n // webrtc, ftp, etc.\n name = _name;\n }\n return { name, version };\n}\n\nfunction getAbsoluteTime(time = 0) {\n return ((browserPerformanceTimeOrigin || performance.timeOrigin) + time) / 1000;\n}\n\nfunction resourceTimingEntryToSpanData(resourceTiming) {\n const { name, version } = extractNetworkProtocol(resourceTiming.nextHopProtocol);\n\n const timingSpanData = [];\n\n timingSpanData.push(['network.protocol.version', version], ['network.protocol.name', name]);\n\n if (!browserPerformanceTimeOrigin) {\n return timingSpanData;\n }\n return [\n ...timingSpanData,\n ['http.request.redirect_start', getAbsoluteTime(resourceTiming.redirectStart)],\n ['http.request.fetch_start', getAbsoluteTime(resourceTiming.fetchStart)],\n ['http.request.domain_lookup_start', getAbsoluteTime(resourceTiming.domainLookupStart)],\n ['http.request.domain_lookup_end', getAbsoluteTime(resourceTiming.domainLookupEnd)],\n ['http.request.connect_start', getAbsoluteTime(resourceTiming.connectStart)],\n ['http.request.secure_connection_start', getAbsoluteTime(resourceTiming.secureConnectionStart)],\n ['http.request.connection_end', getAbsoluteTime(resourceTiming.connectEnd)],\n ['http.request.request_start', getAbsoluteTime(resourceTiming.requestStart)],\n ['http.request.response_start', getAbsoluteTime(resourceTiming.responseStart)],\n ['http.request.response_end', getAbsoluteTime(resourceTiming.responseEnd)],\n ];\n}\n\n/**\n * A function that determines whether to attach tracing headers to a request.\n * This was extracted from `instrumentOutgoingRequests` to make it easier to test shouldAttachHeaders.\n * We only export this fuction for testing purposes.\n */\nfunction shouldAttachHeaders(url, tracePropagationTargets) {\n return stringMatchesSomePattern(url, tracePropagationTargets || DEFAULT_TRACE_PROPAGATION_TARGETS);\n}\n\n/**\n * Create and track xhr request spans\n *\n * @returns Span if a span was created, otherwise void.\n */\n// eslint-disable-next-line complexity\nfunction xhrCallback(\n handlerData,\n shouldCreateSpan,\n shouldAttachHeaders,\n spans,\n) {\n const xhr = handlerData.xhr;\n const sentryXhrData = xhr && xhr[SENTRY_XHR_DATA_KEY];\n\n if (!hasTracingEnabled() || !xhr || xhr.__sentry_own_request__ || !sentryXhrData) {\n return undefined;\n }\n\n const shouldCreateSpanResult = shouldCreateSpan(sentryXhrData.url);\n\n // check first if the request has finished and is tracked by an existing span which should now end\n if (handlerData.endTimestamp && shouldCreateSpanResult) {\n const spanId = xhr.__sentry_xhr_span_id__;\n if (!spanId) return;\n\n const span = spans[spanId];\n if (span && sentryXhrData.status_code !== undefined) {\n setHttpStatus(span, sentryXhrData.status_code);\n span.end();\n\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete spans[spanId];\n }\n return undefined;\n }\n\n const scope = getCurrentScope();\n const isolationScope = getIsolationScope();\n\n const fullUrl = getFullURL(sentryXhrData.url);\n const host = fullUrl ? parseUrl(fullUrl).host : undefined;\n\n const span = shouldCreateSpanResult\n ? startInactiveSpan({\n name: `${sentryXhrData.method} ${sentryXhrData.url}`,\n onlyIfParent: true,\n attributes: {\n type: 'xhr',\n 'http.method': sentryXhrData.method,\n 'http.url': fullUrl,\n url: sentryXhrData.url,\n 'server.address': host,\n [SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN]: 'auto.http.browser',\n },\n op: 'http.client',\n })\n : undefined;\n\n if (span) {\n xhr.__sentry_xhr_span_id__ = span.spanContext().spanId;\n spans[xhr.__sentry_xhr_span_id__] = span;\n }\n\n const client = getClient();\n\n if (xhr.setRequestHeader && shouldAttachHeaders(sentryXhrData.url) && client) {\n const { traceId, spanId, sampled, dsc } = {\n ...isolationScope.getPropagationContext(),\n ...scope.getPropagationContext(),\n };\n\n const sentryTraceHeader = span ? spanToTraceHeader(span) : generateSentryTraceHeader(traceId, spanId, sampled);\n\n const sentryBaggageHeader = dynamicSamplingContextToSentryBaggageHeader(\n dsc ||\n (span ? getDynamicSamplingContextFromSpan(span) : getDynamicSamplingContextFromClient(traceId, client, scope)),\n );\n\n setHeaderOnXhr(xhr, sentryTraceHeader, sentryBaggageHeader);\n }\n\n return span;\n}\n\nfunction setHeaderOnXhr(\n xhr,\n sentryTraceHeader,\n sentryBaggageHeader,\n) {\n try {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n xhr.setRequestHeader('sentry-trace', sentryTraceHeader);\n if (sentryBaggageHeader) {\n // From MDN: \"If this method is called several times with the same header, the values are merged into one single request header.\"\n // We can therefore simply set a baggage header without checking what was there before\n // https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest/setRequestHeader\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n xhr.setRequestHeader(BAGGAGE_HEADER_NAME, sentryBaggageHeader);\n }\n } catch (_) {\n // Error: InvalidStateError: Failed to execute 'setRequestHeader' on 'XMLHttpRequest': The object's state must be OPENED.\n }\n}\n\nfunction getFullURL(url) {\n try {\n // By adding a base URL to new URL(), this will also work for relative urls\n // If `url` is a full URL, the base URL is ignored anyhow\n const parsed = new URL(url, WINDOW.location.origin);\n return parsed.href;\n } catch (e) {\n return undefined;\n }\n}\n\nexport { DEFAULT_TRACE_PROPAGATION_TARGETS, defaultRequestInstrumentationOptions, extractNetworkProtocol, instrumentOutgoingRequests, shouldAttachHeaders, xhrCallback };\n","import { hasTracingEnabled, getCurrentScope, getClient, startInactiveSpan, SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, getIsolationScope, spanToTraceHeader, getDynamicSamplingContextFromSpan, getDynamicSamplingContextFromClient, setHttpStatus } from '@sentry/core';\nimport { parseUrl, generateSentryTraceHeader, dynamicSamplingContextToSentryBaggageHeader, isInstanceOf, BAGGAGE_HEADER_NAME } from '@sentry/utils';\n\n/**\n * Create and track fetch request spans for usage in combination with `addInstrumentationHandler`.\n *\n * @returns Span if a span was created, otherwise void.\n */\nfunction instrumentFetchRequest(\n handlerData,\n shouldCreateSpan,\n shouldAttachHeaders,\n spans,\n spanOrigin = 'auto.http.browser',\n) {\n if (!hasTracingEnabled() || !handlerData.fetchData) {\n return undefined;\n }\n\n const shouldCreateSpanResult = shouldCreateSpan(handlerData.fetchData.url);\n\n if (handlerData.endTimestamp && shouldCreateSpanResult) {\n const spanId = handlerData.fetchData.__span;\n if (!spanId) return;\n\n const span = spans[spanId];\n if (span) {\n endSpan(span, handlerData);\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete spans[spanId];\n }\n return undefined;\n }\n\n const scope = getCurrentScope();\n const client = getClient();\n\n const { method, url } = handlerData.fetchData;\n\n const fullUrl = getFullURL(url);\n const host = fullUrl ? parseUrl(fullUrl).host : undefined;\n\n const span = shouldCreateSpanResult\n ? startInactiveSpan({\n name: `${method} ${url}`,\n onlyIfParent: true,\n attributes: {\n url,\n type: 'fetch',\n 'http.method': method,\n 'http.url': fullUrl,\n 'server.address': host,\n [SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN]: spanOrigin,\n },\n op: 'http.client',\n })\n : undefined;\n\n if (span) {\n handlerData.fetchData.__span = span.spanContext().spanId;\n spans[span.spanContext().spanId] = span;\n }\n\n if (shouldAttachHeaders(handlerData.fetchData.url) && client) {\n const request = handlerData.args[0];\n\n // In case the user hasn't set the second argument of a fetch call we default it to `{}`.\n handlerData.args[1] = handlerData.args[1] || {};\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const options = handlerData.args[1];\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-member-access\n options.headers = addTracingHeadersToFetchRequest(request, client, scope, options, span);\n }\n\n return span;\n}\n\n/**\n * Adds sentry-trace and baggage headers to the various forms of fetch headers\n */\nfunction addTracingHeadersToFetchRequest(\n request, // unknown is actually type Request but we can't export DOM types from this package,\n client,\n scope,\n options\n\n,\n requestSpan,\n) {\n // eslint-disable-next-line deprecation/deprecation\n const span = requestSpan || scope.getSpan();\n\n const isolationScope = getIsolationScope();\n\n const { traceId, spanId, sampled, dsc } = {\n ...isolationScope.getPropagationContext(),\n ...scope.getPropagationContext(),\n };\n\n const sentryTraceHeader = span ? spanToTraceHeader(span) : generateSentryTraceHeader(traceId, spanId, sampled);\n\n const sentryBaggageHeader = dynamicSamplingContextToSentryBaggageHeader(\n dsc ||\n (span ? getDynamicSamplingContextFromSpan(span) : getDynamicSamplingContextFromClient(traceId, client, scope)),\n );\n\n const headers =\n options.headers ||\n (typeof Request !== 'undefined' && isInstanceOf(request, Request) ? (request ).headers : undefined);\n\n if (!headers) {\n return { 'sentry-trace': sentryTraceHeader, baggage: sentryBaggageHeader };\n } else if (typeof Headers !== 'undefined' && isInstanceOf(headers, Headers)) {\n const newHeaders = new Headers(headers );\n\n newHeaders.append('sentry-trace', sentryTraceHeader);\n\n if (sentryBaggageHeader) {\n // If the same header is appended multiple times the browser will merge the values into a single request header.\n // Its therefore safe to simply push a \"baggage\" entry, even though there might already be another baggage header.\n newHeaders.append(BAGGAGE_HEADER_NAME, sentryBaggageHeader);\n }\n\n return newHeaders ;\n } else if (Array.isArray(headers)) {\n const newHeaders = [...headers, ['sentry-trace', sentryTraceHeader]];\n\n if (sentryBaggageHeader) {\n // If there are multiple entries with the same key, the browser will merge the values into a single request header.\n // Its therefore safe to simply push a \"baggage\" entry, even though there might already be another baggage header.\n newHeaders.push([BAGGAGE_HEADER_NAME, sentryBaggageHeader]);\n }\n\n return newHeaders ;\n } else {\n const existingBaggageHeader = 'baggage' in headers ? headers.baggage : undefined;\n const newBaggageHeaders = [];\n\n if (Array.isArray(existingBaggageHeader)) {\n newBaggageHeaders.push(...existingBaggageHeader);\n } else if (existingBaggageHeader) {\n newBaggageHeaders.push(existingBaggageHeader);\n }\n\n if (sentryBaggageHeader) {\n newBaggageHeaders.push(sentryBaggageHeader);\n }\n\n return {\n ...(headers ),\n 'sentry-trace': sentryTraceHeader,\n baggage: newBaggageHeaders.length > 0 ? newBaggageHeaders.join(',') : undefined,\n };\n }\n}\n\nfunction getFullURL(url) {\n try {\n const parsed = new URL(url);\n return parsed.href;\n } catch (e) {\n return undefined;\n }\n}\n\nfunction endSpan(span, handlerData) {\n if (handlerData.response) {\n setHttpStatus(span, handlerData.response.status);\n\n const contentLength =\n handlerData.response && handlerData.response.headers && handlerData.response.headers.get('content-length');\n\n if (contentLength) {\n const contentLengthNum = parseInt(contentLength);\n if (contentLengthNum > 0) {\n span.setAttribute('http.response_content_length', contentLengthNum);\n }\n }\n } else if (handlerData.error) {\n span.setStatus('internal_error');\n }\n span.end();\n}\n\nexport { addTracingHeadersToFetchRequest, instrumentFetchRequest };\n","import * as Sentry from '@sentry/browser';\nimport { getDefaultIntegrations, init as init$1, setContext, WINDOW, browserTracingIntegration as browserTracingIntegration$1, getCurrentScope, startBrowserTracingNavigationSpan } from '@sentry/browser';\nexport * from '@sentry/browser';\nimport * as i0 from '@angular/core';\nimport { VERSION, Injectable, Inject, Directive, Input, NgModule } from '@angular/core';\nimport { applySdkMetadata, SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, getClient, getActiveSpan, startInactiveSpan, spanToJSON, SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN } from '@sentry/core';\nimport { logger, isString, stripUrlQueryAndFragment, timestampInSeconds } from '@sentry/utils';\nimport { HttpErrorResponse } from '@angular/common/http';\nimport * as i1 from '@angular/router';\nimport { NavigationStart, ResolveEnd, NavigationEnd, NavigationCancel, NavigationError } from '@angular/router';\nimport { Subscription } from 'rxjs';\nimport { filter, tap } from 'rxjs/operators';\n\n/*\n * This file defines flags and constants that can be modified during compile time in order to facilitate tree shaking\n * for users.\n *\n * We define \"magic strings\" like `__SENTRY_DEBUG__` that may get replaced with actual values during our, or the user's\n * build process. Take care when introducing new flags - they must not throw if they are not replaced. See the Debug\n * Build Flags section in CONTRIBUTING.md.\n */\n/** Flag that is true for debug builds, false otherwise. */\nconst IS_DEBUG_BUILD = typeof __SENTRY_DEBUG__ === 'undefined' ? true : __SENTRY_DEBUG__;\n\n/**\n * Inits the Angular SDK\n */\nfunction init(options) {\n const opts = Object.assign({ \n // Filter out TryCatch integration as it interferes with our Angular `ErrorHandler`:\n // TryCatch would catch certain errors before they reach the `ErrorHandler` and thus provide a\n // lower fidelity error than what `SentryErrorHandler` (see errorhandler.ts) would provide.\n // see:\n // - https://github.com/getsentry/sentry-javascript/issues/5417#issuecomment-1453407097\n // - https://github.com/getsentry/sentry-javascript/issues/2744\n defaultIntegrations: getDefaultIntegrations(options).filter(integration => {\n return integration.name !== 'TryCatch';\n }) }, options);\n applySdkMetadata(opts, 'angular-ivy');\n checkAndSetAngularVersion();\n init$1(opts);\n}\nfunction checkAndSetAngularVersion() {\n const ANGULAR_MINIMUM_VERSION = 12;\n const angularVersion = VERSION && VERSION.major ? parseInt(VERSION.major, 10) : undefined;\n if (angularVersion) {\n if (angularVersion < ANGULAR_MINIMUM_VERSION) {\n IS_DEBUG_BUILD &&\n logger.warn(`This Sentry SDK does not officially support Angular ${angularVersion}.`, `This SDK only supports Angular ${ANGULAR_MINIMUM_VERSION} and above.`, \"If you're using Angular 10 or 11, please use `@sentry/angular` instead.\", 'Otherwise, please consider upgrading your Angular version.');\n }\n setContext('angular', { version: angularVersion });\n }\n}\n\n// There're 2 types of Angular applications:\n// 1) zone-full (by default)\n// 2) zone-less\n// The developer can avoid importing the `zone.js` package and tells Angular that\n// he is responsible for running the change detection by himself. This is done by\n// \"nooping\" the zone through `CompilerOptions` when bootstrapping the root module.\n// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\nconst isNgZoneEnabled = typeof Zone !== 'undefined' && !!Zone.current;\n/**\n * The function that does the same job as `NgZone.runOutsideAngular`.\n */\nfunction runOutsideAngular(callback) {\n // The `Zone.root.run` basically will run the `callback` in the most parent zone.\n // Any asynchronous API used inside the `callback` won't catch Angular's zone\n // since `Zone.current` will reference `Zone.root`.\n // The Angular's zone is forked from the `Zone.root`. In this case, `zone.js` won't\n // trigger change detection, and `ApplicationRef.tick()` will not be run.\n // Caretaker note: we're using `Zone.root` except `NgZone.runOutsideAngular` since this\n // will require injecting the `NgZone` facade. That will create a breaking change for\n // projects already using the `@sentry/angular`.\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return isNgZoneEnabled ? Zone.root.run(callback) : callback();\n}\n\n// https://github.com/angular/angular/blob/master/packages/core/src/util/errors.ts\nfunction tryToUnwrapZonejsError(error) {\n // TODO: once Angular14 is the minimum requirement ERROR_ORIGINAL_ERROR and\n // getOriginalError from error.ts can be used directly.\n return error && error.ngOriginalError\n ? error.ngOriginalError\n : error;\n}\nfunction extractHttpModuleError(error) {\n // The `error` property of http exception can be either an `Error` object, which we can use directly...\n if (isErrorOrErrorLikeObject(error.error)) {\n return error.error;\n }\n // ... or an`ErrorEvent`, which can provide us with the message but no stack...\n if (error.error instanceof ErrorEvent && error.error.message) {\n return error.error.message;\n }\n // ...or the request body itself, which we can use as a message instead.\n if (typeof error.error === 'string') {\n return `Server returned code ${error.status} with body \"${error.error}\"`;\n }\n // If we don't have any detailed information, fallback to the request message itself.\n return error.message;\n}\nfunction isErrorOrErrorLikeObject(value) {\n if (value instanceof Error) {\n return true;\n }\n if (value === null || typeof value !== 'object') {\n return false;\n }\n const candidate = value;\n return (isString(candidate.name) &&\n isString(candidate.message) &&\n (undefined === candidate.stack || isString(candidate.stack)));\n}\n/**\n * Implementation of Angular's ErrorHandler provider that can be used as a drop-in replacement for the stock one.\n */\nclass SentryErrorHandler {\n constructor(options) {\n this._registeredAfterSendEventHandler = false;\n this._options = Object.assign({ logErrors: true }, options);\n }\n /**\n * Method called for every value captured through the ErrorHandler\n */\n handleError(error) {\n const extractedError = this._extractError(error) || 'Handled unknown error';\n // Capture handled exception and send it to Sentry.\n const eventId = runOutsideAngular(() => Sentry.captureException(extractedError, {\n mechanism: { type: 'angular', handled: false },\n }));\n // When in development mode, log the error to console for immediate feedback.\n if (this._options.logErrors) {\n // eslint-disable-next-line no-console\n console.error(extractedError);\n }\n // Optionally show user dialog to provide details on what happened.\n if (this._options.showDialog) {\n const client = Sentry.getClient();\n if (client && client.on && !this._registeredAfterSendEventHandler) {\n client.on('afterSendEvent', (event) => {\n if (!event.type) {\n // eslint-disable-next-line deprecation/deprecation\n Sentry.showReportDialog(Object.assign(Object.assign({}, this._options.dialogOptions), { eventId: event.event_id }));\n }\n });\n // We only want to register this hook once in the lifetime of the error handler\n this._registeredAfterSendEventHandler = true;\n }\n else if (!client || !client.on) {\n Sentry.showReportDialog(Object.assign(Object.assign({}, this._options.dialogOptions), { eventId }));\n }\n }\n }\n /**\n * Used to pull a desired value that will be used to capture an event out of the raw value captured by ErrorHandler.\n */\n _extractError(error) {\n // Allow custom overrides of extracting function\n if (this._options.extractor) {\n const defaultExtractor = this._defaultExtractor.bind(this);\n return this._options.extractor(error, defaultExtractor);\n }\n return this._defaultExtractor(error);\n }\n /**\n * Default implementation of error extraction that handles default error wrapping, HTTP responses, ErrorEvent and few other known cases.\n */\n _defaultExtractor(errorCandidate) {\n const error = tryToUnwrapZonejsError(errorCandidate);\n // If it's http module error, extract as much information from it as we can.\n if (error instanceof HttpErrorResponse) {\n return extractHttpModuleError(error);\n }\n // We can handle messages and Error objects directly.\n if (typeof error === 'string' || isErrorOrErrorLikeObject(error)) {\n return error;\n }\n // Nothing was extracted, fallback to default error message.\n return null;\n }\n}\nSentryErrorHandler.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: SentryErrorHandler, deps: [{ token: 'errorHandlerOptions' }], target: i0.ɵɵFactoryTarget.Injectable });\nSentryErrorHandler.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: SentryErrorHandler, providedIn: 'root' });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: SentryErrorHandler, decorators: [{\n type: Injectable,\n args: [{ providedIn: 'root' }]\n }], ctorParameters: function () { return [{ type: undefined, decorators: [{\n type: Inject,\n args: ['errorHandlerOptions']\n }] }]; } });\n/**\n * Factory function that creates an instance of a preconfigured ErrorHandler provider.\n */\nfunction createErrorHandler(config) {\n return new SentryErrorHandler(config);\n}\n\nconst ANGULAR_ROUTING_OP = 'ui.angular.routing';\nconst ANGULAR_INIT_OP = 'ui.angular.init';\nconst ANGULAR_OP = 'ui.angular';\n\nlet instrumentationInitialized;\nlet stashedStartTransaction;\nlet stashedStartTransactionOnLocationChange;\nlet hooksBasedInstrumentation = false;\n/**\n * Creates routing instrumentation for Angular Router.\n *\n * @deprecated Use `browserTracingIntegration()` instead, which includes Angular-specific instrumentation out of the box.\n */\nfunction routingInstrumentation(customStartTransaction, startTransactionOnPageLoad = true, startTransactionOnLocationChange = true) {\n instrumentationInitialized = true;\n stashedStartTransaction = customStartTransaction;\n stashedStartTransactionOnLocationChange = startTransactionOnLocationChange;\n if (startTransactionOnPageLoad && WINDOW && WINDOW.location) {\n customStartTransaction({\n name: WINDOW.location.pathname,\n op: 'pageload',\n origin: 'auto.pageload.angular',\n attributes: {\n [SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: 'url',\n },\n });\n }\n}\n/**\n * Creates routing instrumentation for Angular Router.\n *\n * @deprecated Use `browserTracingIntegration()` instead, which includes Angular-specific instrumentation out of the box.\n */\n// eslint-disable-next-line deprecation/deprecation\nconst instrumentAngularRouting = routingInstrumentation;\n/**\n * A custom BrowserTracing integration for Angular.\n *\n * Use this integration in combination with `TraceService`\n */\nfunction browserTracingIntegration(options = {}) {\n // If the user opts out to set this up, we just don't initialize this.\n // That way, the TraceService will not actually do anything, functionally disabling this.\n if (options.instrumentNavigation !== false) {\n instrumentationInitialized = true;\n hooksBasedInstrumentation = true;\n }\n return browserTracingIntegration$1(Object.assign(Object.assign({}, options), { instrumentNavigation: false }));\n}\n/**\n * Grabs active transaction off scope.\n *\n * @deprecated You should not rely on the transaction, but just use `startSpan()` APIs instead.\n */\nfunction getActiveTransaction() {\n // eslint-disable-next-line deprecation/deprecation\n return getCurrentScope().getTransaction();\n}\n/**\n * Angular's Service responsible for hooking into Angular Router and tracking current navigation process.\n * Creates a new transaction for every route change and measures a duration of routing process.\n */\nclass TraceService {\n constructor(_router) {\n this._router = _router;\n this.navStart$ = this._router.events.pipe(filter((event) => event instanceof NavigationStart), tap(navigationEvent => {\n if (!instrumentationInitialized) {\n IS_DEBUG_BUILD &&\n logger.error('Angular integration has tracing enabled, but Tracing integration is not configured');\n return;\n }\n if (this._routingSpan) {\n this._routingSpan.end();\n this._routingSpan = null;\n }\n const client = getClient();\n const strippedUrl = stripUrlQueryAndFragment(navigationEvent.url);\n if (client && hooksBasedInstrumentation) {\n if (!getActiveSpan()) {\n startBrowserTracingNavigationSpan(client, {\n name: strippedUrl,\n origin: 'auto.navigation.angular',\n attributes: {\n [SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: 'url',\n },\n });\n }\n // eslint-disable-next-line deprecation/deprecation\n this._routingSpan =\n startInactiveSpan({\n name: `${navigationEvent.url}`,\n op: ANGULAR_ROUTING_OP,\n origin: 'auto.ui.angular',\n tags: Object.assign({ 'routing.instrumentation': '@sentry/angular', url: strippedUrl }, (navigationEvent.navigationTrigger && {\n navigationTrigger: navigationEvent.navigationTrigger,\n })),\n }) || null;\n return;\n }\n // eslint-disable-next-line deprecation/deprecation\n let activeTransaction = getActiveTransaction();\n if (!activeTransaction && stashedStartTransactionOnLocationChange) {\n activeTransaction = stashedStartTransaction({\n name: strippedUrl,\n op: 'navigation',\n origin: 'auto.navigation.angular',\n attributes: {\n [SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: 'url',\n },\n });\n }\n if (activeTransaction) {\n // eslint-disable-next-line deprecation/deprecation\n this._routingSpan = activeTransaction.startChild({\n description: `${navigationEvent.url}`,\n op: ANGULAR_ROUTING_OP,\n origin: 'auto.ui.angular',\n tags: Object.assign({ 'routing.instrumentation': '@sentry/angular', url: strippedUrl }, (navigationEvent.navigationTrigger && {\n navigationTrigger: navigationEvent.navigationTrigger,\n })),\n });\n }\n }));\n // The ResolveEnd event is fired when the Angular router has resolved the URL and\n // the parameter<->value mapping. It holds the new resolved router state with\n // the mapping and the new URL.\n // Only After this event, the route is activated, meaning that the transaction\n // can be updated with the parameterized route name before e.g. the route's root\n // component is initialized. This should be early enough before outgoing requests\n // are made from the new route, with the exceptions of requests being made during\n // a navigation.\n this.resEnd$ = this._router.events.pipe(filter((event) => event instanceof ResolveEnd), tap(event => {\n const route = getParameterizedRouteFromSnapshot(event.state.root);\n // eslint-disable-next-line deprecation/deprecation\n const transaction = getActiveTransaction();\n // TODO (v8 / #5416): revisit the source condition. Do we want to make the parameterized route the default?\n const attributes = (transaction && spanToJSON(transaction).data) || {};\n if (transaction && attributes[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE] === 'url') {\n transaction.updateName(route);\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, 'route');\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_ORIGIN, `auto.${spanToJSON(transaction).op}.angular`);\n }\n }));\n this.navEnd$ = this._router.events.pipe(filter(event => event instanceof NavigationEnd || event instanceof NavigationCancel || event instanceof NavigationError), tap(() => {\n if (this._routingSpan) {\n runOutsideAngular(() => {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n this._routingSpan.end();\n });\n this._routingSpan = null;\n }\n }));\n this._routingSpan = null;\n this._subscription = new Subscription();\n this._subscription.add(this.navStart$.subscribe());\n this._subscription.add(this.resEnd$.subscribe());\n this._subscription.add(this.navEnd$.subscribe());\n }\n /**\n * This is used to prevent memory leaks when the root view is created and destroyed multiple times,\n * since `subscribe` callbacks capture `this` and prevent many resources from being GC'd.\n */\n ngOnDestroy() {\n this._subscription.unsubscribe();\n }\n}\nTraceService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceService, deps: [{ token: i1.Router }], target: i0.ɵɵFactoryTarget.Injectable });\nTraceService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceService, providedIn: 'root' });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceService, decorators: [{\n type: Injectable,\n args: [{ providedIn: 'root' }]\n }], ctorParameters: function () { return [{ type: i1.Router }]; } });\nconst UNKNOWN_COMPONENT = 'unknown';\n/**\n * A directive that can be used to capture initialization lifecycle of the whole component.\n */\nclass TraceDirective {\n /**\n * Implementation of OnInit lifecycle method\n * @inheritdoc\n */\n ngOnInit() {\n if (!this.componentName) {\n this.componentName = UNKNOWN_COMPONENT;\n }\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction();\n if (activeTransaction) {\n // eslint-disable-next-line deprecation/deprecation\n this._tracingSpan = activeTransaction.startChild({\n description: `<${this.componentName}>`,\n op: ANGULAR_INIT_OP,\n origin: 'auto.ui.angular.trace_directive',\n });\n }\n }\n /**\n * Implementation of AfterViewInit lifecycle method\n * @inheritdoc\n */\n ngAfterViewInit() {\n if (this._tracingSpan) {\n this._tracingSpan.end();\n }\n }\n}\nTraceDirective.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });\nTraceDirective.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"12.0.0\", version: \"12.2.17\", type: TraceDirective, selector: \"[trace]\", inputs: { componentName: [\"trace\", \"componentName\"] }, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceDirective, decorators: [{\n type: Directive,\n args: [{ selector: '[trace]' }]\n }], propDecorators: { componentName: [{\n type: Input,\n args: ['trace']\n }] } });\n/**\n * A module serves as a single compilation unit for the `TraceDirective` and can be re-used by any other module.\n */\nclass TraceModule {\n}\nTraceModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });\nTraceModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceModule, declarations: [TraceDirective], exports: [TraceDirective] });\nTraceModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceModule });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"12.2.17\", ngImport: i0, type: TraceModule, decorators: [{\n type: NgModule,\n args: [{\n declarations: [TraceDirective],\n exports: [TraceDirective],\n }]\n }] });\n/**\n * Decorator function that can be used to capture initialization lifecycle of the whole component.\n */\nfunction TraceClassDecorator() {\n let tracingSpan;\n /* eslint-disable @typescript-eslint/no-unsafe-member-access */\n // eslint-disable-next-line @typescript-eslint/explicit-function-return-type\n return target => {\n const originalOnInit = target.prototype.ngOnInit;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n target.prototype.ngOnInit = function (...args) {\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction();\n if (activeTransaction) {\n // eslint-disable-next-line deprecation/deprecation\n tracingSpan = activeTransaction.startChild({\n description: `<${target.name}>`,\n op: ANGULAR_INIT_OP,\n origin: 'auto.ui.angular.trace_class_decorator',\n });\n }\n if (originalOnInit) {\n return originalOnInit.apply(this, args);\n }\n };\n const originalAfterViewInit = target.prototype.ngAfterViewInit;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n target.prototype.ngAfterViewInit = function (...args) {\n if (tracingSpan) {\n tracingSpan.end();\n }\n if (originalAfterViewInit) {\n return originalAfterViewInit.apply(this, args);\n }\n };\n };\n /* eslint-enable @typescript-eslint/no-unsafe-member-access */\n}\n/**\n * Decorator function that can be used to capture a single lifecycle methods of the component.\n */\nfunction TraceMethodDecorator() {\n // eslint-disable-next-line @typescript-eslint/explicit-function-return-type, @typescript-eslint/ban-types\n return (target, propertyKey, descriptor) => {\n const originalMethod = descriptor.value;\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n descriptor.value = function (...args) {\n const now = timestampInSeconds();\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction();\n if (activeTransaction) {\n // eslint-disable-next-line deprecation/deprecation\n activeTransaction.startChild({\n description: `<${target.constructor.name}>`,\n endTimestamp: now,\n op: `${ANGULAR_OP}.${String(propertyKey)}`,\n origin: 'auto.ui.angular.trace_method_decorator',\n startTimestamp: now,\n });\n }\n if (originalMethod) {\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return originalMethod.apply(this, args);\n }\n };\n return descriptor;\n };\n}\n/**\n * Takes the parameterized route from a given ActivatedRouteSnapshot and concatenates the snapshot's\n * child route with its parent to produce the complete parameterized URL of the activated route.\n * This happens recursively until the last child (i.e. the end of the URL) is reached.\n *\n * @param route the ActivatedRouteSnapshot of which its path and its child's path is concatenated\n *\n * @returns the concatenated parameterized route string\n */\nfunction getParameterizedRouteFromSnapshot(route) {\n const parts = [];\n let currentRoute = route && route.firstChild;\n while (currentRoute) {\n const path = currentRoute && currentRoute.routeConfig && currentRoute.routeConfig.path;\n if (path === null || path === undefined) {\n break;\n }\n parts.push(path);\n currentRoute = currentRoute.firstChild;\n }\n const fullPath = parts.filter(part => part).join('/');\n return fullPath ? `/${fullPath}/` : '/';\n}\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { SentryErrorHandler, TraceClassDecorator, TraceDirective, TraceMethodDecorator, TraceModule, TraceService, browserTracingIntegration, createErrorHandler, getActiveTransaction, init, instrumentAngularRouting, routingInstrumentation };\n","import { extractTraceparentData as extractTraceparentData$1 } from '@sentry/utils';\nexport { stripUrlQueryAndFragment } from '@sentry/utils';\nimport { getCurrentHub } from '../hub.js';\n\n/**\n * Grabs active transaction off scope.\n *\n * @deprecated You should not rely on the transaction, but just use `startSpan()` APIs instead.\n */\n// eslint-disable-next-line deprecation/deprecation\nfunction getActiveTransaction(maybeHub) {\n // eslint-disable-next-line deprecation/deprecation\n const hub = maybeHub || getCurrentHub();\n // eslint-disable-next-line deprecation/deprecation\n const scope = hub.getScope();\n // eslint-disable-next-line deprecation/deprecation\n return scope.getTransaction() ;\n}\n\n/**\n * The `extractTraceparentData` function and `TRACEPARENT_REGEXP` constant used\n * to be declared in this file. It was later moved into `@sentry/utils` as part of a\n * move to remove `@sentry/tracing` dependencies from `@sentry/node` (`extractTraceparentData`\n * is the only tracing function used by `@sentry/node`).\n *\n * These exports are kept here for backwards compatability's sake.\n *\n * See https://github.com/getsentry/sentry-javascript/issues/4642 for more details.\n *\n * @deprecated Import this function from `@sentry/utils` instead\n */\nconst extractTraceparentData = extractTraceparentData$1;\n\nexport { extractTraceparentData, getActiveTransaction };\n","import { addGlobalErrorInstrumentationHandler, addGlobalUnhandledRejectionInstrumentationHandler, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { getActiveTransaction } from './utils.js';\n\nlet errorsInstrumented = false;\n\n/**\n * Configures global error listeners\n */\nfunction registerErrorInstrumentation() {\n if (errorsInstrumented) {\n return;\n }\n\n errorsInstrumented = true;\n addGlobalErrorInstrumentationHandler(errorCallback);\n addGlobalUnhandledRejectionInstrumentationHandler(errorCallback);\n}\n\n/**\n * If an error or unhandled promise occurs, we mark the active transaction as failed\n */\nfunction errorCallback() {\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction();\n if (activeTransaction) {\n const status = 'internal_error';\n DEBUG_BUILD && logger.log(`[Tracing] Transaction: ${status} -> Global error occured`);\n activeTransaction.setStatus(status);\n }\n}\n\n// The function name will be lost when bundling but we need to be able to identify this listener later to maintain the\n// node.js default exit behaviour\nerrorCallback.tag = 'sentry_tracingErrorCallback';\n\nexport { registerErrorInstrumentation };\n","import { isNaN, logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE } from '../semanticAttributes.js';\nimport { hasTracingEnabled } from '../utils/hasTracingEnabled.js';\nimport { spanToJSON } from '../utils/spanUtils.js';\n\n/**\n * Makes a sampling decision for the given transaction and stores it on the transaction.\n *\n * Called every time a transaction is created. Only transactions which emerge with a `sampled` value of `true` will be\n * sent to Sentry.\n *\n * This method muttes the given `transaction` and will set the `sampled` value on it.\n * It returns the same transaction, for convenience.\n */\nfunction sampleTransaction(\n transaction,\n options,\n samplingContext,\n) {\n // nothing to do if tracing is not enabled\n if (!hasTracingEnabled(options)) {\n // eslint-disable-next-line deprecation/deprecation\n transaction.sampled = false;\n return transaction;\n }\n\n // if the user has forced a sampling decision by passing a `sampled` value in their transaction context, go with that\n // eslint-disable-next-line deprecation/deprecation\n if (transaction.sampled !== undefined) {\n // eslint-disable-next-line deprecation/deprecation\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE, Number(transaction.sampled));\n return transaction;\n }\n\n // we would have bailed already if neither `tracesSampler` nor `tracesSampleRate` nor `enableTracing` were defined, so one of these should\n // work; prefer the hook if so\n let sampleRate;\n if (typeof options.tracesSampler === 'function') {\n sampleRate = options.tracesSampler(samplingContext);\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE, Number(sampleRate));\n } else if (samplingContext.parentSampled !== undefined) {\n sampleRate = samplingContext.parentSampled;\n } else if (typeof options.tracesSampleRate !== 'undefined') {\n sampleRate = options.tracesSampleRate;\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE, Number(sampleRate));\n } else {\n // When `enableTracing === true`, we use a sample rate of 100%\n sampleRate = 1;\n transaction.setAttribute(SEMANTIC_ATTRIBUTE_SENTRY_SAMPLE_RATE, sampleRate);\n }\n\n // Since this is coming from the user (or from a function provided by the user), who knows what we might get. (The\n // only valid values are booleans or numbers between 0 and 1.)\n if (!isValidSampleRate(sampleRate)) {\n DEBUG_BUILD && logger.warn('[Tracing] Discarding transaction because of invalid sample rate.');\n // eslint-disable-next-line deprecation/deprecation\n transaction.sampled = false;\n return transaction;\n }\n\n // if the function returned 0 (or false), or if `tracesSampleRate` is 0, it's a sign the transaction should be dropped\n if (!sampleRate) {\n DEBUG_BUILD &&\n logger.log(\n `[Tracing] Discarding transaction because ${\n typeof options.tracesSampler === 'function'\n ? 'tracesSampler returned 0 or false'\n : 'a negative sampling decision was inherited or tracesSampleRate is set to 0'\n }`,\n );\n // eslint-disable-next-line deprecation/deprecation\n transaction.sampled = false;\n return transaction;\n }\n\n // Now we roll the dice. Math.random is inclusive of 0, but not of 1, so strict < is safe here. In case sampleRate is\n // a boolean, the < comparison will cause it to be automatically cast to 1 if it's true and 0 if it's false.\n // eslint-disable-next-line deprecation/deprecation\n transaction.sampled = Math.random() < (sampleRate );\n\n // if we're not going to keep it, we're done\n // eslint-disable-next-line deprecation/deprecation\n if (!transaction.sampled) {\n DEBUG_BUILD &&\n logger.log(\n `[Tracing] Discarding transaction because it's not included in the random sample (sampling rate = ${Number(\n sampleRate,\n )})`,\n );\n return transaction;\n }\n\n DEBUG_BUILD &&\n // eslint-disable-next-line deprecation/deprecation\n logger.log(`[Tracing] starting ${transaction.op} transaction - ${spanToJSON(transaction).description}`);\n return transaction;\n}\n\n/**\n * Checks the given sample rate to make sure it is valid type and value (a boolean, or a number between 0 and 1).\n */\nfunction isValidSampleRate(rate) {\n // we need to check NaN explicitly because it's of type 'number' and therefore wouldn't get caught by this typecheck\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n if (isNaN(rate) || !(typeof rate === 'number' || typeof rate === 'boolean')) {\n DEBUG_BUILD &&\n logger.warn(\n `[Tracing] Given sample rate is invalid. Sample rate must be a boolean or a number between 0 and 1. Got ${JSON.stringify(\n rate,\n )} of type ${JSON.stringify(typeof rate)}.`,\n );\n return false;\n }\n\n // in case sampleRate is a boolean, it will get automatically cast to 1 if it's true and 0 if it's false\n if (rate < 0 || rate > 1) {\n DEBUG_BUILD &&\n logger.warn(`[Tracing] Given sample rate is invalid. Sample rate must be between 0 and 1. Got ${rate}.`);\n return false;\n }\n return true;\n}\n\nexport { isValidSampleRate, sampleTransaction };\n","import { logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../debug-build.js';\nimport { getMainCarrier } from '../hub.js';\nimport { spanToTraceHeader } from '../utils/spanUtils.js';\nimport { registerErrorInstrumentation } from './errors.js';\nimport { IdleTransaction } from './idletransaction.js';\nimport { sampleTransaction } from './sampling.js';\nimport { Transaction } from './transaction.js';\n\n/** Returns all trace headers that are currently on the top scope. */\n// eslint-disable-next-line deprecation/deprecation\nfunction traceHeaders() {\n // eslint-disable-next-line deprecation/deprecation\n const scope = this.getScope();\n // eslint-disable-next-line deprecation/deprecation\n const span = scope.getSpan();\n\n return span\n ? {\n 'sentry-trace': spanToTraceHeader(span),\n }\n : {};\n}\n\n/**\n * Creates a new transaction and adds a sampling decision if it doesn't yet have one.\n *\n * The Hub.startTransaction method delegates to this method to do its work, passing the Hub instance in as `this`, as if\n * it had been called on the hub directly. Exists as a separate function so that it can be injected into the class as an\n * \"extension method.\"\n *\n * @param this: The Hub starting the transaction\n * @param transactionContext: Data used to configure the transaction\n * @param CustomSamplingContext: Optional data to be provided to the `tracesSampler` function (if any)\n *\n * @returns The new transaction\n *\n * @see {@link Hub.startTransaction}\n */\nfunction _startTransaction(\n // eslint-disable-next-line deprecation/deprecation\n\n transactionContext,\n customSamplingContext,\n) {\n // eslint-disable-next-line deprecation/deprecation\n const client = this.getClient();\n const options = (client && client.getOptions()) || {};\n\n const configInstrumenter = options.instrumenter || 'sentry';\n const transactionInstrumenter = transactionContext.instrumenter || 'sentry';\n\n if (configInstrumenter !== transactionInstrumenter) {\n DEBUG_BUILD &&\n logger.error(\n `A transaction was started with instrumenter=\\`${transactionInstrumenter}\\`, but the SDK is configured with the \\`${configInstrumenter}\\` instrumenter.\nThe transaction will not be sampled. Please use the ${configInstrumenter} instrumentation to start transactions.`,\n );\n\n // eslint-disable-next-line deprecation/deprecation\n transactionContext.sampled = false;\n }\n\n // eslint-disable-next-line deprecation/deprecation\n let transaction = new Transaction(transactionContext, this);\n transaction = sampleTransaction(transaction, options, {\n name: transactionContext.name,\n parentSampled: transactionContext.parentSampled,\n transactionContext,\n attributes: {\n // eslint-disable-next-line deprecation/deprecation\n ...transactionContext.data,\n ...transactionContext.attributes,\n },\n ...customSamplingContext,\n });\n if (transaction.isRecording()) {\n transaction.initSpanRecorder(options._experiments && (options._experiments.maxSpans ));\n }\n if (client && client.emit) {\n client.emit('startTransaction', transaction);\n }\n return transaction;\n}\n\n/**\n * Create new idle transaction.\n */\nfunction startIdleTransaction(\n // eslint-disable-next-line deprecation/deprecation\n hub,\n transactionContext,\n idleTimeout,\n finalTimeout,\n onScope,\n customSamplingContext,\n heartbeatInterval,\n delayAutoFinishUntilSignal = false,\n) {\n // eslint-disable-next-line deprecation/deprecation\n const client = hub.getClient();\n const options = (client && client.getOptions()) || {};\n\n // eslint-disable-next-line deprecation/deprecation\n let transaction = new IdleTransaction(\n transactionContext,\n hub,\n idleTimeout,\n finalTimeout,\n heartbeatInterval,\n onScope,\n delayAutoFinishUntilSignal,\n );\n transaction = sampleTransaction(transaction, options, {\n name: transactionContext.name,\n parentSampled: transactionContext.parentSampled,\n transactionContext,\n attributes: {\n // eslint-disable-next-line deprecation/deprecation\n ...transactionContext.data,\n ...transactionContext.attributes,\n },\n ...customSamplingContext,\n });\n if (transaction.isRecording()) {\n transaction.initSpanRecorder(options._experiments && (options._experiments.maxSpans ));\n }\n if (client && client.emit) {\n client.emit('startTransaction', transaction);\n }\n return transaction;\n}\n\n/**\n * Adds tracing extensions to the global hub.\n */\nfunction addTracingExtensions() {\n const carrier = getMainCarrier();\n if (!carrier.__SENTRY__) {\n return;\n }\n carrier.__SENTRY__.extensions = carrier.__SENTRY__.extensions || {};\n if (!carrier.__SENTRY__.extensions.startTransaction) {\n carrier.__SENTRY__.extensions.startTransaction = _startTransaction;\n }\n if (!carrier.__SENTRY__.extensions.traceHeaders) {\n carrier.__SENTRY__.extensions.traceHeaders = traceHeaders;\n }\n\n registerErrorInstrumentation();\n}\n\nexport { addTracingExtensions, startIdleTransaction };\n","import { dsnToString, createEnvelope } from '@sentry/utils';\n\n/**\n * Create envelope from Span item.\n */\nfunction createSpanEnvelope(spans, dsn) {\n const headers = {\n sent_at: new Date().toISOString(),\n };\n\n if (dsn) {\n headers.dsn = dsnToString(dsn);\n }\n\n const items = spans.map(createSpanItem);\n return createEnvelope(headers, items);\n}\n\nfunction createSpanItem(span) {\n const spanHeaders = {\n type: 'span',\n };\n return [spanHeaders, span];\n}\n\nexport { createSpanEnvelope };\n","/**\n * Checks if a given value is a valid measurement value.\n */\nfunction isMeasurementValue(value) {\n return typeof value === 'number' && isFinite(value);\n}\n\n/**\n * Helper function to start child on transactions. This function will make sure that the transaction will\n * use the start timestamp of the created child span if it is earlier than the transactions actual\n * start timestamp.\n *\n * Note: this will not be possible anymore in v8,\n * unless we do some special handling for browser here...\n */\nfunction _startChild(transaction, { startTimestamp, ...ctx }) {\n // eslint-disable-next-line deprecation/deprecation\n if (startTimestamp && transaction.startTimestamp > startTimestamp) {\n // eslint-disable-next-line deprecation/deprecation\n transaction.startTimestamp = startTimestamp;\n }\n\n // eslint-disable-next-line deprecation/deprecation\n return transaction.startChild({\n startTimestamp,\n ...ctx,\n });\n}\n\nexport { _startChild, isMeasurementValue };\n","import { getActiveTransaction, spanToJSON, setMeasurement, getClient, Span, createSpanEnvelope, hasTracingEnabled, isValidSampleRate } from '@sentry/core';\nimport { browserPerformanceTimeOrigin, htmlTreeAsString, getComponentName, logger, parseUrl } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../../common/debug-build.js';\nimport { addPerformanceInstrumentationHandler, addClsInstrumentationHandler, addLcpInstrumentationHandler, addFidInstrumentationHandler, addTtfbInstrumentationHandler, addInpInstrumentationHandler } from '../instrument.js';\nimport { WINDOW } from '../types.js';\nimport { getVisibilityWatcher } from '../web-vitals/lib/getVisibilityWatcher.js';\nimport { _startChild, isMeasurementValue } from './utils.js';\nimport { getNavigationEntry } from '../web-vitals/lib/getNavigationEntry.js';\n\nconst MAX_INT_AS_BYTES = 2147483647;\n\n/**\n * Converts from milliseconds to seconds\n * @param time time in ms\n */\nfunction msToSec(time) {\n return time / 1000;\n}\n\nfunction getBrowserPerformanceAPI() {\n // @ts-expect-error we want to make sure all of these are available, even if TS is sure they are\n return WINDOW && WINDOW.addEventListener && WINDOW.performance;\n}\n\nlet _performanceCursor = 0;\n\nlet _measurements = {};\nlet _lcpEntry;\nlet _clsEntry;\n\n/**\n * Start tracking web vitals.\n * The callback returned by this function can be used to stop tracking & ensure all measurements are final & captured.\n *\n * @returns A function that forces web vitals collection\n */\nfunction startTrackingWebVitals() {\n const performance = getBrowserPerformanceAPI();\n if (performance && browserPerformanceTimeOrigin) {\n // @ts-expect-error we want to make sure all of these are available, even if TS is sure they are\n if (performance.mark) {\n WINDOW.performance.mark('sentry-tracing-init');\n }\n const fidCallback = _trackFID();\n const clsCallback = _trackCLS();\n const lcpCallback = _trackLCP();\n const ttfbCallback = _trackTtfb();\n\n return () => {\n fidCallback();\n clsCallback();\n lcpCallback();\n ttfbCallback();\n };\n }\n\n return () => undefined;\n}\n\n/**\n * Start tracking long tasks.\n */\nfunction startTrackingLongTasks() {\n addPerformanceInstrumentationHandler('longtask', ({ entries }) => {\n for (const entry of entries) {\n // eslint-disable-next-line deprecation/deprecation\n const transaction = getActiveTransaction() ;\n if (!transaction) {\n return;\n }\n const startTime = msToSec((browserPerformanceTimeOrigin ) + entry.startTime);\n const duration = msToSec(entry.duration);\n\n // eslint-disable-next-line deprecation/deprecation\n transaction.startChild({\n description: 'Main UI thread blocked',\n op: 'ui.long-task',\n origin: 'auto.ui.browser.metrics',\n startTimestamp: startTime,\n endTimestamp: startTime + duration,\n });\n }\n });\n}\n\n/**\n * Start tracking interaction events.\n */\nfunction startTrackingInteractions() {\n addPerformanceInstrumentationHandler('event', ({ entries }) => {\n for (const entry of entries) {\n // eslint-disable-next-line deprecation/deprecation\n const transaction = getActiveTransaction() ;\n if (!transaction) {\n return;\n }\n\n if (entry.name === 'click') {\n const startTime = msToSec((browserPerformanceTimeOrigin ) + entry.startTime);\n const duration = msToSec(entry.duration);\n\n const span = {\n description: htmlTreeAsString(entry.target),\n op: `ui.interaction.${entry.name}`,\n origin: 'auto.ui.browser.metrics',\n startTimestamp: startTime,\n endTimestamp: startTime + duration,\n };\n\n const componentName = getComponentName(entry.target);\n if (componentName) {\n span.attributes = { 'ui.component_name': componentName };\n }\n\n // eslint-disable-next-line deprecation/deprecation\n transaction.startChild(span);\n }\n }\n });\n}\n\n/**\n * Start tracking INP webvital events.\n */\nfunction startTrackingINP(\n interactionIdtoRouteNameMapping,\n interactionsSampleRate,\n) {\n const performance = getBrowserPerformanceAPI();\n if (performance && browserPerformanceTimeOrigin) {\n const inpCallback = _trackINP(interactionIdtoRouteNameMapping, interactionsSampleRate);\n\n return () => {\n inpCallback();\n };\n }\n\n return () => undefined;\n}\n\n/** Starts tracking the Cumulative Layout Shift on the current page. */\nfunction _trackCLS() {\n return addClsInstrumentationHandler(({ metric }) => {\n const entry = metric.entries[metric.entries.length - 1];\n if (!entry) {\n return;\n }\n\n DEBUG_BUILD && logger.log('[Measurements] Adding CLS');\n _measurements['cls'] = { value: metric.value, unit: '' };\n _clsEntry = entry ;\n }, true);\n}\n\n/** Starts tracking the Largest Contentful Paint on the current page. */\nfunction _trackLCP() {\n return addLcpInstrumentationHandler(({ metric }) => {\n const entry = metric.entries[metric.entries.length - 1];\n if (!entry) {\n return;\n }\n\n DEBUG_BUILD && logger.log('[Measurements] Adding LCP');\n _measurements['lcp'] = { value: metric.value, unit: 'millisecond' };\n _lcpEntry = entry ;\n }, true);\n}\n\n/** Starts tracking the First Input Delay on the current page. */\nfunction _trackFID() {\n return addFidInstrumentationHandler(({ metric }) => {\n const entry = metric.entries[metric.entries.length - 1];\n if (!entry) {\n return;\n }\n\n const timeOrigin = msToSec(browserPerformanceTimeOrigin );\n const startTime = msToSec(entry.startTime);\n DEBUG_BUILD && logger.log('[Measurements] Adding FID');\n _measurements['fid'] = { value: metric.value, unit: 'millisecond' };\n _measurements['mark.fid'] = { value: timeOrigin + startTime, unit: 'second' };\n });\n}\n\nfunction _trackTtfb() {\n return addTtfbInstrumentationHandler(({ metric }) => {\n const entry = metric.entries[metric.entries.length - 1];\n if (!entry) {\n return;\n }\n\n DEBUG_BUILD && logger.log('[Measurements] Adding TTFB');\n _measurements['ttfb'] = { value: metric.value, unit: 'millisecond' };\n });\n}\n\nconst INP_ENTRY_MAP = {\n click: 'click',\n pointerdown: 'click',\n pointerup: 'click',\n mousedown: 'click',\n mouseup: 'click',\n touchstart: 'click',\n touchend: 'click',\n mouseover: 'hover',\n mouseout: 'hover',\n mouseenter: 'hover',\n mouseleave: 'hover',\n pointerover: 'hover',\n pointerout: 'hover',\n pointerenter: 'hover',\n pointerleave: 'hover',\n dragstart: 'drag',\n dragend: 'drag',\n drag: 'drag',\n dragenter: 'drag',\n dragleave: 'drag',\n dragover: 'drag',\n drop: 'drag',\n keydown: 'press',\n keyup: 'press',\n keypress: 'press',\n input: 'press',\n};\n\n/** Starts tracking the Interaction to Next Paint on the current page. */\nfunction _trackINP(\n interactionIdToRouteNameMapping,\n interactionsSampleRate,\n) {\n return addInpInstrumentationHandler(({ metric }) => {\n if (metric.value === undefined) {\n return;\n }\n const entry = metric.entries.find(\n entry => entry.duration === metric.value && INP_ENTRY_MAP[entry.name] !== undefined,\n );\n const client = getClient();\n if (!entry || !client) {\n return;\n }\n const interactionType = INP_ENTRY_MAP[entry.name];\n const options = client.getOptions();\n /** Build the INP span, create an envelope from the span, and then send the envelope */\n const startTime = msToSec((browserPerformanceTimeOrigin ) + entry.startTime);\n const duration = msToSec(metric.value);\n const interaction =\n entry.interactionId !== undefined ? interactionIdToRouteNameMapping[entry.interactionId] : undefined;\n if (interaction === undefined) {\n return;\n }\n const { routeName, parentContext, activeTransaction, user, replayId } = interaction;\n const userDisplay = user !== undefined ? user.email || user.id || user.ip_address : undefined;\n // eslint-disable-next-line deprecation/deprecation\n const profileId = activeTransaction !== undefined ? activeTransaction.getProfileId() : undefined;\n const span = new Span({\n startTimestamp: startTime,\n endTimestamp: startTime + duration,\n op: `ui.interaction.${interactionType}`,\n name: htmlTreeAsString(entry.target),\n attributes: {\n release: options.release,\n environment: options.environment,\n transaction: routeName,\n ...(userDisplay !== undefined && userDisplay !== '' ? { user: userDisplay } : {}),\n ...(profileId !== undefined ? { profile_id: profileId } : {}),\n ...(replayId !== undefined ? { replay_id: replayId } : {}),\n },\n exclusiveTime: metric.value,\n measurements: {\n inp: { value: metric.value, unit: 'millisecond' },\n },\n });\n\n /** Check to see if the span should be sampled */\n const sampleRate = getSampleRate(parentContext, options, interactionsSampleRate);\n\n if (!sampleRate) {\n return;\n }\n\n if (Math.random() < (sampleRate )) {\n const envelope = span ? createSpanEnvelope([span], client.getDsn()) : undefined;\n const transport = client && client.getTransport();\n if (transport && envelope) {\n transport.send(envelope).then(null, reason => {\n DEBUG_BUILD && logger.error('Error while sending interaction:', reason);\n });\n }\n return;\n }\n });\n}\n\n/** Add performance related spans to a transaction */\nfunction addPerformanceEntries(transaction) {\n const performance = getBrowserPerformanceAPI();\n if (!performance || !WINDOW.performance.getEntries || !browserPerformanceTimeOrigin) {\n // Gatekeeper if performance API not available\n return;\n }\n\n DEBUG_BUILD && logger.log('[Tracing] Adding & adjusting spans using Performance API');\n const timeOrigin = msToSec(browserPerformanceTimeOrigin);\n\n const performanceEntries = performance.getEntries();\n\n const { op, start_timestamp: transactionStartTime } = spanToJSON(transaction);\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n performanceEntries.slice(_performanceCursor).forEach((entry) => {\n const startTime = msToSec(entry.startTime);\n const duration = msToSec(entry.duration);\n\n // eslint-disable-next-line deprecation/deprecation\n if (transaction.op === 'navigation' && transactionStartTime && timeOrigin + startTime < transactionStartTime) {\n return;\n }\n\n switch (entry.entryType) {\n case 'navigation': {\n _addNavigationSpans(transaction, entry, timeOrigin);\n break;\n }\n case 'mark':\n case 'paint':\n case 'measure': {\n _addMeasureSpans(transaction, entry, startTime, duration, timeOrigin);\n\n // capture web vitals\n const firstHidden = getVisibilityWatcher();\n // Only report if the page wasn't hidden prior to the web vital.\n const shouldRecord = entry.startTime < firstHidden.firstHiddenTime;\n\n if (entry.name === 'first-paint' && shouldRecord) {\n DEBUG_BUILD && logger.log('[Measurements] Adding FP');\n _measurements['fp'] = { value: entry.startTime, unit: 'millisecond' };\n }\n if (entry.name === 'first-contentful-paint' && shouldRecord) {\n DEBUG_BUILD && logger.log('[Measurements] Adding FCP');\n _measurements['fcp'] = { value: entry.startTime, unit: 'millisecond' };\n }\n break;\n }\n case 'resource': {\n _addResourceSpans(transaction, entry, entry.name , startTime, duration, timeOrigin);\n break;\n }\n // Ignore other entry types.\n }\n });\n\n _performanceCursor = Math.max(performanceEntries.length - 1, 0);\n\n _trackNavigator(transaction);\n\n // Measurements are only available for pageload transactions\n if (op === 'pageload') {\n _addTtfbRequestTimeToMeasurements(_measurements);\n\n ['fcp', 'fp', 'lcp'].forEach(name => {\n if (!_measurements[name] || !transactionStartTime || timeOrigin >= transactionStartTime) {\n return;\n }\n // The web vitals, fcp, fp, lcp, and ttfb, all measure relative to timeOrigin.\n // Unfortunately, timeOrigin is not captured within the transaction span data, so these web vitals will need\n // to be adjusted to be relative to transaction.startTimestamp.\n const oldValue = _measurements[name].value;\n const measurementTimestamp = timeOrigin + msToSec(oldValue);\n\n // normalizedValue should be in milliseconds\n const normalizedValue = Math.abs((measurementTimestamp - transactionStartTime) * 1000);\n const delta = normalizedValue - oldValue;\n\n DEBUG_BUILD && logger.log(`[Measurements] Normalized ${name} from ${oldValue} to ${normalizedValue} (${delta})`);\n _measurements[name].value = normalizedValue;\n });\n\n const fidMark = _measurements['mark.fid'];\n if (fidMark && _measurements['fid']) {\n // create span for FID\n _startChild(transaction, {\n description: 'first input delay',\n endTimestamp: fidMark.value + msToSec(_measurements['fid'].value),\n op: 'ui.action',\n origin: 'auto.ui.browser.metrics',\n startTimestamp: fidMark.value,\n });\n\n // Delete mark.fid as we don't want it to be part of final payload\n delete _measurements['mark.fid'];\n }\n\n // If FCP is not recorded we should not record the cls value\n // according to the new definition of CLS.\n if (!('fcp' in _measurements)) {\n delete _measurements.cls;\n }\n\n Object.keys(_measurements).forEach(measurementName => {\n setMeasurement(measurementName, _measurements[measurementName].value, _measurements[measurementName].unit);\n });\n\n _tagMetricInfo(transaction);\n }\n\n _lcpEntry = undefined;\n _clsEntry = undefined;\n _measurements = {};\n}\n\n/** Create measure related spans */\nfunction _addMeasureSpans(\n transaction,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n entry,\n startTime,\n duration,\n timeOrigin,\n) {\n const measureStartTimestamp = timeOrigin + startTime;\n const measureEndTimestamp = measureStartTimestamp + duration;\n\n _startChild(transaction, {\n description: entry.name ,\n endTimestamp: measureEndTimestamp,\n op: entry.entryType ,\n origin: 'auto.resource.browser.metrics',\n startTimestamp: measureStartTimestamp,\n });\n\n return measureStartTimestamp;\n}\n\n/** Instrument navigation entries */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction _addNavigationSpans(transaction, entry, timeOrigin) {\n ['unloadEvent', 'redirect', 'domContentLoadedEvent', 'loadEvent', 'connect'].forEach(event => {\n _addPerformanceNavigationTiming(transaction, entry, event, timeOrigin);\n });\n _addPerformanceNavigationTiming(transaction, entry, 'secureConnection', timeOrigin, 'TLS/SSL', 'connectEnd');\n _addPerformanceNavigationTiming(transaction, entry, 'fetch', timeOrigin, 'cache', 'domainLookupStart');\n _addPerformanceNavigationTiming(transaction, entry, 'domainLookup', timeOrigin, 'DNS');\n _addRequest(transaction, entry, timeOrigin);\n}\n\n/** Create performance navigation related spans */\nfunction _addPerformanceNavigationTiming(\n transaction,\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n entry,\n event,\n timeOrigin,\n description,\n eventEnd,\n) {\n const end = eventEnd ? (entry[eventEnd] ) : (entry[`${event}End`] );\n const start = entry[`${event}Start`] ;\n if (!start || !end) {\n return;\n }\n _startChild(transaction, {\n op: 'browser',\n origin: 'auto.browser.browser.metrics',\n description: description || event,\n startTimestamp: timeOrigin + msToSec(start),\n endTimestamp: timeOrigin + msToSec(end),\n });\n}\n\n/** Create request and response related spans */\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\nfunction _addRequest(transaction, entry, timeOrigin) {\n if (entry.responseEnd) {\n // It is possible that we are collecting these metrics when the page hasn't finished loading yet, for example when the HTML slowly streams in.\n // In this case, ie. when the document request hasn't finished yet, `entry.responseEnd` will be 0.\n // In order not to produce faulty spans, where the end timestamp is before the start timestamp, we will only collect\n // these spans when the responseEnd value is available. The backend (Relay) would drop the entire transaction if it contained faulty spans.\n _startChild(transaction, {\n op: 'browser',\n origin: 'auto.browser.browser.metrics',\n description: 'request',\n startTimestamp: timeOrigin + msToSec(entry.requestStart ),\n endTimestamp: timeOrigin + msToSec(entry.responseEnd ),\n });\n\n _startChild(transaction, {\n op: 'browser',\n origin: 'auto.browser.browser.metrics',\n description: 'response',\n startTimestamp: timeOrigin + msToSec(entry.responseStart ),\n endTimestamp: timeOrigin + msToSec(entry.responseEnd ),\n });\n }\n}\n\n/** Create resource-related spans */\nfunction _addResourceSpans(\n transaction,\n entry,\n resourceUrl,\n startTime,\n duration,\n timeOrigin,\n) {\n // we already instrument based on fetch and xhr, so we don't need to\n // duplicate spans here.\n if (entry.initiatorType === 'xmlhttprequest' || entry.initiatorType === 'fetch') {\n return;\n }\n\n const parsedUrl = parseUrl(resourceUrl);\n\n // eslint-disable-next-line @typescript-eslint/no-explicit-any\n const data = {};\n setResourceEntrySizeData(data, entry, 'transferSize', 'http.response_transfer_size');\n setResourceEntrySizeData(data, entry, 'encodedBodySize', 'http.response_content_length');\n setResourceEntrySizeData(data, entry, 'decodedBodySize', 'http.decoded_response_content_length');\n\n if ('renderBlockingStatus' in entry) {\n data['resource.render_blocking_status'] = entry.renderBlockingStatus;\n }\n if (parsedUrl.protocol) {\n data['url.scheme'] = parsedUrl.protocol.split(':').pop(); // the protocol returned by parseUrl includes a :, but OTEL spec does not, so we remove it.\n }\n\n if (parsedUrl.host) {\n data['server.address'] = parsedUrl.host;\n }\n\n data['url.same_origin'] = resourceUrl.includes(WINDOW.location.origin);\n\n const startTimestamp = timeOrigin + startTime;\n const endTimestamp = startTimestamp + duration;\n\n _startChild(transaction, {\n description: resourceUrl.replace(WINDOW.location.origin, ''),\n endTimestamp,\n op: entry.initiatorType ? `resource.${entry.initiatorType}` : 'resource.other',\n origin: 'auto.resource.browser.metrics',\n startTimestamp,\n data,\n });\n}\n\n/**\n * Capture the information of the user agent.\n */\nfunction _trackNavigator(transaction) {\n const navigator = WINDOW.navigator ;\n if (!navigator) {\n return;\n }\n\n // track network connectivity\n const connection = navigator.connection;\n if (connection) {\n if (connection.effectiveType) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('effectiveConnectionType', connection.effectiveType);\n }\n\n if (connection.type) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('connectionType', connection.type);\n }\n\n if (isMeasurementValue(connection.rtt)) {\n _measurements['connection.rtt'] = { value: connection.rtt, unit: 'millisecond' };\n }\n }\n\n if (isMeasurementValue(navigator.deviceMemory)) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('deviceMemory', `${navigator.deviceMemory} GB`);\n }\n\n if (isMeasurementValue(navigator.hardwareConcurrency)) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('hardwareConcurrency', String(navigator.hardwareConcurrency));\n }\n}\n\n/** Add LCP / CLS data to transaction to allow debugging */\nfunction _tagMetricInfo(transaction) {\n if (_lcpEntry) {\n DEBUG_BUILD && logger.log('[Measurements] Adding LCP Data');\n\n // Capture Properties of the LCP element that contributes to the LCP.\n\n if (_lcpEntry.element) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('lcp.element', htmlTreeAsString(_lcpEntry.element));\n }\n\n if (_lcpEntry.id) {\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('lcp.id', _lcpEntry.id);\n }\n\n if (_lcpEntry.url) {\n // Trim URL to the first 200 characters.\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('lcp.url', _lcpEntry.url.trim().slice(0, 200));\n }\n\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag('lcp.size', _lcpEntry.size);\n }\n\n // See: https://developer.mozilla.org/en-US/docs/Web/API/LayoutShift\n if (_clsEntry && _clsEntry.sources) {\n DEBUG_BUILD && logger.log('[Measurements] Adding CLS Data');\n _clsEntry.sources.forEach((source, index) =>\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n transaction.setTag(`cls.source.${index + 1}`, htmlTreeAsString(source.node)),\n );\n }\n}\n\nfunction setResourceEntrySizeData(\n data,\n entry,\n key,\n dataKey,\n) {\n const entryVal = entry[key];\n if (entryVal != null && entryVal < MAX_INT_AS_BYTES) {\n data[dataKey] = entryVal;\n }\n}\n\n/**\n * Add ttfb request time information to measurements.\n *\n * ttfb information is added via vendored web vitals library.\n */\nfunction _addTtfbRequestTimeToMeasurements(_measurements) {\n const navEntry = getNavigationEntry();\n if (!navEntry) {\n return;\n }\n\n const { responseStart, requestStart } = navEntry;\n\n if (requestStart <= responseStart) {\n DEBUG_BUILD && logger.log('[Measurements] Adding TTFB Request Time');\n _measurements['ttfb.requestTime'] = {\n value: responseStart - requestStart,\n unit: 'millisecond',\n };\n }\n}\n\n/** Taken from @sentry/core sampling.ts */\nfunction getSampleRate(\n transactionContext,\n options,\n interactionsSampleRate,\n) {\n if (!hasTracingEnabled(options)) {\n return false;\n }\n let sampleRate;\n if (transactionContext !== undefined && typeof options.tracesSampler === 'function') {\n sampleRate = options.tracesSampler({\n transactionContext,\n name: transactionContext.name,\n parentSampled: transactionContext.parentSampled,\n attributes: {\n // eslint-disable-next-line deprecation/deprecation\n ...transactionContext.data,\n ...transactionContext.attributes,\n },\n location: WINDOW.location,\n });\n } else if (transactionContext !== undefined && transactionContext.sampled !== undefined) {\n sampleRate = transactionContext.sampled;\n } else if (typeof options.tracesSampleRate !== 'undefined') {\n sampleRate = options.tracesSampleRate;\n } else {\n sampleRate = 1;\n }\n if (!isValidSampleRate(sampleRate)) {\n DEBUG_BUILD && logger.warn('[Tracing] Discarding interaction span because of invalid sample rate.');\n return false;\n }\n if (sampleRate === true) {\n return interactionsSampleRate;\n } else if (sampleRate === false) {\n return 0;\n }\n return sampleRate * interactionsSampleRate;\n}\n\nexport { _addMeasureSpans, _addResourceSpans, addPerformanceEntries, startTrackingINP, startTrackingInteractions, startTrackingLongTasks, startTrackingWebVitals };\n","import { TRACING_DEFAULTS, addTracingExtensions, startIdleTransaction, getActiveTransaction, SEMANTIC_ATTRIBUTE_SENTRY_SOURCE, getClient, getCurrentScope } from '@sentry/core';\nimport { logger, propagationContextFromHeaders, getDomElement } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../common/debug-build.js';\nimport { registerBackgroundTabDetection } from './backgroundtab.js';\nimport { addPerformanceInstrumentationHandler } from './instrument.js';\nimport { startTrackingWebVitals, startTrackingINP, startTrackingLongTasks, startTrackingInteractions, addPerformanceEntries } from './metrics/index.js';\nimport { defaultRequestInstrumentationOptions, instrumentOutgoingRequests } from './request.js';\nimport { instrumentRoutingWithDefaults } from './router.js';\nimport { WINDOW } from './types.js';\n\nconst BROWSER_TRACING_INTEGRATION_ID = 'BrowserTracing';\n\n/** Options for Browser Tracing integration */\n\nconst DEFAULT_BROWSER_TRACING_OPTIONS = {\n ...TRACING_DEFAULTS,\n markBackgroundTransactions: true,\n routingInstrumentation: instrumentRoutingWithDefaults,\n startTransactionOnLocationChange: true,\n startTransactionOnPageLoad: true,\n enableLongTask: true,\n enableInp: false,\n interactionsSampleRate: 1,\n _experiments: {},\n ...defaultRequestInstrumentationOptions,\n};\n\n/** We store up to 10 interaction candidates max to cap memory usage. This is the same cap as getINP from web-vitals */\nconst MAX_INTERACTIONS = 10;\n\n/**\n * The Browser Tracing integration automatically instruments browser pageload/navigation\n * actions as transactions, and captures requests, metrics and errors as spans.\n *\n * The integration can be configured with a variety of options, and can be extended to use\n * any routing library. This integration uses {@see IdleTransaction} to create transactions.\n *\n * @deprecated Use `browserTracingIntegration()` instead.\n */\nclass BrowserTracing {\n // This class currently doesn't have a static `id` field like the other integration classes, because it prevented\n // @sentry/tracing from being treeshaken. Tree shakers do not like static fields, because they behave like side effects.\n // TODO: Come up with a better plan, than using static fields on integration classes, and use that plan on all\n // integrations.\n\n /** Browser Tracing integration options */\n\n /**\n * @inheritDoc\n */\n\n // eslint-disable-next-line deprecation/deprecation\n\n constructor(_options) {\n this.name = BROWSER_TRACING_INTEGRATION_ID;\n this._hasSetTracePropagationTargets = false;\n\n addTracingExtensions();\n\n if (DEBUG_BUILD) {\n this._hasSetTracePropagationTargets = !!(\n _options &&\n // eslint-disable-next-line deprecation/deprecation\n (_options.tracePropagationTargets || _options.tracingOrigins)\n );\n }\n\n this.options = {\n ...DEFAULT_BROWSER_TRACING_OPTIONS,\n ..._options,\n };\n\n // Special case: enableLongTask can be set in _experiments\n // TODO (v8): Remove this in v8\n if (this.options._experiments.enableLongTask !== undefined) {\n this.options.enableLongTask = this.options._experiments.enableLongTask;\n }\n\n // TODO (v8): remove this block after tracingOrigins is removed\n // Set tracePropagationTargets to tracingOrigins if specified by the user\n // In case both are specified, tracePropagationTargets takes precedence\n // eslint-disable-next-line deprecation/deprecation\n if (_options && !_options.tracePropagationTargets && _options.tracingOrigins) {\n // eslint-disable-next-line deprecation/deprecation\n this.options.tracePropagationTargets = _options.tracingOrigins;\n }\n\n this._collectWebVitals = startTrackingWebVitals();\n /** Stores a mapping of interactionIds from PerformanceEventTimings to the origin interaction path */\n this._interactionIdToRouteNameMapping = {};\n\n if (this.options.enableInp) {\n startTrackingINP(this._interactionIdToRouteNameMapping, this.options.interactionsSampleRate);\n }\n if (this.options.enableLongTask) {\n startTrackingLongTasks();\n }\n if (this.options._experiments.enableInteractions) {\n startTrackingInteractions();\n }\n\n this._latestRoute = {\n name: undefined,\n context: undefined,\n };\n }\n\n /**\n * @inheritDoc\n */\n // eslint-disable-next-line deprecation/deprecation\n setupOnce(_, getCurrentHub) {\n this._getCurrentHub = getCurrentHub;\n const hub = getCurrentHub();\n // eslint-disable-next-line deprecation/deprecation\n const client = hub.getClient();\n const clientOptions = client && client.getOptions();\n\n const {\n routingInstrumentation: instrumentRouting,\n startTransactionOnLocationChange,\n startTransactionOnPageLoad,\n markBackgroundTransactions,\n traceFetch,\n traceXHR,\n shouldCreateSpanForRequest,\n enableHTTPTimings,\n _experiments,\n } = this.options;\n\n const clientOptionsTracePropagationTargets = clientOptions && clientOptions.tracePropagationTargets;\n // There are three ways to configure tracePropagationTargets:\n // 1. via top level client option `tracePropagationTargets`\n // 2. via BrowserTracing option `tracePropagationTargets`\n // 3. via BrowserTracing option `tracingOrigins` (deprecated)\n //\n // To avoid confusion, favour top level client option `tracePropagationTargets`, and fallback to\n // BrowserTracing option `tracePropagationTargets` and then `tracingOrigins` (deprecated).\n // This is done as it minimizes bundle size (we don't have to have undefined checks).\n //\n // If both 1 and either one of 2 or 3 are set (from above), we log out a warning.\n // eslint-disable-next-line deprecation/deprecation\n const tracePropagationTargets = clientOptionsTracePropagationTargets || this.options.tracePropagationTargets;\n if (DEBUG_BUILD && this._hasSetTracePropagationTargets && clientOptionsTracePropagationTargets) {\n logger.warn(\n '[Tracing] The `tracePropagationTargets` option was set in the BrowserTracing integration and top level `Sentry.init`. The top level `Sentry.init` value is being used.',\n );\n }\n\n instrumentRouting(\n (context) => {\n const transaction = this._createRouteTransaction(context);\n\n this.options._experiments.onStartRouteTransaction &&\n this.options._experiments.onStartRouteTransaction(transaction, context, getCurrentHub);\n\n return transaction;\n },\n startTransactionOnPageLoad,\n startTransactionOnLocationChange,\n );\n\n if (markBackgroundTransactions) {\n registerBackgroundTabDetection();\n }\n\n if (_experiments.enableInteractions) {\n this._registerInteractionListener();\n }\n\n if (this.options.enableInp) {\n this._registerInpInteractionListener();\n }\n\n instrumentOutgoingRequests({\n traceFetch,\n traceXHR,\n tracePropagationTargets,\n shouldCreateSpanForRequest,\n enableHTTPTimings,\n });\n }\n\n /** Create routing idle transaction. */\n _createRouteTransaction(context) {\n if (!this._getCurrentHub) {\n DEBUG_BUILD &&\n logger.warn(`[Tracing] Did not create ${context.op} transaction because _getCurrentHub is invalid.`);\n return undefined;\n }\n\n const hub = this._getCurrentHub();\n\n const { beforeNavigate, idleTimeout, finalTimeout, heartbeatInterval } = this.options;\n\n const isPageloadTransaction = context.op === 'pageload';\n\n let expandedContext;\n if (isPageloadTransaction) {\n const sentryTrace = isPageloadTransaction ? getMetaContent('sentry-trace') : '';\n const baggage = isPageloadTransaction ? getMetaContent('baggage') : undefined;\n const { traceId, dsc, parentSpanId, sampled } = propagationContextFromHeaders(sentryTrace, baggage);\n expandedContext = {\n traceId,\n parentSpanId,\n parentSampled: sampled,\n ...context,\n metadata: {\n // eslint-disable-next-line deprecation/deprecation\n ...context.metadata,\n dynamicSamplingContext: dsc,\n },\n trimEnd: true,\n };\n } else {\n expandedContext = {\n trimEnd: true,\n ...context,\n };\n }\n\n const modifiedContext = typeof beforeNavigate === 'function' ? beforeNavigate(expandedContext) : expandedContext;\n\n // For backwards compatibility reasons, beforeNavigate can return undefined to \"drop\" the transaction (prevent it\n // from being sent to Sentry).\n const finalContext = modifiedContext === undefined ? { ...expandedContext, sampled: false } : modifiedContext;\n\n // If `beforeNavigate` set a custom name, record that fact\n // eslint-disable-next-line deprecation/deprecation\n finalContext.metadata =\n finalContext.name !== expandedContext.name\n ? // eslint-disable-next-line deprecation/deprecation\n { ...finalContext.metadata, source: 'custom' }\n : // eslint-disable-next-line deprecation/deprecation\n finalContext.metadata;\n\n this._latestRoute.name = finalContext.name;\n this._latestRoute.context = finalContext;\n\n // eslint-disable-next-line deprecation/deprecation\n if (finalContext.sampled === false) {\n DEBUG_BUILD && logger.log(`[Tracing] Will not send ${finalContext.op} transaction because of beforeNavigate.`);\n }\n\n DEBUG_BUILD && logger.log(`[Tracing] Starting ${finalContext.op} transaction on scope`);\n\n const { location } = WINDOW;\n\n const idleTransaction = startIdleTransaction(\n hub,\n finalContext,\n idleTimeout,\n finalTimeout,\n true,\n { location }, // for use in the tracesSampler\n heartbeatInterval,\n isPageloadTransaction, // should wait for finish signal if it's a pageload transaction\n );\n\n if (isPageloadTransaction) {\n if (WINDOW.document) {\n WINDOW.document.addEventListener('readystatechange', () => {\n if (['interactive', 'complete'].includes(WINDOW.document.readyState)) {\n idleTransaction.sendAutoFinishSignal();\n }\n });\n\n if (['interactive', 'complete'].includes(WINDOW.document.readyState)) {\n idleTransaction.sendAutoFinishSignal();\n }\n }\n }\n\n idleTransaction.registerBeforeFinishCallback(transaction => {\n this._collectWebVitals();\n addPerformanceEntries(transaction);\n });\n\n return idleTransaction ;\n }\n\n /** Start listener for interaction transactions */\n _registerInteractionListener() {\n let inflightInteractionTransaction;\n const registerInteractionTransaction = () => {\n const { idleTimeout, finalTimeout, heartbeatInterval } = this.options;\n const op = 'ui.action.click';\n\n // eslint-disable-next-line deprecation/deprecation\n const currentTransaction = getActiveTransaction();\n if (currentTransaction && currentTransaction.op && ['navigation', 'pageload'].includes(currentTransaction.op)) {\n DEBUG_BUILD &&\n logger.warn(\n `[Tracing] Did not create ${op} transaction because a pageload or navigation transaction is in progress.`,\n );\n return undefined;\n }\n\n if (inflightInteractionTransaction) {\n inflightInteractionTransaction.setFinishReason('interactionInterrupted');\n inflightInteractionTransaction.end();\n inflightInteractionTransaction = undefined;\n }\n\n if (!this._getCurrentHub) {\n DEBUG_BUILD && logger.warn(`[Tracing] Did not create ${op} transaction because _getCurrentHub is invalid.`);\n return undefined;\n }\n\n if (!this._latestRoute.name) {\n DEBUG_BUILD && logger.warn(`[Tracing] Did not create ${op} transaction because _latestRouteName is missing.`);\n return undefined;\n }\n\n const hub = this._getCurrentHub();\n const { location } = WINDOW;\n\n const context = {\n name: this._latestRoute.name,\n op,\n trimEnd: true,\n data: {\n [SEMANTIC_ATTRIBUTE_SENTRY_SOURCE]: this._latestRoute.context\n ? getSource(this._latestRoute.context)\n : 'url',\n },\n };\n\n inflightInteractionTransaction = startIdleTransaction(\n hub,\n context,\n idleTimeout,\n finalTimeout,\n true,\n { location }, // for use in the tracesSampler\n heartbeatInterval,\n );\n };\n\n ['click'].forEach(type => {\n if (WINDOW.document) {\n addEventListener(type, registerInteractionTransaction, { once: false, capture: true });\n }\n });\n }\n\n /** Creates a listener on interaction entries, and maps interactionIds to the origin path of the interaction */\n _registerInpInteractionListener() {\n const handleEntries = ({ entries }) => {\n const client = getClient();\n // We need to get the replay, user, and activeTransaction from the current scope\n // so that we can associate replay id, profile id, and a user display to the span\n const replay =\n client !== undefined && client.getIntegrationByName !== undefined\n ? (client.getIntegrationByName('Replay') )\n : undefined;\n const replayId = replay !== undefined ? replay.getReplayId() : undefined;\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction();\n const currentScope = getCurrentScope();\n const user = currentScope !== undefined ? currentScope.getUser() : undefined;\n entries.forEach(entry => {\n if (isPerformanceEventTiming(entry)) {\n const interactionId = entry.interactionId;\n if (interactionId === undefined) {\n return;\n }\n const existingInteraction = this._interactionIdToRouteNameMapping[interactionId];\n const duration = entry.duration;\n const startTime = entry.startTime;\n const keys = Object.keys(this._interactionIdToRouteNameMapping);\n const minInteractionId =\n keys.length > 0\n ? keys.reduce((a, b) => {\n return this._interactionIdToRouteNameMapping[a].duration <\n this._interactionIdToRouteNameMapping[b].duration\n ? a\n : b;\n })\n : undefined;\n // For a first input event to be considered, we must check that an interaction event does not already exist with the same duration and start time.\n // This is also checked in the web-vitals library.\n if (entry.entryType === 'first-input') {\n const matchingEntry = keys\n .map(key => this._interactionIdToRouteNameMapping[key])\n .some(interaction => {\n return interaction.duration === duration && interaction.startTime === startTime;\n });\n if (matchingEntry) {\n return;\n }\n }\n // Interactions with an id of 0 and are not first-input are not valid.\n if (!interactionId) {\n return;\n }\n // If the interaction already exists, we want to use the duration of the longest entry, since that is what the INP metric uses.\n if (existingInteraction) {\n existingInteraction.duration = Math.max(existingInteraction.duration, duration);\n } else if (\n keys.length < MAX_INTERACTIONS ||\n minInteractionId === undefined ||\n duration > this._interactionIdToRouteNameMapping[minInteractionId].duration\n ) {\n // If the interaction does not exist, we want to add it to the mapping if there is space, or if the duration is longer than the shortest entry.\n const routeName = this._latestRoute.name;\n const parentContext = this._latestRoute.context;\n if (routeName && parentContext) {\n if (minInteractionId && Object.keys(this._interactionIdToRouteNameMapping).length >= MAX_INTERACTIONS) {\n // eslint-disable-next-line @typescript-eslint/no-dynamic-delete\n delete this._interactionIdToRouteNameMapping[minInteractionId];\n }\n this._interactionIdToRouteNameMapping[interactionId] = {\n routeName,\n duration,\n parentContext,\n user,\n activeTransaction,\n replayId,\n startTime,\n };\n }\n }\n }\n });\n };\n addPerformanceInstrumentationHandler('event', handleEntries);\n addPerformanceInstrumentationHandler('first-input', handleEntries);\n }\n}\n\n/** Returns the value of a meta tag */\nfunction getMetaContent(metaName) {\n // Can't specify generic to `getDomElement` because tracing can be used\n // in a variety of environments, have to disable `no-unsafe-member-access`\n // as a result.\n const metaTag = getDomElement(`meta[name=${metaName}]`);\n // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n return metaTag ? metaTag.getAttribute('content') : undefined;\n}\n\nfunction getSource(context) {\n const sourceFromAttributes = context.attributes && context.attributes[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE];\n // eslint-disable-next-line deprecation/deprecation\n const sourceFromData = context.data && context.data[SEMANTIC_ATTRIBUTE_SENTRY_SOURCE];\n // eslint-disable-next-line deprecation/deprecation\n const sourceFromMetadata = context.metadata && context.metadata.source;\n\n return sourceFromAttributes || sourceFromData || sourceFromMetadata;\n}\n\nfunction isPerformanceEventTiming(entry) {\n return 'duration' in entry;\n}\n\nexport { BROWSER_TRACING_INTEGRATION_ID, BrowserTracing, getMetaContent };\n","import { logger, browserPerformanceTimeOrigin, addHistoryInstrumentationHandler } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../common/debug-build.js';\nimport { WINDOW } from './types.js';\n\n/**\n * Default function implementing pageload and navigation transactions\n */\nfunction instrumentRoutingWithDefaults(\n customStartTransaction,\n startTransactionOnPageLoad = true,\n startTransactionOnLocationChange = true,\n) {\n if (!WINDOW || !WINDOW.location) {\n DEBUG_BUILD && logger.warn('Could not initialize routing instrumentation due to invalid location');\n return;\n }\n\n let startingUrl = WINDOW.location.href;\n\n let activeTransaction;\n if (startTransactionOnPageLoad) {\n activeTransaction = customStartTransaction({\n name: WINDOW.location.pathname,\n // pageload should always start at timeOrigin (and needs to be in s, not ms)\n startTimestamp: browserPerformanceTimeOrigin ? browserPerformanceTimeOrigin / 1000 : undefined,\n op: 'pageload',\n origin: 'auto.pageload.browser',\n metadata: { source: 'url' },\n });\n }\n\n if (startTransactionOnLocationChange) {\n addHistoryInstrumentationHandler(({ to, from }) => {\n /**\n * This early return is there to account for some cases where a navigation transaction starts right after\n * long-running pageload. We make sure that if `from` is undefined and a valid `startingURL` exists, we don't\n * create an uneccessary navigation transaction.\n *\n * This was hard to duplicate, but this behavior stopped as soon as this fix was applied. This issue might also\n * only be caused in certain development environments where the usage of a hot module reloader is causing\n * errors.\n */\n if (from === undefined && startingUrl && startingUrl.indexOf(to) !== -1) {\n startingUrl = undefined;\n return;\n }\n\n if (from !== to) {\n startingUrl = undefined;\n if (activeTransaction) {\n DEBUG_BUILD && logger.log(`[Tracing] Finishing current transaction with op: ${activeTransaction.op}`);\n // If there's an open transaction on the scope, we need to finish it before creating an new one.\n activeTransaction.end();\n }\n activeTransaction = customStartTransaction({\n name: WINDOW.location.pathname,\n op: 'navigation',\n origin: 'auto.navigation.browser',\n metadata: { source: 'url' },\n });\n }\n });\n }\n}\n\nexport { instrumentRoutingWithDefaults };\n","\n","import { Component, OnDestroy, OnInit } from '@angular/core';\nimport { CacheBustingService } from '@app/modules/shared/services/cache-busting.service';\nimport { AuthService } from './modules/shared/services/auth.service';\n\n@Component({\n selector: 'tu-root',\n templateUrl: './app.component.html',\n})\nexport class AppComponent implements OnInit, OnDestroy {\n public title = 'tu works!';\n constructor(private readonly cacheBuster: CacheBustingService, public auth: AuthService) {}\n\n ngOnInit() {\n if (process.env.NODE_ENV === 'development') return;\n // 5 minutes\n this.cacheBuster.initVersionCheck(1000 * 60 * 5);\n }\n\n ngOnDestroy() {\n this.cacheBuster.clearVersionCheck();\n }\n}\n","/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n// THIS CODE IS GENERATED - DO NOT MODIFY.\nconst u = undefined;\nexport default [[[\"minuit\", \"midi\", \"mat.\", \"ap.m.\", \"soir\", \"nuit\"], u, [\"minuit\", \"midi\", \"du matin\", \"de l’après-midi\", \"du soir\", \"du matin\"]], [[\"minuit\", \"midi\", \"mat.\", \"ap.m.\", \"soir\", \"nuit\"], u, [\"minuit\", \"midi\", \"matin\", \"après-midi\", \"soir\", \"nuit\"]], [\"00:00\", \"12:00\", [\"04:00\", \"12:00\"], [\"12:00\", \"18:00\"], [\"18:00\", \"24:00\"], [\"00:00\", \"04:00\"]]];\n","/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n// THIS CODE IS GENERATED - DO NOT MODIFY.\nconst u = undefined;\nfunction plural(val) {\n const n = val, i = Math.floor(Math.abs(val)), v = val.toString().replace(/^[^.]*\\.?/, '').length, e = parseInt(val.toString().replace(/^[^e]*(e([-+]?\\d+))?/, '$2')) || 0;\n if (i === 0 || i === 1)\n return 1;\n if (e === 0 && (!(i === 0) && (i % 1000000 === 0 && v === 0)) || !(e >= 0 && e <= 5))\n return 4;\n return 5;\n}\nexport default [\"fr\", [[\"AM\", \"PM\"], u, u], u, [[\"D\", \"L\", \"M\", \"M\", \"J\", \"V\", \"S\"], [\"dim.\", \"lun.\", \"mar.\", \"mer.\", \"jeu.\", \"ven.\", \"sam.\"], [\"dimanche\", \"lundi\", \"mardi\", \"mercredi\", \"jeudi\", \"vendredi\", \"samedi\"], [\"di\", \"lu\", \"ma\", \"me\", \"je\", \"ve\", \"sa\"]], u, [[\"J\", \"F\", \"M\", \"A\", \"M\", \"J\", \"J\", \"A\", \"S\", \"O\", \"N\", \"D\"], [\"janv.\", \"févr.\", \"mars\", \"avr.\", \"mai\", \"juin\", \"juil.\", \"août\", \"sept.\", \"oct.\", \"nov.\", \"déc.\"], [\"janvier\", \"février\", \"mars\", \"avril\", \"mai\", \"juin\", \"juillet\", \"août\", \"septembre\", \"octobre\", \"novembre\", \"décembre\"]], u, [[\"av. J.-C.\", \"ap. J.-C.\"], u, [\"avant Jésus-Christ\", \"après Jésus-Christ\"]], 1, [6, 0], [\"dd/MM/y\", \"d MMM y\", \"d MMMM y\", \"EEEE d MMMM y\"], [\"HH:mm\", \"HH:mm:ss\", \"HH:mm:ss z\", \"HH:mm:ss zzzz\"], [\"{1} {0}\", \"{1}, {0}\", \"{1} 'à' {0}\", u], [\",\", \" \", \";\", \"%\", \"+\", \"-\", \"E\", \"×\", \"‰\", \"∞\", \"NaN\", \":\"], [\"#,##0.###\", \"#,##0 %\", \"#,##0.00 ¤\", \"#E0\"], \"EUR\", \"€\", \"euro\", { \"ARS\": [\"$AR\", \"$\"], \"AUD\": [\"$AU\", \"$\"], \"BEF\": [\"FB\"], \"BMD\": [\"$BM\", \"$\"], \"BND\": [\"$BN\", \"$\"], \"BYN\": [u, \"р.\"], \"BZD\": [\"$BZ\", \"$\"], \"CAD\": [\"$CA\", \"$\"], \"CLP\": [\"$CL\", \"$\"], \"CNY\": [u, \"¥\"], \"COP\": [\"$CO\", \"$\"], \"CYP\": [\"£CY\"], \"EGP\": [u, \"£E\"], \"FJD\": [\"$FJ\", \"$\"], \"FKP\": [\"£FK\", \"£\"], \"FRF\": [\"F\"], \"GBP\": [\"£GB\", \"£\"], \"GIP\": [\"£GI\", \"£\"], \"HKD\": [u, \"$\"], \"IEP\": [\"£IE\"], \"ILP\": [\"£IL\"], \"ITL\": [\"₤IT\"], \"JPY\": [u, \"¥\"], \"KMF\": [u, \"FC\"], \"LBP\": [\"£LB\", \"£L\"], \"MTP\": [\"£MT\"], \"MXN\": [\"$MX\", \"$\"], \"NAD\": [\"$NA\", \"$\"], \"NIO\": [u, \"$C\"], \"NZD\": [\"$NZ\", \"$\"], \"PHP\": [u, \"₱\"], \"RHD\": [\"$RH\"], \"RON\": [u, \"L\"], \"RWF\": [u, \"FR\"], \"SBD\": [\"$SB\", \"$\"], \"SGD\": [\"$SG\", \"$\"], \"SRD\": [\"$SR\", \"$\"], \"TOP\": [u, \"$T\"], \"TTD\": [\"$TT\", \"$\"], \"TWD\": [u, \"NT$\"], \"USD\": [\"$US\", \"$\"], \"UYU\": [\"$UY\", \"$\"], \"WST\": [\"$WS\"], \"XCD\": [u, \"$\"], \"XPF\": [\"FCFP\"], \"ZMW\": [u, \"Kw\"] }, \"ltr\", plural];\n","/**\n * @license Angular v14.3.0\n * (c) 2010-2022 Google LLC. https://angular.io/\n * License: MIT\n */\n\nimport { ɵAnimationGroupPlayer, NoopAnimationPlayer, AUTO_STYLE, ɵPRE_STYLE, sequence, style } from '@angular/animations';\nimport * as i0 from '@angular/core';\nimport { ɵRuntimeError, Injectable } from '@angular/core';\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst LINE_START = '\\n - ';\nfunction invalidTimingValue(exp) {\n return new ɵRuntimeError(3000 /* RuntimeErrorCode.INVALID_TIMING_VALUE */, ngDevMode && `The provided timing value \"${exp}\" is invalid.`);\n}\nfunction negativeStepValue() {\n return new ɵRuntimeError(3100 /* RuntimeErrorCode.NEGATIVE_STEP_VALUE */, ngDevMode && 'Duration values below 0 are not allowed for this animation step.');\n}\nfunction negativeDelayValue() {\n return new ɵRuntimeError(3101 /* RuntimeErrorCode.NEGATIVE_DELAY_VALUE */, ngDevMode && 'Delay values below 0 are not allowed for this animation step.');\n}\nfunction invalidStyleParams(varName) {\n return new ɵRuntimeError(3001 /* RuntimeErrorCode.INVALID_STYLE_PARAMS */, ngDevMode &&\n `Unable to resolve the local animation param ${varName} in the given list of values`);\n}\nfunction invalidParamValue(varName) {\n return new ɵRuntimeError(3003 /* RuntimeErrorCode.INVALID_PARAM_VALUE */, ngDevMode && `Please provide a value for the animation param ${varName}`);\n}\nfunction invalidNodeType(nodeType) {\n return new ɵRuntimeError(3004 /* RuntimeErrorCode.INVALID_NODE_TYPE */, ngDevMode && `Unable to resolve animation metadata node #${nodeType}`);\n}\nfunction invalidCssUnitValue(userProvidedProperty, value) {\n return new ɵRuntimeError(3005 /* RuntimeErrorCode.INVALID_CSS_UNIT_VALUE */, ngDevMode && `Please provide a CSS unit value for ${userProvidedProperty}:${value}`);\n}\nfunction invalidTrigger() {\n return new ɵRuntimeError(3006 /* RuntimeErrorCode.INVALID_TRIGGER */, ngDevMode &&\n 'animation triggers cannot be prefixed with an `@` sign (e.g. trigger(\\'@foo\\', [...]))');\n}\nfunction invalidDefinition() {\n return new ɵRuntimeError(3007 /* RuntimeErrorCode.INVALID_DEFINITION */, ngDevMode && 'only state() and transition() definitions can sit inside of a trigger()');\n}\nfunction invalidState(metadataName, missingSubs) {\n return new ɵRuntimeError(3008 /* RuntimeErrorCode.INVALID_STATE */, ngDevMode &&\n `state(\"${metadataName}\", ...) must define default values for all the following style substitutions: ${missingSubs.join(', ')}`);\n}\nfunction invalidStyleValue(value) {\n return new ɵRuntimeError(3002 /* RuntimeErrorCode.INVALID_STYLE_VALUE */, ngDevMode && `The provided style string value ${value} is not allowed.`);\n}\nfunction invalidProperty(prop) {\n return new ɵRuntimeError(3009 /* RuntimeErrorCode.INVALID_PROPERTY */, ngDevMode &&\n `The provided animation property \"${prop}\" is not a supported CSS property for animations`);\n}\nfunction invalidParallelAnimation(prop, firstStart, firstEnd, secondStart, secondEnd) {\n return new ɵRuntimeError(3010 /* RuntimeErrorCode.INVALID_PARALLEL_ANIMATION */, ngDevMode &&\n `The CSS property \"${prop}\" that exists between the times of \"${firstStart}ms\" and \"${firstEnd}ms\" is also being animated in a parallel animation between the times of \"${secondStart}ms\" and \"${secondEnd}ms\"`);\n}\nfunction invalidKeyframes() {\n return new ɵRuntimeError(3011 /* RuntimeErrorCode.INVALID_KEYFRAMES */, ngDevMode && `keyframes() must be placed inside of a call to animate()`);\n}\nfunction invalidOffset() {\n return new ɵRuntimeError(3012 /* RuntimeErrorCode.INVALID_OFFSET */, ngDevMode && `Please ensure that all keyframe offsets are between 0 and 1`);\n}\nfunction keyframeOffsetsOutOfOrder() {\n return new ɵRuntimeError(3200 /* RuntimeErrorCode.KEYFRAME_OFFSETS_OUT_OF_ORDER */, ngDevMode && `Please ensure that all keyframe offsets are in order`);\n}\nfunction keyframesMissingOffsets() {\n return new ɵRuntimeError(3202 /* RuntimeErrorCode.KEYFRAMES_MISSING_OFFSETS */, ngDevMode && `Not all style() steps within the declared keyframes() contain offsets`);\n}\nfunction invalidStagger() {\n return new ɵRuntimeError(3013 /* RuntimeErrorCode.INVALID_STAGGER */, ngDevMode && `stagger() can only be used inside of query()`);\n}\nfunction invalidQuery(selector) {\n return new ɵRuntimeError(3014 /* RuntimeErrorCode.INVALID_QUERY */, ngDevMode &&\n `\\`query(\"${selector}\")\\` returned zero elements. (Use \\`query(\"${selector}\", { optional: true })\\` if you wish to allow this.)`);\n}\nfunction invalidExpression(expr) {\n return new ɵRuntimeError(3015 /* RuntimeErrorCode.INVALID_EXPRESSION */, ngDevMode && `The provided transition expression \"${expr}\" is not supported`);\n}\nfunction invalidTransitionAlias(alias) {\n return new ɵRuntimeError(3016 /* RuntimeErrorCode.INVALID_TRANSITION_ALIAS */, ngDevMode && `The transition alias value \"${alias}\" is not supported`);\n}\nfunction validationFailed(errors) {\n return new ɵRuntimeError(3500 /* RuntimeErrorCode.VALIDATION_FAILED */, ngDevMode && `animation validation failed:\\n${errors.map(err => err.message).join('\\n')}`);\n}\nfunction buildingFailed(errors) {\n return new ɵRuntimeError(3501 /* RuntimeErrorCode.BUILDING_FAILED */, ngDevMode && `animation building failed:\\n${errors.map(err => err.message).join('\\n')}`);\n}\nfunction triggerBuildFailed(name, errors) {\n return new ɵRuntimeError(3404 /* RuntimeErrorCode.TRIGGER_BUILD_FAILED */, ngDevMode &&\n `The animation trigger \"${name}\" has failed to build due to the following errors:\\n - ${errors.map(err => err.message).join('\\n - ')}`);\n}\nfunction animationFailed(errors) {\n return new ɵRuntimeError(3502 /* RuntimeErrorCode.ANIMATION_FAILED */, ngDevMode &&\n `Unable to animate due to the following errors:${LINE_START}${errors.map(err => err.message).join(LINE_START)}`);\n}\nfunction registerFailed(errors) {\n return new ɵRuntimeError(3503 /* RuntimeErrorCode.REGISTRATION_FAILED */, ngDevMode &&\n `Unable to build the animation due to the following errors: ${errors.map(err => err.message).join('\\n')}`);\n}\nfunction missingOrDestroyedAnimation() {\n return new ɵRuntimeError(3300 /* RuntimeErrorCode.MISSING_OR_DESTROYED_ANIMATION */, ngDevMode && 'The requested animation doesn\\'t exist or has already been destroyed');\n}\nfunction createAnimationFailed(errors) {\n return new ɵRuntimeError(3504 /* RuntimeErrorCode.CREATE_ANIMATION_FAILED */, ngDevMode &&\n `Unable to create the animation due to the following errors:${errors.map(err => err.message).join('\\n')}`);\n}\nfunction missingPlayer(id) {\n return new ɵRuntimeError(3301 /* RuntimeErrorCode.MISSING_PLAYER */, ngDevMode && `Unable to find the timeline player referenced by ${id}`);\n}\nfunction missingTrigger(phase, name) {\n return new ɵRuntimeError(3302 /* RuntimeErrorCode.MISSING_TRIGGER */, ngDevMode &&\n `Unable to listen on the animation trigger event \"${phase}\" because the animation trigger \"${name}\" doesn\\'t exist!`);\n}\nfunction missingEvent(name) {\n return new ɵRuntimeError(3303 /* RuntimeErrorCode.MISSING_EVENT */, ngDevMode &&\n `Unable to listen on the animation trigger \"${name}\" because the provided event is undefined!`);\n}\nfunction unsupportedTriggerEvent(phase, name) {\n return new ɵRuntimeError(3400 /* RuntimeErrorCode.UNSUPPORTED_TRIGGER_EVENT */, ngDevMode &&\n `The provided animation trigger event \"${phase}\" for the animation trigger \"${name}\" is not supported!`);\n}\nfunction unregisteredTrigger(name) {\n return new ɵRuntimeError(3401 /* RuntimeErrorCode.UNREGISTERED_TRIGGER */, ngDevMode && `The provided animation trigger \"${name}\" has not been registered!`);\n}\nfunction triggerTransitionsFailed(errors) {\n return new ɵRuntimeError(3402 /* RuntimeErrorCode.TRIGGER_TRANSITIONS_FAILED */, ngDevMode &&\n `Unable to process animations due to the following failed trigger transitions\\n ${errors.map(err => err.message).join('\\n')}`);\n}\nfunction triggerParsingFailed(name, errors) {\n return new ɵRuntimeError(3403 /* RuntimeErrorCode.TRIGGER_PARSING_FAILED */, ngDevMode &&\n `Animation parsing for the ${name} trigger have failed:${LINE_START}${errors.map(err => err.message).join(LINE_START)}`);\n}\nfunction transitionFailed(name, errors) {\n return new ɵRuntimeError(3505 /* RuntimeErrorCode.TRANSITION_FAILED */, ngDevMode && `@${name} has failed due to:\\n ${errors.map(err => err.message).join('\\n- ')}`);\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * Set of all animatable CSS properties\n *\n * @see https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_animated_properties\n */\nconst ANIMATABLE_PROP_SET = new Set([\n '-moz-outline-radius',\n '-moz-outline-radius-bottomleft',\n '-moz-outline-radius-bottomright',\n '-moz-outline-radius-topleft',\n '-moz-outline-radius-topright',\n '-ms-grid-columns',\n '-ms-grid-rows',\n '-webkit-line-clamp',\n '-webkit-text-fill-color',\n '-webkit-text-stroke',\n '-webkit-text-stroke-color',\n 'accent-color',\n 'all',\n 'backdrop-filter',\n 'background',\n 'background-color',\n 'background-position',\n 'background-size',\n 'block-size',\n 'border',\n 'border-block-end',\n 'border-block-end-color',\n 'border-block-end-width',\n 'border-block-start',\n 'border-block-start-color',\n 'border-block-start-width',\n 'border-bottom',\n 'border-bottom-color',\n 'border-bottom-left-radius',\n 'border-bottom-right-radius',\n 'border-bottom-width',\n 'border-color',\n 'border-end-end-radius',\n 'border-end-start-radius',\n 'border-image-outset',\n 'border-image-slice',\n 'border-image-width',\n 'border-inline-end',\n 'border-inline-end-color',\n 'border-inline-end-width',\n 'border-inline-start',\n 'border-inline-start-color',\n 'border-inline-start-width',\n 'border-left',\n 'border-left-color',\n 'border-left-width',\n 'border-radius',\n 'border-right',\n 'border-right-color',\n 'border-right-width',\n 'border-start-end-radius',\n 'border-start-start-radius',\n 'border-top',\n 'border-top-color',\n 'border-top-left-radius',\n 'border-top-right-radius',\n 'border-top-width',\n 'border-width',\n 'bottom',\n 'box-shadow',\n 'caret-color',\n 'clip',\n 'clip-path',\n 'color',\n 'column-count',\n 'column-gap',\n 'column-rule',\n 'column-rule-color',\n 'column-rule-width',\n 'column-width',\n 'columns',\n 'filter',\n 'flex',\n 'flex-basis',\n 'flex-grow',\n 'flex-shrink',\n 'font',\n 'font-size',\n 'font-size-adjust',\n 'font-stretch',\n 'font-variation-settings',\n 'font-weight',\n 'gap',\n 'grid-column-gap',\n 'grid-gap',\n 'grid-row-gap',\n 'grid-template-columns',\n 'grid-template-rows',\n 'height',\n 'inline-size',\n 'input-security',\n 'inset',\n 'inset-block',\n 'inset-block-end',\n 'inset-block-start',\n 'inset-inline',\n 'inset-inline-end',\n 'inset-inline-start',\n 'left',\n 'letter-spacing',\n 'line-clamp',\n 'line-height',\n 'margin',\n 'margin-block-end',\n 'margin-block-start',\n 'margin-bottom',\n 'margin-inline-end',\n 'margin-inline-start',\n 'margin-left',\n 'margin-right',\n 'margin-top',\n 'mask',\n 'mask-border',\n 'mask-position',\n 'mask-size',\n 'max-block-size',\n 'max-height',\n 'max-inline-size',\n 'max-lines',\n 'max-width',\n 'min-block-size',\n 'min-height',\n 'min-inline-size',\n 'min-width',\n 'object-position',\n 'offset',\n 'offset-anchor',\n 'offset-distance',\n 'offset-path',\n 'offset-position',\n 'offset-rotate',\n 'opacity',\n 'order',\n 'outline',\n 'outline-color',\n 'outline-offset',\n 'outline-width',\n 'padding',\n 'padding-block-end',\n 'padding-block-start',\n 'padding-bottom',\n 'padding-inline-end',\n 'padding-inline-start',\n 'padding-left',\n 'padding-right',\n 'padding-top',\n 'perspective',\n 'perspective-origin',\n 'right',\n 'rotate',\n 'row-gap',\n 'scale',\n 'scroll-margin',\n 'scroll-margin-block',\n 'scroll-margin-block-end',\n 'scroll-margin-block-start',\n 'scroll-margin-bottom',\n 'scroll-margin-inline',\n 'scroll-margin-inline-end',\n 'scroll-margin-inline-start',\n 'scroll-margin-left',\n 'scroll-margin-right',\n 'scroll-margin-top',\n 'scroll-padding',\n 'scroll-padding-block',\n 'scroll-padding-block-end',\n 'scroll-padding-block-start',\n 'scroll-padding-bottom',\n 'scroll-padding-inline',\n 'scroll-padding-inline-end',\n 'scroll-padding-inline-start',\n 'scroll-padding-left',\n 'scroll-padding-right',\n 'scroll-padding-top',\n 'scroll-snap-coordinate',\n 'scroll-snap-destination',\n 'scrollbar-color',\n 'shape-image-threshold',\n 'shape-margin',\n 'shape-outside',\n 'tab-size',\n 'text-decoration',\n 'text-decoration-color',\n 'text-decoration-thickness',\n 'text-emphasis',\n 'text-emphasis-color',\n 'text-indent',\n 'text-shadow',\n 'text-underline-offset',\n 'top',\n 'transform',\n 'transform-origin',\n 'translate',\n 'vertical-align',\n 'visibility',\n 'width',\n 'word-spacing',\n 'z-index',\n 'zoom',\n]);\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nfunction isBrowser() {\n return (typeof window !== 'undefined' && typeof window.document !== 'undefined');\n}\nfunction isNode() {\n // Checking only for `process` isn't enough to identify whether or not we're in a Node\n // environment, because Webpack by default will polyfill the `process`. While we can discern\n // that Webpack polyfilled it by looking at `process.browser`, it's very Webpack-specific and\n // might not be future-proof. Instead we look at the stringified version of `process` which\n // is `[object process]` in Node and `[object Object]` when polyfilled.\n return typeof process !== 'undefined' && {}.toString.call(process) === '[object process]';\n}\nfunction optimizeGroupPlayer(players) {\n switch (players.length) {\n case 0:\n return new NoopAnimationPlayer();\n case 1:\n return players[0];\n default:\n return new ɵAnimationGroupPlayer(players);\n }\n}\nfunction normalizeKeyframes$1(driver, normalizer, element, keyframes, preStyles = new Map(), postStyles = new Map()) {\n const errors = [];\n const normalizedKeyframes = [];\n let previousOffset = -1;\n let previousKeyframe = null;\n keyframes.forEach(kf => {\n const offset = kf.get('offset');\n const isSameOffset = offset == previousOffset;\n const normalizedKeyframe = (isSameOffset && previousKeyframe) || new Map();\n kf.forEach((val, prop) => {\n let normalizedProp = prop;\n let normalizedValue = val;\n if (prop !== 'offset') {\n normalizedProp = normalizer.normalizePropertyName(normalizedProp, errors);\n switch (normalizedValue) {\n case ɵPRE_STYLE:\n normalizedValue = preStyles.get(prop);\n break;\n case AUTO_STYLE:\n normalizedValue = postStyles.get(prop);\n break;\n default:\n normalizedValue =\n normalizer.normalizeStyleValue(prop, normalizedProp, normalizedValue, errors);\n break;\n }\n }\n normalizedKeyframe.set(normalizedProp, normalizedValue);\n });\n if (!isSameOffset) {\n normalizedKeyframes.push(normalizedKeyframe);\n }\n previousKeyframe = normalizedKeyframe;\n previousOffset = offset;\n });\n if (errors.length) {\n throw animationFailed(errors);\n }\n return normalizedKeyframes;\n}\nfunction listenOnPlayer(player, eventName, event, callback) {\n switch (eventName) {\n case 'start':\n player.onStart(() => callback(event && copyAnimationEvent(event, 'start', player)));\n break;\n case 'done':\n player.onDone(() => callback(event && copyAnimationEvent(event, 'done', player)));\n break;\n case 'destroy':\n player.onDestroy(() => callback(event && copyAnimationEvent(event, 'destroy', player)));\n break;\n }\n}\nfunction copyAnimationEvent(e, phaseName, player) {\n const totalTime = player.totalTime;\n const disabled = player.disabled ? true : false;\n const event = makeAnimationEvent(e.element, e.triggerName, e.fromState, e.toState, phaseName || e.phaseName, totalTime == undefined ? e.totalTime : totalTime, disabled);\n const data = e['_data'];\n if (data != null) {\n event['_data'] = data;\n }\n return event;\n}\nfunction makeAnimationEvent(element, triggerName, fromState, toState, phaseName = '', totalTime = 0, disabled) {\n return { element, triggerName, fromState, toState, phaseName, totalTime, disabled: !!disabled };\n}\nfunction getOrSetDefaultValue(map, key, defaultValue) {\n let value = map.get(key);\n if (!value) {\n map.set(key, value = defaultValue);\n }\n return value;\n}\nfunction parseTimelineCommand(command) {\n const separatorPos = command.indexOf(':');\n const id = command.substring(1, separatorPos);\n const action = command.slice(separatorPos + 1);\n return [id, action];\n}\nlet _contains = (elm1, elm2) => false;\nlet _query = (element, selector, multi) => {\n return [];\n};\nlet _documentElement = null;\nfunction getParentElement(element) {\n const parent = element.parentNode || element.host; // consider host to support shadow DOM\n if (parent === _documentElement) {\n return null;\n }\n return parent;\n}\n// Define utility methods for browsers and platform-server(domino) where Element\n// and utility methods exist.\nconst _isNode = isNode();\nif (_isNode || typeof Element !== 'undefined') {\n if (!isBrowser()) {\n _contains = (elm1, elm2) => elm1.contains(elm2);\n }\n else {\n // Read the document element in an IIFE that's been marked pure to avoid a top-level property\n // read that may prevent tree-shaking.\n _documentElement = /* @__PURE__ */ (() => document.documentElement)();\n _contains = (elm1, elm2) => {\n while (elm2) {\n if (elm2 === elm1) {\n return true;\n }\n elm2 = getParentElement(elm2);\n }\n return false;\n };\n }\n _query = (element, selector, multi) => {\n if (multi) {\n return Array.from(element.querySelectorAll(selector));\n }\n const elem = element.querySelector(selector);\n return elem ? [elem] : [];\n };\n}\nfunction containsVendorPrefix(prop) {\n // Webkit is the only real popular vendor prefix nowadays\n // cc: http://shouldiprefix.com/\n return prop.substring(1, 6) == 'ebkit'; // webkit or Webkit\n}\nlet _CACHED_BODY = null;\nlet _IS_WEBKIT = false;\nfunction validateStyleProperty(prop) {\n if (!_CACHED_BODY) {\n _CACHED_BODY = getBodyNode() || {};\n _IS_WEBKIT = _CACHED_BODY.style ? ('WebkitAppearance' in _CACHED_BODY.style) : false;\n }\n let result = true;\n if (_CACHED_BODY.style && !containsVendorPrefix(prop)) {\n result = prop in _CACHED_BODY.style;\n if (!result && _IS_WEBKIT) {\n const camelProp = 'Webkit' + prop.charAt(0).toUpperCase() + prop.slice(1);\n result = camelProp in _CACHED_BODY.style;\n }\n }\n return result;\n}\nfunction validateWebAnimatableStyleProperty(prop) {\n return ANIMATABLE_PROP_SET.has(prop);\n}\nfunction getBodyNode() {\n if (typeof document != 'undefined') {\n return document.body;\n }\n return null;\n}\nconst containsElement = _contains;\nconst invokeQuery = _query;\nfunction hypenatePropsKeys(original) {\n const newMap = new Map();\n original.forEach((val, prop) => {\n const newProp = prop.replace(/([a-z])([A-Z])/g, '$1-$2');\n newMap.set(newProp, val);\n });\n return newMap;\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * @publicApi\n */\nclass NoopAnimationDriver {\n validateStyleProperty(prop) {\n return validateStyleProperty(prop);\n }\n matchesElement(_element, _selector) {\n // This method is deprecated and no longer in use so we return false.\n return false;\n }\n containsElement(elm1, elm2) {\n return containsElement(elm1, elm2);\n }\n getParentElement(element) {\n return getParentElement(element);\n }\n query(element, selector, multi) {\n return invokeQuery(element, selector, multi);\n }\n computeStyle(element, prop, defaultValue) {\n return defaultValue || '';\n }\n animate(element, keyframes, duration, delay, easing, previousPlayers = [], scrubberAccessRequested) {\n return new NoopAnimationPlayer(duration, delay);\n }\n}\nNoopAnimationDriver.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationDriver, deps: [], target: i0.ɵɵFactoryTarget.Injectable });\nNoopAnimationDriver.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationDriver });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationDriver, decorators: [{\n type: Injectable\n }] });\n/**\n * @publicApi\n */\nclass AnimationDriver {\n}\nAnimationDriver.NOOP = ( /* @__PURE__ */new NoopAnimationDriver());\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst ONE_SECOND = 1000;\nconst SUBSTITUTION_EXPR_START = '{{';\nconst SUBSTITUTION_EXPR_END = '}}';\nconst ENTER_CLASSNAME = 'ng-enter';\nconst LEAVE_CLASSNAME = 'ng-leave';\nconst NG_TRIGGER_CLASSNAME = 'ng-trigger';\nconst NG_TRIGGER_SELECTOR = '.ng-trigger';\nconst NG_ANIMATING_CLASSNAME = 'ng-animating';\nconst NG_ANIMATING_SELECTOR = '.ng-animating';\nfunction resolveTimingValue(value) {\n if (typeof value == 'number')\n return value;\n const matches = value.match(/^(-?[\\.\\d]+)(m?s)/);\n if (!matches || matches.length < 2)\n return 0;\n return _convertTimeValueToMS(parseFloat(matches[1]), matches[2]);\n}\nfunction _convertTimeValueToMS(value, unit) {\n switch (unit) {\n case 's':\n return value * ONE_SECOND;\n default: // ms or something else\n return value;\n }\n}\nfunction resolveTiming(timings, errors, allowNegativeValues) {\n return timings.hasOwnProperty('duration') ?\n timings :\n parseTimeExpression(timings, errors, allowNegativeValues);\n}\nfunction parseTimeExpression(exp, errors, allowNegativeValues) {\n const regex = /^(-?[\\.\\d]+)(m?s)(?:\\s+(-?[\\.\\d]+)(m?s))?(?:\\s+([-a-z]+(?:\\(.+?\\))?))?$/i;\n let duration;\n let delay = 0;\n let easing = '';\n if (typeof exp === 'string') {\n const matches = exp.match(regex);\n if (matches === null) {\n errors.push(invalidTimingValue(exp));\n return { duration: 0, delay: 0, easing: '' };\n }\n duration = _convertTimeValueToMS(parseFloat(matches[1]), matches[2]);\n const delayMatch = matches[3];\n if (delayMatch != null) {\n delay = _convertTimeValueToMS(parseFloat(delayMatch), matches[4]);\n }\n const easingVal = matches[5];\n if (easingVal) {\n easing = easingVal;\n }\n }\n else {\n duration = exp;\n }\n if (!allowNegativeValues) {\n let containsErrors = false;\n let startIndex = errors.length;\n if (duration < 0) {\n errors.push(negativeStepValue());\n containsErrors = true;\n }\n if (delay < 0) {\n errors.push(negativeDelayValue());\n containsErrors = true;\n }\n if (containsErrors) {\n errors.splice(startIndex, 0, invalidTimingValue(exp));\n }\n }\n return { duration, delay, easing };\n}\nfunction copyObj(obj, destination = {}) {\n Object.keys(obj).forEach(prop => {\n destination[prop] = obj[prop];\n });\n return destination;\n}\nfunction convertToMap(obj) {\n const styleMap = new Map();\n Object.keys(obj).forEach(prop => {\n const val = obj[prop];\n styleMap.set(prop, val);\n });\n return styleMap;\n}\nfunction normalizeKeyframes(keyframes) {\n if (!keyframes.length) {\n return [];\n }\n if (keyframes[0] instanceof Map) {\n return keyframes;\n }\n return keyframes.map(kf => convertToMap(kf));\n}\nfunction normalizeStyles(styles) {\n const normalizedStyles = new Map();\n if (Array.isArray(styles)) {\n styles.forEach(data => copyStyles(data, normalizedStyles));\n }\n else {\n copyStyles(styles, normalizedStyles);\n }\n return normalizedStyles;\n}\nfunction copyStyles(styles, destination = new Map(), backfill) {\n if (backfill) {\n for (let [prop, val] of backfill) {\n destination.set(prop, val);\n }\n }\n for (let [prop, val] of styles) {\n destination.set(prop, val);\n }\n return destination;\n}\nfunction getStyleAttributeString(element, key, value) {\n // Return the key-value pair string to be added to the style attribute for the\n // given CSS style key.\n if (value) {\n return key + ':' + value + ';';\n }\n else {\n return '';\n }\n}\nfunction writeStyleAttribute(element) {\n // Read the style property of the element and manually reflect it to the\n // style attribute. This is needed because Domino on platform-server doesn't\n // understand the full set of allowed CSS properties and doesn't reflect some\n // of them automatically.\n let styleAttrValue = '';\n for (let i = 0; i < element.style.length; i++) {\n const key = element.style.item(i);\n styleAttrValue += getStyleAttributeString(element, key, element.style.getPropertyValue(key));\n }\n for (const key in element.style) {\n // Skip internal Domino properties that don't need to be reflected.\n if (!element.style.hasOwnProperty(key) || key.startsWith('_')) {\n continue;\n }\n const dashKey = camelCaseToDashCase(key);\n styleAttrValue += getStyleAttributeString(element, dashKey, element.style[key]);\n }\n element.setAttribute('style', styleAttrValue);\n}\nfunction setStyles(element, styles, formerStyles) {\n if (element['style']) {\n styles.forEach((val, prop) => {\n const camelProp = dashCaseToCamelCase(prop);\n if (formerStyles && !formerStyles.has(prop)) {\n formerStyles.set(prop, element.style[camelProp]);\n }\n element.style[camelProp] = val;\n });\n // On the server set the 'style' attribute since it's not automatically reflected.\n if (isNode()) {\n writeStyleAttribute(element);\n }\n }\n}\nfunction eraseStyles(element, styles) {\n if (element['style']) {\n styles.forEach((_, prop) => {\n const camelProp = dashCaseToCamelCase(prop);\n element.style[camelProp] = '';\n });\n // On the server set the 'style' attribute since it's not automatically reflected.\n if (isNode()) {\n writeStyleAttribute(element);\n }\n }\n}\nfunction normalizeAnimationEntry(steps) {\n if (Array.isArray(steps)) {\n if (steps.length == 1)\n return steps[0];\n return sequence(steps);\n }\n return steps;\n}\nfunction validateStyleParams(value, options, errors) {\n const params = options.params || {};\n const matches = extractStyleParams(value);\n if (matches.length) {\n matches.forEach(varName => {\n if (!params.hasOwnProperty(varName)) {\n errors.push(invalidStyleParams(varName));\n }\n });\n }\n}\nconst PARAM_REGEX = new RegExp(`${SUBSTITUTION_EXPR_START}\\\\s*(.+?)\\\\s*${SUBSTITUTION_EXPR_END}`, 'g');\nfunction extractStyleParams(value) {\n let params = [];\n if (typeof value === 'string') {\n let match;\n while (match = PARAM_REGEX.exec(value)) {\n params.push(match[1]);\n }\n PARAM_REGEX.lastIndex = 0;\n }\n return params;\n}\nfunction interpolateParams(value, params, errors) {\n const original = value.toString();\n const str = original.replace(PARAM_REGEX, (_, varName) => {\n let localVal = params[varName];\n // this means that the value was never overridden by the data passed in by the user\n if (localVal == null) {\n errors.push(invalidParamValue(varName));\n localVal = '';\n }\n return localVal.toString();\n });\n // we do this to assert that numeric values stay as they are\n return str == original ? value : str;\n}\nfunction iteratorToArray(iterator) {\n const arr = [];\n let item = iterator.next();\n while (!item.done) {\n arr.push(item.value);\n item = iterator.next();\n }\n return arr;\n}\nconst DASH_CASE_REGEXP = /-+([a-z0-9])/g;\nfunction dashCaseToCamelCase(input) {\n return input.replace(DASH_CASE_REGEXP, (...m) => m[1].toUpperCase());\n}\nfunction camelCaseToDashCase(input) {\n return input.replace(/([a-z])([A-Z])/g, '$1-$2').toLowerCase();\n}\nfunction allowPreviousPlayerStylesMerge(duration, delay) {\n return duration === 0 || delay === 0;\n}\nfunction balancePreviousStylesIntoKeyframes(element, keyframes, previousStyles) {\n if (previousStyles.size && keyframes.length) {\n let startingKeyframe = keyframes[0];\n let missingStyleProps = [];\n previousStyles.forEach((val, prop) => {\n if (!startingKeyframe.has(prop)) {\n missingStyleProps.push(prop);\n }\n startingKeyframe.set(prop, val);\n });\n if (missingStyleProps.length) {\n for (let i = 1; i < keyframes.length; i++) {\n let kf = keyframes[i];\n missingStyleProps.forEach(prop => kf.set(prop, computeStyle(element, prop)));\n }\n }\n }\n return keyframes;\n}\nfunction visitDslNode(visitor, node, context) {\n switch (node.type) {\n case 7 /* AnimationMetadataType.Trigger */:\n return visitor.visitTrigger(node, context);\n case 0 /* AnimationMetadataType.State */:\n return visitor.visitState(node, context);\n case 1 /* AnimationMetadataType.Transition */:\n return visitor.visitTransition(node, context);\n case 2 /* AnimationMetadataType.Sequence */:\n return visitor.visitSequence(node, context);\n case 3 /* AnimationMetadataType.Group */:\n return visitor.visitGroup(node, context);\n case 4 /* AnimationMetadataType.Animate */:\n return visitor.visitAnimate(node, context);\n case 5 /* AnimationMetadataType.Keyframes */:\n return visitor.visitKeyframes(node, context);\n case 6 /* AnimationMetadataType.Style */:\n return visitor.visitStyle(node, context);\n case 8 /* AnimationMetadataType.Reference */:\n return visitor.visitReference(node, context);\n case 9 /* AnimationMetadataType.AnimateChild */:\n return visitor.visitAnimateChild(node, context);\n case 10 /* AnimationMetadataType.AnimateRef */:\n return visitor.visitAnimateRef(node, context);\n case 11 /* AnimationMetadataType.Query */:\n return visitor.visitQuery(node, context);\n case 12 /* AnimationMetadataType.Stagger */:\n return visitor.visitStagger(node, context);\n default:\n throw invalidNodeType(node.type);\n }\n}\nfunction computeStyle(element, prop) {\n return window.getComputedStyle(element)[prop];\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst NG_DEV_MODE = typeof ngDevMode === 'undefined' || !!ngDevMode;\nfunction createListOfWarnings(warnings) {\n const LINE_START = '\\n - ';\n return `${LINE_START}${warnings.filter(Boolean).map(warning => warning).join(LINE_START)}`;\n}\nfunction warnValidation(warnings) {\n NG_DEV_MODE && console.warn(`animation validation warnings:${createListOfWarnings(warnings)}`);\n}\nfunction warnTriggerBuild(name, warnings) {\n NG_DEV_MODE &&\n console.warn(`The animation trigger \"${name}\" has built with the following warnings:${createListOfWarnings(warnings)}`);\n}\nfunction warnRegister(warnings) {\n NG_DEV_MODE &&\n console.warn(`Animation built with the following warnings:${createListOfWarnings(warnings)}`);\n}\nfunction triggerParsingWarnings(name, warnings) {\n NG_DEV_MODE &&\n console.warn(`Animation parsing for the ${name} trigger presents the following warnings:${createListOfWarnings(warnings)}`);\n}\nfunction pushUnrecognizedPropertiesWarning(warnings, props) {\n if (props.length) {\n warnings.push(`The following provided properties are not recognized: ${props.join(', ')}`);\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst ANY_STATE = '*';\nfunction parseTransitionExpr(transitionValue, errors) {\n const expressions = [];\n if (typeof transitionValue == 'string') {\n transitionValue.split(/\\s*,\\s*/).forEach(str => parseInnerTransitionStr(str, expressions, errors));\n }\n else {\n expressions.push(transitionValue);\n }\n return expressions;\n}\nfunction parseInnerTransitionStr(eventStr, expressions, errors) {\n if (eventStr[0] == ':') {\n const result = parseAnimationAlias(eventStr, errors);\n if (typeof result == 'function') {\n expressions.push(result);\n return;\n }\n eventStr = result;\n }\n const match = eventStr.match(/^(\\*|[-\\w]+)\\s*()\\s*(\\*|[-\\w]+)$/);\n if (match == null || match.length < 4) {\n errors.push(invalidExpression(eventStr));\n return expressions;\n }\n const fromState = match[1];\n const separator = match[2];\n const toState = match[3];\n expressions.push(makeLambdaFromStates(fromState, toState));\n const isFullAnyStateExpr = fromState == ANY_STATE && toState == ANY_STATE;\n if (separator[0] == '<' && !isFullAnyStateExpr) {\n expressions.push(makeLambdaFromStates(toState, fromState));\n }\n}\nfunction parseAnimationAlias(alias, errors) {\n switch (alias) {\n case ':enter':\n return 'void => *';\n case ':leave':\n return '* => void';\n case ':increment':\n return (fromState, toState) => parseFloat(toState) > parseFloat(fromState);\n case ':decrement':\n return (fromState, toState) => parseFloat(toState) < parseFloat(fromState);\n default:\n errors.push(invalidTransitionAlias(alias));\n return '* => *';\n }\n}\n// DO NOT REFACTOR ... keep the follow set instantiations\n// with the values intact (closure compiler for some reason\n// removes follow-up lines that add the values outside of\n// the constructor...\nconst TRUE_BOOLEAN_VALUES = new Set(['true', '1']);\nconst FALSE_BOOLEAN_VALUES = new Set(['false', '0']);\nfunction makeLambdaFromStates(lhs, rhs) {\n const LHS_MATCH_BOOLEAN = TRUE_BOOLEAN_VALUES.has(lhs) || FALSE_BOOLEAN_VALUES.has(lhs);\n const RHS_MATCH_BOOLEAN = TRUE_BOOLEAN_VALUES.has(rhs) || FALSE_BOOLEAN_VALUES.has(rhs);\n return (fromState, toState) => {\n let lhsMatch = lhs == ANY_STATE || lhs == fromState;\n let rhsMatch = rhs == ANY_STATE || rhs == toState;\n if (!lhsMatch && LHS_MATCH_BOOLEAN && typeof fromState === 'boolean') {\n lhsMatch = fromState ? TRUE_BOOLEAN_VALUES.has(lhs) : FALSE_BOOLEAN_VALUES.has(lhs);\n }\n if (!rhsMatch && RHS_MATCH_BOOLEAN && typeof toState === 'boolean') {\n rhsMatch = toState ? TRUE_BOOLEAN_VALUES.has(rhs) : FALSE_BOOLEAN_VALUES.has(rhs);\n }\n return lhsMatch && rhsMatch;\n };\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst SELF_TOKEN = ':self';\nconst SELF_TOKEN_REGEX = new RegExp(`\\s*${SELF_TOKEN}\\s*,?`, 'g');\n/*\n * [Validation]\n * The visitor code below will traverse the animation AST generated by the animation verb functions\n * (the output is a tree of objects) and attempt to perform a series of validations on the data. The\n * following corner-cases will be validated:\n *\n * 1. Overlap of animations\n * Given that a CSS property cannot be animated in more than one place at the same time, it's\n * important that this behavior is detected and validated. The way in which this occurs is that\n * each time a style property is examined, a string-map containing the property will be updated with\n * the start and end times for when the property is used within an animation step.\n *\n * If there are two or more parallel animations that are currently running (these are invoked by the\n * group()) on the same element then the validator will throw an error. Since the start/end timing\n * values are collected for each property then if the current animation step is animating the same\n * property and its timing values fall anywhere into the window of time that the property is\n * currently being animated within then this is what causes an error.\n *\n * 2. Timing values\n * The validator will validate to see if a timing value of `duration delay easing` or\n * `durationNumber` is valid or not.\n *\n * (note that upon validation the code below will replace the timing data with an object containing\n * {duration,delay,easing}.\n *\n * 3. Offset Validation\n * Each of the style() calls are allowed to have an offset value when placed inside of keyframes().\n * Offsets within keyframes() are considered valid when:\n *\n * - No offsets are used at all\n * - Each style() entry contains an offset value\n * - Each offset is between 0 and 1\n * - Each offset is greater to or equal than the previous one\n *\n * Otherwise an error will be thrown.\n */\nfunction buildAnimationAst(driver, metadata, errors, warnings) {\n return new AnimationAstBuilderVisitor(driver).build(metadata, errors, warnings);\n}\nconst ROOT_SELECTOR = '';\nclass AnimationAstBuilderVisitor {\n constructor(_driver) {\n this._driver = _driver;\n }\n build(metadata, errors, warnings) {\n const context = new AnimationAstBuilderContext(errors);\n this._resetContextStyleTimingState(context);\n const ast = visitDslNode(this, normalizeAnimationEntry(metadata), context);\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n if (context.unsupportedCSSPropertiesFound.size) {\n pushUnrecognizedPropertiesWarning(warnings, [...context.unsupportedCSSPropertiesFound.keys()]);\n }\n }\n return ast;\n }\n _resetContextStyleTimingState(context) {\n context.currentQuerySelector = ROOT_SELECTOR;\n context.collectedStyles = new Map();\n context.collectedStyles.set(ROOT_SELECTOR, new Map());\n context.currentTime = 0;\n }\n visitTrigger(metadata, context) {\n let queryCount = context.queryCount = 0;\n let depCount = context.depCount = 0;\n const states = [];\n const transitions = [];\n if (metadata.name.charAt(0) == '@') {\n context.errors.push(invalidTrigger());\n }\n metadata.definitions.forEach(def => {\n this._resetContextStyleTimingState(context);\n if (def.type == 0 /* AnimationMetadataType.State */) {\n const stateDef = def;\n const name = stateDef.name;\n name.toString().split(/\\s*,\\s*/).forEach(n => {\n stateDef.name = n;\n states.push(this.visitState(stateDef, context));\n });\n stateDef.name = name;\n }\n else if (def.type == 1 /* AnimationMetadataType.Transition */) {\n const transition = this.visitTransition(def, context);\n queryCount += transition.queryCount;\n depCount += transition.depCount;\n transitions.push(transition);\n }\n else {\n context.errors.push(invalidDefinition());\n }\n });\n return {\n type: 7 /* AnimationMetadataType.Trigger */,\n name: metadata.name,\n states,\n transitions,\n queryCount,\n depCount,\n options: null\n };\n }\n visitState(metadata, context) {\n const styleAst = this.visitStyle(metadata.styles, context);\n const astParams = (metadata.options && metadata.options.params) || null;\n if (styleAst.containsDynamicStyles) {\n const missingSubs = new Set();\n const params = astParams || {};\n styleAst.styles.forEach(style => {\n if (style instanceof Map) {\n style.forEach(value => {\n extractStyleParams(value).forEach(sub => {\n if (!params.hasOwnProperty(sub)) {\n missingSubs.add(sub);\n }\n });\n });\n }\n });\n if (missingSubs.size) {\n const missingSubsArr = iteratorToArray(missingSubs.values());\n context.errors.push(invalidState(metadata.name, missingSubsArr));\n }\n }\n return {\n type: 0 /* AnimationMetadataType.State */,\n name: metadata.name,\n style: styleAst,\n options: astParams ? { params: astParams } : null\n };\n }\n visitTransition(metadata, context) {\n context.queryCount = 0;\n context.depCount = 0;\n const animation = visitDslNode(this, normalizeAnimationEntry(metadata.animation), context);\n const matchers = parseTransitionExpr(metadata.expr, context.errors);\n return {\n type: 1 /* AnimationMetadataType.Transition */,\n matchers,\n animation,\n queryCount: context.queryCount,\n depCount: context.depCount,\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitSequence(metadata, context) {\n return {\n type: 2 /* AnimationMetadataType.Sequence */,\n steps: metadata.steps.map(s => visitDslNode(this, s, context)),\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitGroup(metadata, context) {\n const currentTime = context.currentTime;\n let furthestTime = 0;\n const steps = metadata.steps.map(step => {\n context.currentTime = currentTime;\n const innerAst = visitDslNode(this, step, context);\n furthestTime = Math.max(furthestTime, context.currentTime);\n return innerAst;\n });\n context.currentTime = furthestTime;\n return {\n type: 3 /* AnimationMetadataType.Group */,\n steps,\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitAnimate(metadata, context) {\n const timingAst = constructTimingAst(metadata.timings, context.errors);\n context.currentAnimateTimings = timingAst;\n let styleAst;\n let styleMetadata = metadata.styles ? metadata.styles : style({});\n if (styleMetadata.type == 5 /* AnimationMetadataType.Keyframes */) {\n styleAst = this.visitKeyframes(styleMetadata, context);\n }\n else {\n let styleMetadata = metadata.styles;\n let isEmpty = false;\n if (!styleMetadata) {\n isEmpty = true;\n const newStyleData = {};\n if (timingAst.easing) {\n newStyleData['easing'] = timingAst.easing;\n }\n styleMetadata = style(newStyleData);\n }\n context.currentTime += timingAst.duration + timingAst.delay;\n const _styleAst = this.visitStyle(styleMetadata, context);\n _styleAst.isEmptyStep = isEmpty;\n styleAst = _styleAst;\n }\n context.currentAnimateTimings = null;\n return {\n type: 4 /* AnimationMetadataType.Animate */,\n timings: timingAst,\n style: styleAst,\n options: null\n };\n }\n visitStyle(metadata, context) {\n const ast = this._makeStyleAst(metadata, context);\n this._validateStyleAst(ast, context);\n return ast;\n }\n _makeStyleAst(metadata, context) {\n const styles = [];\n const metadataStyles = Array.isArray(metadata.styles) ? metadata.styles : [metadata.styles];\n for (let styleTuple of metadataStyles) {\n if (typeof styleTuple === 'string') {\n if (styleTuple === AUTO_STYLE) {\n styles.push(styleTuple);\n }\n else {\n context.errors.push(invalidStyleValue(styleTuple));\n }\n }\n else {\n styles.push(convertToMap(styleTuple));\n }\n }\n let containsDynamicStyles = false;\n let collectedEasing = null;\n styles.forEach(styleData => {\n if (styleData instanceof Map) {\n if (styleData.has('easing')) {\n collectedEasing = styleData.get('easing');\n styleData.delete('easing');\n }\n if (!containsDynamicStyles) {\n for (let value of styleData.values()) {\n if (value.toString().indexOf(SUBSTITUTION_EXPR_START) >= 0) {\n containsDynamicStyles = true;\n break;\n }\n }\n }\n }\n });\n return {\n type: 6 /* AnimationMetadataType.Style */,\n styles,\n easing: collectedEasing,\n offset: metadata.offset,\n containsDynamicStyles,\n options: null\n };\n }\n _validateStyleAst(ast, context) {\n const timings = context.currentAnimateTimings;\n let endTime = context.currentTime;\n let startTime = context.currentTime;\n if (timings && startTime > 0) {\n startTime -= timings.duration + timings.delay;\n }\n ast.styles.forEach(tuple => {\n if (typeof tuple === 'string')\n return;\n tuple.forEach((value, prop) => {\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n if (!this._driver.validateStyleProperty(prop)) {\n tuple.delete(prop);\n context.unsupportedCSSPropertiesFound.add(prop);\n return;\n }\n }\n // This is guaranteed to have a defined Map at this querySelector location making it\n // safe to add the assertion here. It is set as a default empty map in prior methods.\n const collectedStyles = context.collectedStyles.get(context.currentQuerySelector);\n const collectedEntry = collectedStyles.get(prop);\n let updateCollectedStyle = true;\n if (collectedEntry) {\n if (startTime != endTime && startTime >= collectedEntry.startTime &&\n endTime <= collectedEntry.endTime) {\n context.errors.push(invalidParallelAnimation(prop, collectedEntry.startTime, collectedEntry.endTime, startTime, endTime));\n updateCollectedStyle = false;\n }\n // we always choose the smaller start time value since we\n // want to have a record of the entire animation window where\n // the style property is being animated in between\n startTime = collectedEntry.startTime;\n }\n if (updateCollectedStyle) {\n collectedStyles.set(prop, { startTime, endTime });\n }\n if (context.options) {\n validateStyleParams(value, context.options, context.errors);\n }\n });\n });\n }\n visitKeyframes(metadata, context) {\n const ast = { type: 5 /* AnimationMetadataType.Keyframes */, styles: [], options: null };\n if (!context.currentAnimateTimings) {\n context.errors.push(invalidKeyframes());\n return ast;\n }\n const MAX_KEYFRAME_OFFSET = 1;\n let totalKeyframesWithOffsets = 0;\n const offsets = [];\n let offsetsOutOfOrder = false;\n let keyframesOutOfRange = false;\n let previousOffset = 0;\n const keyframes = metadata.steps.map(styles => {\n const style = this._makeStyleAst(styles, context);\n let offsetVal = style.offset != null ? style.offset : consumeOffset(style.styles);\n let offset = 0;\n if (offsetVal != null) {\n totalKeyframesWithOffsets++;\n offset = style.offset = offsetVal;\n }\n keyframesOutOfRange = keyframesOutOfRange || offset < 0 || offset > 1;\n offsetsOutOfOrder = offsetsOutOfOrder || offset < previousOffset;\n previousOffset = offset;\n offsets.push(offset);\n return style;\n });\n if (keyframesOutOfRange) {\n context.errors.push(invalidOffset());\n }\n if (offsetsOutOfOrder) {\n context.errors.push(keyframeOffsetsOutOfOrder());\n }\n const length = metadata.steps.length;\n let generatedOffset = 0;\n if (totalKeyframesWithOffsets > 0 && totalKeyframesWithOffsets < length) {\n context.errors.push(keyframesMissingOffsets());\n }\n else if (totalKeyframesWithOffsets == 0) {\n generatedOffset = MAX_KEYFRAME_OFFSET / (length - 1);\n }\n const limit = length - 1;\n const currentTime = context.currentTime;\n const currentAnimateTimings = context.currentAnimateTimings;\n const animateDuration = currentAnimateTimings.duration;\n keyframes.forEach((kf, i) => {\n const offset = generatedOffset > 0 ? (i == limit ? 1 : (generatedOffset * i)) : offsets[i];\n const durationUpToThisFrame = offset * animateDuration;\n context.currentTime = currentTime + currentAnimateTimings.delay + durationUpToThisFrame;\n currentAnimateTimings.duration = durationUpToThisFrame;\n this._validateStyleAst(kf, context);\n kf.offset = offset;\n ast.styles.push(kf);\n });\n return ast;\n }\n visitReference(metadata, context) {\n return {\n type: 8 /* AnimationMetadataType.Reference */,\n animation: visitDslNode(this, normalizeAnimationEntry(metadata.animation), context),\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitAnimateChild(metadata, context) {\n context.depCount++;\n return {\n type: 9 /* AnimationMetadataType.AnimateChild */,\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitAnimateRef(metadata, context) {\n return {\n type: 10 /* AnimationMetadataType.AnimateRef */,\n animation: this.visitReference(metadata.animation, context),\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitQuery(metadata, context) {\n const parentSelector = context.currentQuerySelector;\n const options = (metadata.options || {});\n context.queryCount++;\n context.currentQuery = metadata;\n const [selector, includeSelf] = normalizeSelector(metadata.selector);\n context.currentQuerySelector =\n parentSelector.length ? (parentSelector + ' ' + selector) : selector;\n getOrSetDefaultValue(context.collectedStyles, context.currentQuerySelector, new Map());\n const animation = visitDslNode(this, normalizeAnimationEntry(metadata.animation), context);\n context.currentQuery = null;\n context.currentQuerySelector = parentSelector;\n return {\n type: 11 /* AnimationMetadataType.Query */,\n selector,\n limit: options.limit || 0,\n optional: !!options.optional,\n includeSelf,\n animation,\n originalSelector: metadata.selector,\n options: normalizeAnimationOptions(metadata.options)\n };\n }\n visitStagger(metadata, context) {\n if (!context.currentQuery) {\n context.errors.push(invalidStagger());\n }\n const timings = metadata.timings === 'full' ?\n { duration: 0, delay: 0, easing: 'full' } :\n resolveTiming(metadata.timings, context.errors, true);\n return {\n type: 12 /* AnimationMetadataType.Stagger */,\n animation: visitDslNode(this, normalizeAnimationEntry(metadata.animation), context),\n timings,\n options: null\n };\n }\n}\nfunction normalizeSelector(selector) {\n const hasAmpersand = selector.split(/\\s*,\\s*/).find(token => token == SELF_TOKEN) ? true : false;\n if (hasAmpersand) {\n selector = selector.replace(SELF_TOKEN_REGEX, '');\n }\n // Note: the :enter and :leave aren't normalized here since those\n // selectors are filled in at runtime during timeline building\n selector = selector.replace(/@\\*/g, NG_TRIGGER_SELECTOR)\n .replace(/@\\w+/g, match => NG_TRIGGER_SELECTOR + '-' + match.slice(1))\n .replace(/:animating/g, NG_ANIMATING_SELECTOR);\n return [selector, hasAmpersand];\n}\nfunction normalizeParams(obj) {\n return obj ? copyObj(obj) : null;\n}\nclass AnimationAstBuilderContext {\n constructor(errors) {\n this.errors = errors;\n this.queryCount = 0;\n this.depCount = 0;\n this.currentTransition = null;\n this.currentQuery = null;\n this.currentQuerySelector = null;\n this.currentAnimateTimings = null;\n this.currentTime = 0;\n this.collectedStyles = new Map();\n this.options = null;\n this.unsupportedCSSPropertiesFound = new Set();\n }\n}\nfunction consumeOffset(styles) {\n if (typeof styles == 'string')\n return null;\n let offset = null;\n if (Array.isArray(styles)) {\n styles.forEach(styleTuple => {\n if (styleTuple instanceof Map && styleTuple.has('offset')) {\n const obj = styleTuple;\n offset = parseFloat(obj.get('offset'));\n obj.delete('offset');\n }\n });\n }\n else if (styles instanceof Map && styles.has('offset')) {\n const obj = styles;\n offset = parseFloat(obj.get('offset'));\n obj.delete('offset');\n }\n return offset;\n}\nfunction constructTimingAst(value, errors) {\n if (value.hasOwnProperty('duration')) {\n return value;\n }\n if (typeof value == 'number') {\n const duration = resolveTiming(value, errors).duration;\n return makeTimingAst(duration, 0, '');\n }\n const strValue = value;\n const isDynamic = strValue.split(/\\s+/).some(v => v.charAt(0) == '{' && v.charAt(1) == '{');\n if (isDynamic) {\n const ast = makeTimingAst(0, 0, '');\n ast.dynamic = true;\n ast.strValue = strValue;\n return ast;\n }\n const timings = resolveTiming(strValue, errors);\n return makeTimingAst(timings.duration, timings.delay, timings.easing);\n}\nfunction normalizeAnimationOptions(options) {\n if (options) {\n options = copyObj(options);\n if (options['params']) {\n options['params'] = normalizeParams(options['params']);\n }\n }\n else {\n options = {};\n }\n return options;\n}\nfunction makeTimingAst(duration, delay, easing) {\n return { duration, delay, easing };\n}\n\nfunction createTimelineInstruction(element, keyframes, preStyleProps, postStyleProps, duration, delay, easing = null, subTimeline = false) {\n return {\n type: 1 /* AnimationTransitionInstructionType.TimelineAnimation */,\n element,\n keyframes,\n preStyleProps,\n postStyleProps,\n duration,\n delay,\n totalTime: duration + delay,\n easing,\n subTimeline\n };\n}\n\nclass ElementInstructionMap {\n constructor() {\n this._map = new Map();\n }\n get(element) {\n return this._map.get(element) || [];\n }\n append(element, instructions) {\n let existingInstructions = this._map.get(element);\n if (!existingInstructions) {\n this._map.set(element, existingInstructions = []);\n }\n existingInstructions.push(...instructions);\n }\n has(element) {\n return this._map.has(element);\n }\n clear() {\n this._map.clear();\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst ONE_FRAME_IN_MILLISECONDS = 1;\nconst ENTER_TOKEN = ':enter';\nconst ENTER_TOKEN_REGEX = new RegExp(ENTER_TOKEN, 'g');\nconst LEAVE_TOKEN = ':leave';\nconst LEAVE_TOKEN_REGEX = new RegExp(LEAVE_TOKEN, 'g');\n/*\n * The code within this file aims to generate web-animations-compatible keyframes from Angular's\n * animation DSL code.\n *\n * The code below will be converted from:\n *\n * ```\n * sequence([\n * style({ opacity: 0 }),\n * animate(1000, style({ opacity: 0 }))\n * ])\n * ```\n *\n * To:\n * ```\n * keyframes = [{ opacity: 0, offset: 0 }, { opacity: 1, offset: 1 }]\n * duration = 1000\n * delay = 0\n * easing = ''\n * ```\n *\n * For this operation to cover the combination of animation verbs (style, animate, group, etc...) a\n * combination of AST traversal and merge-sort-like algorithms are used.\n *\n * [AST Traversal]\n * Each of the animation verbs, when executed, will return an string-map object representing what\n * type of action it is (style, animate, group, etc...) and the data associated with it. This means\n * that when functional composition mix of these functions is evaluated (like in the example above)\n * then it will end up producing a tree of objects representing the animation itself.\n *\n * When this animation object tree is processed by the visitor code below it will visit each of the\n * verb statements within the visitor. And during each visit it will build the context of the\n * animation keyframes by interacting with the `TimelineBuilder`.\n *\n * [TimelineBuilder]\n * This class is responsible for tracking the styles and building a series of keyframe objects for a\n * timeline between a start and end time. The builder starts off with an initial timeline and each\n * time the AST comes across a `group()`, `keyframes()` or a combination of the two within a\n * `sequence()` then it will generate a sub timeline for each step as well as a new one after\n * they are complete.\n *\n * As the AST is traversed, the timing state on each of the timelines will be incremented. If a sub\n * timeline was created (based on one of the cases above) then the parent timeline will attempt to\n * merge the styles used within the sub timelines into itself (only with group() this will happen).\n * This happens with a merge operation (much like how the merge works in mergeSort) and it will only\n * copy the most recently used styles from the sub timelines into the parent timeline. This ensures\n * that if the styles are used later on in another phase of the animation then they will be the most\n * up-to-date values.\n *\n * [How Missing Styles Are Updated]\n * Each timeline has a `backFill` property which is responsible for filling in new styles into\n * already processed keyframes if a new style shows up later within the animation sequence.\n *\n * ```\n * sequence([\n * style({ width: 0 }),\n * animate(1000, style({ width: 100 })),\n * animate(1000, style({ width: 200 })),\n * animate(1000, style({ width: 300 }))\n * animate(1000, style({ width: 400, height: 400 })) // notice how `height` doesn't exist anywhere\n * else\n * ])\n * ```\n *\n * What is happening here is that the `height` value is added later in the sequence, but is missing\n * from all previous animation steps. Therefore when a keyframe is created it would also be missing\n * from all previous keyframes up until where it is first used. For the timeline keyframe generation\n * to properly fill in the style it will place the previous value (the value from the parent\n * timeline) or a default value of `*` into the backFill map. The `copyStyles` method in util.ts\n * handles propagating that backfill map to the styles object.\n *\n * When a sub-timeline is created it will have its own backFill property. This is done so that\n * styles present within the sub-timeline do not accidentally seep into the previous/future timeline\n * keyframes\n *\n * [Validation]\n * The code in this file is not responsible for validation. That functionality happens with within\n * the `AnimationValidatorVisitor` code.\n */\nfunction buildAnimationTimelines(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles = new Map(), finalStyles = new Map(), options, subInstructions, errors = []) {\n return new AnimationTimelineBuilderVisitor().buildKeyframes(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles, finalStyles, options, subInstructions, errors);\n}\nclass AnimationTimelineBuilderVisitor {\n buildKeyframes(driver, rootElement, ast, enterClassName, leaveClassName, startingStyles, finalStyles, options, subInstructions, errors = []) {\n subInstructions = subInstructions || new ElementInstructionMap();\n const context = new AnimationTimelineContext(driver, rootElement, subInstructions, enterClassName, leaveClassName, errors, []);\n context.options = options;\n const delay = options.delay ? resolveTimingValue(options.delay) : 0;\n context.currentTimeline.delayNextStep(delay);\n context.currentTimeline.setStyles([startingStyles], null, context.errors, options);\n visitDslNode(this, ast, context);\n // this checks to see if an actual animation happened\n const timelines = context.timelines.filter(timeline => timeline.containsAnimation());\n // note: we just want to apply the final styles for the rootElement, so we do not\n // just apply the styles to the last timeline but the last timeline which\n // element is the root one (basically `*`-styles are replaced with the actual\n // state style values only for the root element)\n if (timelines.length && finalStyles.size) {\n let lastRootTimeline;\n for (let i = timelines.length - 1; i >= 0; i--) {\n const timeline = timelines[i];\n if (timeline.element === rootElement) {\n lastRootTimeline = timeline;\n break;\n }\n }\n if (lastRootTimeline && !lastRootTimeline.allowOnlyTimelineStyles()) {\n lastRootTimeline.setStyles([finalStyles], null, context.errors, options);\n }\n }\n return timelines.length ?\n timelines.map(timeline => timeline.buildKeyframes()) :\n [createTimelineInstruction(rootElement, [], [], [], 0, delay, '', false)];\n }\n visitTrigger(ast, context) {\n // these values are not visited in this AST\n }\n visitState(ast, context) {\n // these values are not visited in this AST\n }\n visitTransition(ast, context) {\n // these values are not visited in this AST\n }\n visitAnimateChild(ast, context) {\n const elementInstructions = context.subInstructions.get(context.element);\n if (elementInstructions) {\n const innerContext = context.createSubContext(ast.options);\n const startTime = context.currentTimeline.currentTime;\n const endTime = this._visitSubInstructions(elementInstructions, innerContext, innerContext.options);\n if (startTime != endTime) {\n // we do this on the upper context because we created a sub context for\n // the sub child animations\n context.transformIntoNewTimeline(endTime);\n }\n }\n context.previousNode = ast;\n }\n visitAnimateRef(ast, context) {\n const innerContext = context.createSubContext(ast.options);\n innerContext.transformIntoNewTimeline();\n this._applyAnimationRefDelays([ast.options, ast.animation.options], context, innerContext);\n this.visitReference(ast.animation, innerContext);\n context.transformIntoNewTimeline(innerContext.currentTimeline.currentTime);\n context.previousNode = ast;\n }\n _applyAnimationRefDelays(animationsRefsOptions, context, innerContext) {\n for (const animationRefOptions of animationsRefsOptions) {\n const animationDelay = animationRefOptions?.delay;\n if (animationDelay) {\n const animationDelayValue = typeof animationDelay === 'number' ?\n animationDelay :\n resolveTimingValue(interpolateParams(animationDelay, animationRefOptions?.params ?? {}, context.errors));\n innerContext.delayNextStep(animationDelayValue);\n }\n }\n }\n _visitSubInstructions(instructions, context, options) {\n const startTime = context.currentTimeline.currentTime;\n let furthestTime = startTime;\n // this is a special-case for when a user wants to skip a sub\n // animation from being fired entirely.\n const duration = options.duration != null ? resolveTimingValue(options.duration) : null;\n const delay = options.delay != null ? resolveTimingValue(options.delay) : null;\n if (duration !== 0) {\n instructions.forEach(instruction => {\n const instructionTimings = context.appendInstructionToTimeline(instruction, duration, delay);\n furthestTime =\n Math.max(furthestTime, instructionTimings.duration + instructionTimings.delay);\n });\n }\n return furthestTime;\n }\n visitReference(ast, context) {\n context.updateOptions(ast.options, true);\n visitDslNode(this, ast.animation, context);\n context.previousNode = ast;\n }\n visitSequence(ast, context) {\n const subContextCount = context.subContextCount;\n let ctx = context;\n const options = ast.options;\n if (options && (options.params || options.delay)) {\n ctx = context.createSubContext(options);\n ctx.transformIntoNewTimeline();\n if (options.delay != null) {\n if (ctx.previousNode.type == 6 /* AnimationMetadataType.Style */) {\n ctx.currentTimeline.snapshotCurrentStyles();\n ctx.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;\n }\n const delay = resolveTimingValue(options.delay);\n ctx.delayNextStep(delay);\n }\n }\n if (ast.steps.length) {\n ast.steps.forEach(s => visitDslNode(this, s, ctx));\n // this is here just in case the inner steps only contain or end with a style() call\n ctx.currentTimeline.applyStylesToKeyframe();\n // this means that some animation function within the sequence\n // ended up creating a sub timeline (which means the current\n // timeline cannot overlap with the contents of the sequence)\n if (ctx.subContextCount > subContextCount) {\n ctx.transformIntoNewTimeline();\n }\n }\n context.previousNode = ast;\n }\n visitGroup(ast, context) {\n const innerTimelines = [];\n let furthestTime = context.currentTimeline.currentTime;\n const delay = ast.options && ast.options.delay ? resolveTimingValue(ast.options.delay) : 0;\n ast.steps.forEach(s => {\n const innerContext = context.createSubContext(ast.options);\n if (delay) {\n innerContext.delayNextStep(delay);\n }\n visitDslNode(this, s, innerContext);\n furthestTime = Math.max(furthestTime, innerContext.currentTimeline.currentTime);\n innerTimelines.push(innerContext.currentTimeline);\n });\n // this operation is run after the AST loop because otherwise\n // if the parent timeline's collected styles were updated then\n // it would pass in invalid data into the new-to-be forked items\n innerTimelines.forEach(timeline => context.currentTimeline.mergeTimelineCollectedStyles(timeline));\n context.transformIntoNewTimeline(furthestTime);\n context.previousNode = ast;\n }\n _visitTiming(ast, context) {\n if (ast.dynamic) {\n const strValue = ast.strValue;\n const timingValue = context.params ? interpolateParams(strValue, context.params, context.errors) : strValue;\n return resolveTiming(timingValue, context.errors);\n }\n else {\n return { duration: ast.duration, delay: ast.delay, easing: ast.easing };\n }\n }\n visitAnimate(ast, context) {\n const timings = context.currentAnimateTimings = this._visitTiming(ast.timings, context);\n const timeline = context.currentTimeline;\n if (timings.delay) {\n context.incrementTime(timings.delay);\n timeline.snapshotCurrentStyles();\n }\n const style = ast.style;\n if (style.type == 5 /* AnimationMetadataType.Keyframes */) {\n this.visitKeyframes(style, context);\n }\n else {\n context.incrementTime(timings.duration);\n this.visitStyle(style, context);\n timeline.applyStylesToKeyframe();\n }\n context.currentAnimateTimings = null;\n context.previousNode = ast;\n }\n visitStyle(ast, context) {\n const timeline = context.currentTimeline;\n const timings = context.currentAnimateTimings;\n // this is a special case for when a style() call\n // directly follows an animate() call (but not inside of an animate() call)\n if (!timings && timeline.hasCurrentStyleProperties()) {\n timeline.forwardFrame();\n }\n const easing = (timings && timings.easing) || ast.easing;\n if (ast.isEmptyStep) {\n timeline.applyEmptyStep(easing);\n }\n else {\n timeline.setStyles(ast.styles, easing, context.errors, context.options);\n }\n context.previousNode = ast;\n }\n visitKeyframes(ast, context) {\n const currentAnimateTimings = context.currentAnimateTimings;\n const startTime = (context.currentTimeline).duration;\n const duration = currentAnimateTimings.duration;\n const innerContext = context.createSubContext();\n const innerTimeline = innerContext.currentTimeline;\n innerTimeline.easing = currentAnimateTimings.easing;\n ast.styles.forEach(step => {\n const offset = step.offset || 0;\n innerTimeline.forwardTime(offset * duration);\n innerTimeline.setStyles(step.styles, step.easing, context.errors, context.options);\n innerTimeline.applyStylesToKeyframe();\n });\n // this will ensure that the parent timeline gets all the styles from\n // the child even if the new timeline below is not used\n context.currentTimeline.mergeTimelineCollectedStyles(innerTimeline);\n // we do this because the window between this timeline and the sub timeline\n // should ensure that the styles within are exactly the same as they were before\n context.transformIntoNewTimeline(startTime + duration);\n context.previousNode = ast;\n }\n visitQuery(ast, context) {\n // in the event that the first step before this is a style step we need\n // to ensure the styles are applied before the children are animated\n const startTime = context.currentTimeline.currentTime;\n const options = (ast.options || {});\n const delay = options.delay ? resolveTimingValue(options.delay) : 0;\n if (delay &&\n (context.previousNode.type === 6 /* AnimationMetadataType.Style */ ||\n (startTime == 0 && context.currentTimeline.hasCurrentStyleProperties()))) {\n context.currentTimeline.snapshotCurrentStyles();\n context.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;\n }\n let furthestTime = startTime;\n const elms = context.invokeQuery(ast.selector, ast.originalSelector, ast.limit, ast.includeSelf, options.optional ? true : false, context.errors);\n context.currentQueryTotal = elms.length;\n let sameElementTimeline = null;\n elms.forEach((element, i) => {\n context.currentQueryIndex = i;\n const innerContext = context.createSubContext(ast.options, element);\n if (delay) {\n innerContext.delayNextStep(delay);\n }\n if (element === context.element) {\n sameElementTimeline = innerContext.currentTimeline;\n }\n visitDslNode(this, ast.animation, innerContext);\n // this is here just incase the inner steps only contain or end\n // with a style() call (which is here to signal that this is a preparatory\n // call to style an element before it is animated again)\n innerContext.currentTimeline.applyStylesToKeyframe();\n const endTime = innerContext.currentTimeline.currentTime;\n furthestTime = Math.max(furthestTime, endTime);\n });\n context.currentQueryIndex = 0;\n context.currentQueryTotal = 0;\n context.transformIntoNewTimeline(furthestTime);\n if (sameElementTimeline) {\n context.currentTimeline.mergeTimelineCollectedStyles(sameElementTimeline);\n context.currentTimeline.snapshotCurrentStyles();\n }\n context.previousNode = ast;\n }\n visitStagger(ast, context) {\n const parentContext = context.parentContext;\n const tl = context.currentTimeline;\n const timings = ast.timings;\n const duration = Math.abs(timings.duration);\n const maxTime = duration * (context.currentQueryTotal - 1);\n let delay = duration * context.currentQueryIndex;\n let staggerTransformer = timings.duration < 0 ? 'reverse' : timings.easing;\n switch (staggerTransformer) {\n case 'reverse':\n delay = maxTime - delay;\n break;\n case 'full':\n delay = parentContext.currentStaggerTime;\n break;\n }\n const timeline = context.currentTimeline;\n if (delay) {\n timeline.delayNextStep(delay);\n }\n const startingTime = timeline.currentTime;\n visitDslNode(this, ast.animation, context);\n context.previousNode = ast;\n // time = duration + delay\n // the reason why this computation is so complex is because\n // the inner timeline may either have a delay value or a stretched\n // keyframe depending on if a subtimeline is not used or is used.\n parentContext.currentStaggerTime =\n (tl.currentTime - startingTime) + (tl.startTime - parentContext.currentTimeline.startTime);\n }\n}\nconst DEFAULT_NOOP_PREVIOUS_NODE = {};\nclass AnimationTimelineContext {\n constructor(_driver, element, subInstructions, _enterClassName, _leaveClassName, errors, timelines, initialTimeline) {\n this._driver = _driver;\n this.element = element;\n this.subInstructions = subInstructions;\n this._enterClassName = _enterClassName;\n this._leaveClassName = _leaveClassName;\n this.errors = errors;\n this.timelines = timelines;\n this.parentContext = null;\n this.currentAnimateTimings = null;\n this.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;\n this.subContextCount = 0;\n this.options = {};\n this.currentQueryIndex = 0;\n this.currentQueryTotal = 0;\n this.currentStaggerTime = 0;\n this.currentTimeline = initialTimeline || new TimelineBuilder(this._driver, element, 0);\n timelines.push(this.currentTimeline);\n }\n get params() {\n return this.options.params;\n }\n updateOptions(options, skipIfExists) {\n if (!options)\n return;\n const newOptions = options;\n let optionsToUpdate = this.options;\n // NOTE: this will get patched up when other animation methods support duration overrides\n if (newOptions.duration != null) {\n optionsToUpdate.duration = resolveTimingValue(newOptions.duration);\n }\n if (newOptions.delay != null) {\n optionsToUpdate.delay = resolveTimingValue(newOptions.delay);\n }\n const newParams = newOptions.params;\n if (newParams) {\n let paramsToUpdate = optionsToUpdate.params;\n if (!paramsToUpdate) {\n paramsToUpdate = this.options.params = {};\n }\n Object.keys(newParams).forEach(name => {\n if (!skipIfExists || !paramsToUpdate.hasOwnProperty(name)) {\n paramsToUpdate[name] = interpolateParams(newParams[name], paramsToUpdate, this.errors);\n }\n });\n }\n }\n _copyOptions() {\n const options = {};\n if (this.options) {\n const oldParams = this.options.params;\n if (oldParams) {\n const params = options['params'] = {};\n Object.keys(oldParams).forEach(name => {\n params[name] = oldParams[name];\n });\n }\n }\n return options;\n }\n createSubContext(options = null, element, newTime) {\n const target = element || this.element;\n const context = new AnimationTimelineContext(this._driver, target, this.subInstructions, this._enterClassName, this._leaveClassName, this.errors, this.timelines, this.currentTimeline.fork(target, newTime || 0));\n context.previousNode = this.previousNode;\n context.currentAnimateTimings = this.currentAnimateTimings;\n context.options = this._copyOptions();\n context.updateOptions(options);\n context.currentQueryIndex = this.currentQueryIndex;\n context.currentQueryTotal = this.currentQueryTotal;\n context.parentContext = this;\n this.subContextCount++;\n return context;\n }\n transformIntoNewTimeline(newTime) {\n this.previousNode = DEFAULT_NOOP_PREVIOUS_NODE;\n this.currentTimeline = this.currentTimeline.fork(this.element, newTime);\n this.timelines.push(this.currentTimeline);\n return this.currentTimeline;\n }\n appendInstructionToTimeline(instruction, duration, delay) {\n const updatedTimings = {\n duration: duration != null ? duration : instruction.duration,\n delay: this.currentTimeline.currentTime + (delay != null ? delay : 0) + instruction.delay,\n easing: ''\n };\n const builder = new SubTimelineBuilder(this._driver, instruction.element, instruction.keyframes, instruction.preStyleProps, instruction.postStyleProps, updatedTimings, instruction.stretchStartingKeyframe);\n this.timelines.push(builder);\n return updatedTimings;\n }\n incrementTime(time) {\n this.currentTimeline.forwardTime(this.currentTimeline.duration + time);\n }\n delayNextStep(delay) {\n // negative delays are not yet supported\n if (delay > 0) {\n this.currentTimeline.delayNextStep(delay);\n }\n }\n invokeQuery(selector, originalSelector, limit, includeSelf, optional, errors) {\n let results = [];\n if (includeSelf) {\n results.push(this.element);\n }\n if (selector.length > 0) { // only if :self is used then the selector can be empty\n selector = selector.replace(ENTER_TOKEN_REGEX, '.' + this._enterClassName);\n selector = selector.replace(LEAVE_TOKEN_REGEX, '.' + this._leaveClassName);\n const multi = limit != 1;\n let elements = this._driver.query(this.element, selector, multi);\n if (limit !== 0) {\n elements = limit < 0 ? elements.slice(elements.length + limit, elements.length) :\n elements.slice(0, limit);\n }\n results.push(...elements);\n }\n if (!optional && results.length == 0) {\n errors.push(invalidQuery(originalSelector));\n }\n return results;\n }\n}\nclass TimelineBuilder {\n constructor(_driver, element, startTime, _elementTimelineStylesLookup) {\n this._driver = _driver;\n this.element = element;\n this.startTime = startTime;\n this._elementTimelineStylesLookup = _elementTimelineStylesLookup;\n this.duration = 0;\n this._previousKeyframe = new Map();\n this._currentKeyframe = new Map();\n this._keyframes = new Map();\n this._styleSummary = new Map();\n this._localTimelineStyles = new Map();\n this._pendingStyles = new Map();\n this._backFill = new Map();\n this._currentEmptyStepKeyframe = null;\n if (!this._elementTimelineStylesLookup) {\n this._elementTimelineStylesLookup = new Map();\n }\n this._globalTimelineStyles = this._elementTimelineStylesLookup.get(element);\n if (!this._globalTimelineStyles) {\n this._globalTimelineStyles = this._localTimelineStyles;\n this._elementTimelineStylesLookup.set(element, this._localTimelineStyles);\n }\n this._loadKeyframe();\n }\n containsAnimation() {\n switch (this._keyframes.size) {\n case 0:\n return false;\n case 1:\n return this.hasCurrentStyleProperties();\n default:\n return true;\n }\n }\n hasCurrentStyleProperties() {\n return this._currentKeyframe.size > 0;\n }\n get currentTime() {\n return this.startTime + this.duration;\n }\n delayNextStep(delay) {\n // in the event that a style() step is placed right before a stagger()\n // and that style() step is the very first style() value in the animation\n // then we need to make a copy of the keyframe [0, copy, 1] so that the delay\n // properly applies the style() values to work with the stagger...\n const hasPreStyleStep = this._keyframes.size === 1 && this._pendingStyles.size;\n if (this.duration || hasPreStyleStep) {\n this.forwardTime(this.currentTime + delay);\n if (hasPreStyleStep) {\n this.snapshotCurrentStyles();\n }\n }\n else {\n this.startTime += delay;\n }\n }\n fork(element, currentTime) {\n this.applyStylesToKeyframe();\n return new TimelineBuilder(this._driver, element, currentTime || this.currentTime, this._elementTimelineStylesLookup);\n }\n _loadKeyframe() {\n if (this._currentKeyframe) {\n this._previousKeyframe = this._currentKeyframe;\n }\n this._currentKeyframe = this._keyframes.get(this.duration);\n if (!this._currentKeyframe) {\n this._currentKeyframe = new Map();\n this._keyframes.set(this.duration, this._currentKeyframe);\n }\n }\n forwardFrame() {\n this.duration += ONE_FRAME_IN_MILLISECONDS;\n this._loadKeyframe();\n }\n forwardTime(time) {\n this.applyStylesToKeyframe();\n this.duration = time;\n this._loadKeyframe();\n }\n _updateStyle(prop, value) {\n this._localTimelineStyles.set(prop, value);\n this._globalTimelineStyles.set(prop, value);\n this._styleSummary.set(prop, { time: this.currentTime, value });\n }\n allowOnlyTimelineStyles() {\n return this._currentEmptyStepKeyframe !== this._currentKeyframe;\n }\n applyEmptyStep(easing) {\n if (easing) {\n this._previousKeyframe.set('easing', easing);\n }\n // special case for animate(duration):\n // all missing styles are filled with a `*` value then\n // if any destination styles are filled in later on the same\n // keyframe then they will override the overridden styles\n // We use `_globalTimelineStyles` here because there may be\n // styles in previous keyframes that are not present in this timeline\n for (let [prop, value] of this._globalTimelineStyles) {\n this._backFill.set(prop, value || AUTO_STYLE);\n this._currentKeyframe.set(prop, AUTO_STYLE);\n }\n this._currentEmptyStepKeyframe = this._currentKeyframe;\n }\n setStyles(input, easing, errors, options) {\n if (easing) {\n this._previousKeyframe.set('easing', easing);\n }\n const params = (options && options.params) || {};\n const styles = flattenStyles(input, this._globalTimelineStyles);\n for (let [prop, value] of styles) {\n const val = interpolateParams(value, params, errors);\n this._pendingStyles.set(prop, val);\n if (!this._localTimelineStyles.has(prop)) {\n this._backFill.set(prop, this._globalTimelineStyles.get(prop) ?? AUTO_STYLE);\n }\n this._updateStyle(prop, val);\n }\n }\n applyStylesToKeyframe() {\n if (this._pendingStyles.size == 0)\n return;\n this._pendingStyles.forEach((val, prop) => {\n this._currentKeyframe.set(prop, val);\n });\n this._pendingStyles.clear();\n this._localTimelineStyles.forEach((val, prop) => {\n if (!this._currentKeyframe.has(prop)) {\n this._currentKeyframe.set(prop, val);\n }\n });\n }\n snapshotCurrentStyles() {\n for (let [prop, val] of this._localTimelineStyles) {\n this._pendingStyles.set(prop, val);\n this._updateStyle(prop, val);\n }\n }\n getFinalKeyframe() {\n return this._keyframes.get(this.duration);\n }\n get properties() {\n const properties = [];\n for (let prop in this._currentKeyframe) {\n properties.push(prop);\n }\n return properties;\n }\n mergeTimelineCollectedStyles(timeline) {\n timeline._styleSummary.forEach((details1, prop) => {\n const details0 = this._styleSummary.get(prop);\n if (!details0 || details1.time > details0.time) {\n this._updateStyle(prop, details1.value);\n }\n });\n }\n buildKeyframes() {\n this.applyStylesToKeyframe();\n const preStyleProps = new Set();\n const postStyleProps = new Set();\n const isEmpty = this._keyframes.size === 1 && this.duration === 0;\n let finalKeyframes = [];\n this._keyframes.forEach((keyframe, time) => {\n const finalKeyframe = copyStyles(keyframe, new Map(), this._backFill);\n finalKeyframe.forEach((value, prop) => {\n if (value === ɵPRE_STYLE) {\n preStyleProps.add(prop);\n }\n else if (value === AUTO_STYLE) {\n postStyleProps.add(prop);\n }\n });\n if (!isEmpty) {\n finalKeyframe.set('offset', time / this.duration);\n }\n finalKeyframes.push(finalKeyframe);\n });\n const preProps = preStyleProps.size ? iteratorToArray(preStyleProps.values()) : [];\n const postProps = postStyleProps.size ? iteratorToArray(postStyleProps.values()) : [];\n // special case for a 0-second animation (which is designed just to place styles onscreen)\n if (isEmpty) {\n const kf0 = finalKeyframes[0];\n const kf1 = new Map(kf0);\n kf0.set('offset', 0);\n kf1.set('offset', 1);\n finalKeyframes = [kf0, kf1];\n }\n return createTimelineInstruction(this.element, finalKeyframes, preProps, postProps, this.duration, this.startTime, this.easing, false);\n }\n}\nclass SubTimelineBuilder extends TimelineBuilder {\n constructor(driver, element, keyframes, preStyleProps, postStyleProps, timings, _stretchStartingKeyframe = false) {\n super(driver, element, timings.delay);\n this.keyframes = keyframes;\n this.preStyleProps = preStyleProps;\n this.postStyleProps = postStyleProps;\n this._stretchStartingKeyframe = _stretchStartingKeyframe;\n this.timings = { duration: timings.duration, delay: timings.delay, easing: timings.easing };\n }\n containsAnimation() {\n return this.keyframes.length > 1;\n }\n buildKeyframes() {\n let keyframes = this.keyframes;\n let { delay, duration, easing } = this.timings;\n if (this._stretchStartingKeyframe && delay) {\n const newKeyframes = [];\n const totalTime = duration + delay;\n const startingGap = delay / totalTime;\n // the original starting keyframe now starts once the delay is done\n const newFirstKeyframe = copyStyles(keyframes[0]);\n newFirstKeyframe.set('offset', 0);\n newKeyframes.push(newFirstKeyframe);\n const oldFirstKeyframe = copyStyles(keyframes[0]);\n oldFirstKeyframe.set('offset', roundOffset(startingGap));\n newKeyframes.push(oldFirstKeyframe);\n /*\n When the keyframe is stretched then it means that the delay before the animation\n starts is gone. Instead the first keyframe is placed at the start of the animation\n and it is then copied to where it starts when the original delay is over. This basically\n means nothing animates during that delay, but the styles are still rendered. For this\n to work the original offset values that exist in the original keyframes must be \"warped\"\n so that they can take the new keyframe + delay into account.\n \n delay=1000, duration=1000, keyframes = 0 .5 1\n \n turns into\n \n delay=0, duration=2000, keyframes = 0 .33 .66 1\n */\n // offsets between 1 ... n -1 are all warped by the keyframe stretch\n const limit = keyframes.length - 1;\n for (let i = 1; i <= limit; i++) {\n let kf = copyStyles(keyframes[i]);\n const oldOffset = kf.get('offset');\n const timeAtKeyframe = delay + oldOffset * duration;\n kf.set('offset', roundOffset(timeAtKeyframe / totalTime));\n newKeyframes.push(kf);\n }\n // the new starting keyframe should be added at the start\n duration = totalTime;\n delay = 0;\n easing = '';\n keyframes = newKeyframes;\n }\n return createTimelineInstruction(this.element, keyframes, this.preStyleProps, this.postStyleProps, duration, delay, easing, true);\n }\n}\nfunction roundOffset(offset, decimalPoints = 3) {\n const mult = Math.pow(10, decimalPoints - 1);\n return Math.round(offset * mult) / mult;\n}\nfunction flattenStyles(input, allStyles) {\n const styles = new Map();\n let allProperties;\n input.forEach(token => {\n if (token === '*') {\n allProperties = allProperties || allStyles.keys();\n for (let prop of allProperties) {\n styles.set(prop, AUTO_STYLE);\n }\n }\n else {\n copyStyles(token, styles);\n }\n });\n return styles;\n}\n\nclass Animation {\n constructor(_driver, input) {\n this._driver = _driver;\n const errors = [];\n const warnings = [];\n const ast = buildAnimationAst(_driver, input, errors, warnings);\n if (errors.length) {\n throw validationFailed(errors);\n }\n if (warnings.length) {\n warnValidation(warnings);\n }\n this._animationAst = ast;\n }\n buildTimelines(element, startingStyles, destinationStyles, options, subInstructions) {\n const start = Array.isArray(startingStyles) ? normalizeStyles(startingStyles) :\n startingStyles;\n const dest = Array.isArray(destinationStyles) ? normalizeStyles(destinationStyles) :\n destinationStyles;\n const errors = [];\n subInstructions = subInstructions || new ElementInstructionMap();\n const result = buildAnimationTimelines(this._driver, element, this._animationAst, ENTER_CLASSNAME, LEAVE_CLASSNAME, start, dest, options, subInstructions, errors);\n if (errors.length) {\n throw buildingFailed(errors);\n }\n return result;\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * @publicApi\n */\nclass AnimationStyleNormalizer {\n}\n/**\n * @publicApi\n */\nclass NoopAnimationStyleNormalizer {\n normalizePropertyName(propertyName, errors) {\n return propertyName;\n }\n normalizeStyleValue(userProvidedProperty, normalizedProperty, value, errors) {\n return value;\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst DIMENSIONAL_PROP_SET = new Set([\n 'width',\n 'height',\n 'minWidth',\n 'minHeight',\n 'maxWidth',\n 'maxHeight',\n 'left',\n 'top',\n 'bottom',\n 'right',\n 'fontSize',\n 'outlineWidth',\n 'outlineOffset',\n 'paddingTop',\n 'paddingLeft',\n 'paddingBottom',\n 'paddingRight',\n 'marginTop',\n 'marginLeft',\n 'marginBottom',\n 'marginRight',\n 'borderRadius',\n 'borderWidth',\n 'borderTopWidth',\n 'borderLeftWidth',\n 'borderRightWidth',\n 'borderBottomWidth',\n 'textIndent',\n 'perspective'\n]);\nclass WebAnimationsStyleNormalizer extends AnimationStyleNormalizer {\n normalizePropertyName(propertyName, errors) {\n return dashCaseToCamelCase(propertyName);\n }\n normalizeStyleValue(userProvidedProperty, normalizedProperty, value, errors) {\n let unit = '';\n const strVal = value.toString().trim();\n if (DIMENSIONAL_PROP_SET.has(normalizedProperty) && value !== 0 && value !== '0') {\n if (typeof value === 'number') {\n unit = 'px';\n }\n else {\n const valAndSuffixMatch = value.match(/^[+-]?[\\d\\.]+([a-z]*)$/);\n if (valAndSuffixMatch && valAndSuffixMatch[1].length == 0) {\n errors.push(invalidCssUnitValue(userProvidedProperty, value));\n }\n }\n }\n return strVal + unit;\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nfunction createTransitionInstruction(element, triggerName, fromState, toState, isRemovalTransition, fromStyles, toStyles, timelines, queriedElements, preStyleProps, postStyleProps, totalTime, errors) {\n return {\n type: 0 /* AnimationTransitionInstructionType.TransitionAnimation */,\n element,\n triggerName,\n isRemovalTransition,\n fromState,\n fromStyles,\n toState,\n toStyles,\n timelines,\n queriedElements,\n preStyleProps,\n postStyleProps,\n totalTime,\n errors\n };\n}\n\nconst EMPTY_OBJECT = {};\nclass AnimationTransitionFactory {\n constructor(_triggerName, ast, _stateStyles) {\n this._triggerName = _triggerName;\n this.ast = ast;\n this._stateStyles = _stateStyles;\n }\n match(currentState, nextState, element, params) {\n return oneOrMoreTransitionsMatch(this.ast.matchers, currentState, nextState, element, params);\n }\n buildStyles(stateName, params, errors) {\n let styler = this._stateStyles.get('*');\n if (stateName !== undefined) {\n styler = this._stateStyles.get(stateName?.toString()) || styler;\n }\n return styler ? styler.buildStyles(params, errors) : new Map();\n }\n build(driver, element, currentState, nextState, enterClassName, leaveClassName, currentOptions, nextOptions, subInstructions, skipAstBuild) {\n const errors = [];\n const transitionAnimationParams = this.ast.options && this.ast.options.params || EMPTY_OBJECT;\n const currentAnimationParams = currentOptions && currentOptions.params || EMPTY_OBJECT;\n const currentStateStyles = this.buildStyles(currentState, currentAnimationParams, errors);\n const nextAnimationParams = nextOptions && nextOptions.params || EMPTY_OBJECT;\n const nextStateStyles = this.buildStyles(nextState, nextAnimationParams, errors);\n const queriedElements = new Set();\n const preStyleMap = new Map();\n const postStyleMap = new Map();\n const isRemoval = nextState === 'void';\n const animationOptions = {\n params: applyParamDefaults(nextAnimationParams, transitionAnimationParams),\n delay: this.ast.options?.delay,\n };\n const timelines = skipAstBuild ?\n [] :\n buildAnimationTimelines(driver, element, this.ast.animation, enterClassName, leaveClassName, currentStateStyles, nextStateStyles, animationOptions, subInstructions, errors);\n let totalTime = 0;\n timelines.forEach(tl => {\n totalTime = Math.max(tl.duration + tl.delay, totalTime);\n });\n if (errors.length) {\n return createTransitionInstruction(element, this._triggerName, currentState, nextState, isRemoval, currentStateStyles, nextStateStyles, [], [], preStyleMap, postStyleMap, totalTime, errors);\n }\n timelines.forEach(tl => {\n const elm = tl.element;\n const preProps = getOrSetDefaultValue(preStyleMap, elm, new Set());\n tl.preStyleProps.forEach(prop => preProps.add(prop));\n const postProps = getOrSetDefaultValue(postStyleMap, elm, new Set());\n tl.postStyleProps.forEach(prop => postProps.add(prop));\n if (elm !== element) {\n queriedElements.add(elm);\n }\n });\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n checkNonAnimatableInTimelines(timelines, this._triggerName, driver);\n }\n const queriedElementsList = iteratorToArray(queriedElements.values());\n return createTransitionInstruction(element, this._triggerName, currentState, nextState, isRemoval, currentStateStyles, nextStateStyles, timelines, queriedElementsList, preStyleMap, postStyleMap, totalTime);\n }\n}\n/**\n * Checks inside a set of timelines if they try to animate a css property which is not considered\n * animatable, in that case it prints a warning on the console.\n * Besides that the function doesn't have any other effect.\n *\n * Note: this check is done here after the timelines are built instead of doing on a lower level so\n * that we can make sure that the warning appears only once per instruction (we can aggregate here\n * all the issues instead of finding them separately).\n *\n * @param timelines The built timelines for the current instruction.\n * @param triggerName The name of the trigger for the current instruction.\n * @param driver Animation driver used to perform the check.\n *\n */\nfunction checkNonAnimatableInTimelines(timelines, triggerName, driver) {\n if (!driver.validateAnimatableStyleProperty) {\n return;\n }\n const invalidNonAnimatableProps = new Set();\n timelines.forEach(({ keyframes }) => {\n const nonAnimatablePropsInitialValues = new Map();\n keyframes.forEach(keyframe => {\n for (const [prop, value] of keyframe.entries()) {\n if (!driver.validateAnimatableStyleProperty(prop)) {\n if (nonAnimatablePropsInitialValues.has(prop) && !invalidNonAnimatableProps.has(prop)) {\n const propInitialValue = nonAnimatablePropsInitialValues.get(prop);\n if (propInitialValue !== value) {\n invalidNonAnimatableProps.add(prop);\n }\n }\n else {\n nonAnimatablePropsInitialValues.set(prop, value);\n }\n }\n }\n });\n });\n if (invalidNonAnimatableProps.size > 0) {\n console.warn(`Warning: The animation trigger \"${triggerName}\" is attempting to animate the following` +\n ' not animatable properties: ' + Array.from(invalidNonAnimatableProps).join(', ') + '\\n' +\n '(to check the list of all animatable properties visit https://developer.mozilla.org/en-US/docs/Web/CSS/CSS_animated_properties)');\n }\n}\nfunction oneOrMoreTransitionsMatch(matchFns, currentState, nextState, element, params) {\n return matchFns.some(fn => fn(currentState, nextState, element, params));\n}\nfunction applyParamDefaults(userParams, defaults) {\n const result = copyObj(defaults);\n for (const key in userParams) {\n if (userParams.hasOwnProperty(key) && userParams[key] != null) {\n result[key] = userParams[key];\n }\n }\n return result;\n}\nclass AnimationStateStyles {\n constructor(styles, defaultParams, normalizer) {\n this.styles = styles;\n this.defaultParams = defaultParams;\n this.normalizer = normalizer;\n }\n buildStyles(params, errors) {\n const finalStyles = new Map();\n const combinedParams = copyObj(this.defaultParams);\n Object.keys(params).forEach(key => {\n const value = params[key];\n if (value !== null) {\n combinedParams[key] = value;\n }\n });\n this.styles.styles.forEach(value => {\n if (typeof value !== 'string') {\n value.forEach((val, prop) => {\n if (val) {\n val = interpolateParams(val, combinedParams, errors);\n }\n const normalizedProp = this.normalizer.normalizePropertyName(prop, errors);\n val = this.normalizer.normalizeStyleValue(prop, normalizedProp, val, errors);\n finalStyles.set(normalizedProp, val);\n });\n }\n });\n return finalStyles;\n }\n}\n\nfunction buildTrigger(name, ast, normalizer) {\n return new AnimationTrigger(name, ast, normalizer);\n}\nclass AnimationTrigger {\n constructor(name, ast, _normalizer) {\n this.name = name;\n this.ast = ast;\n this._normalizer = _normalizer;\n this.transitionFactories = [];\n this.states = new Map();\n ast.states.forEach(ast => {\n const defaultParams = (ast.options && ast.options.params) || {};\n this.states.set(ast.name, new AnimationStateStyles(ast.style, defaultParams, _normalizer));\n });\n balanceProperties(this.states, 'true', '1');\n balanceProperties(this.states, 'false', '0');\n ast.transitions.forEach(ast => {\n this.transitionFactories.push(new AnimationTransitionFactory(name, ast, this.states));\n });\n this.fallbackTransition = createFallbackTransition(name, this.states, this._normalizer);\n }\n get containsQueries() {\n return this.ast.queryCount > 0;\n }\n matchTransition(currentState, nextState, element, params) {\n const entry = this.transitionFactories.find(f => f.match(currentState, nextState, element, params));\n return entry || null;\n }\n matchStyles(currentState, params, errors) {\n return this.fallbackTransition.buildStyles(currentState, params, errors);\n }\n}\nfunction createFallbackTransition(triggerName, states, normalizer) {\n const matchers = [(fromState, toState) => true];\n const animation = { type: 2 /* AnimationMetadataType.Sequence */, steps: [], options: null };\n const transition = {\n type: 1 /* AnimationMetadataType.Transition */,\n animation,\n matchers,\n options: null,\n queryCount: 0,\n depCount: 0\n };\n return new AnimationTransitionFactory(triggerName, transition, states);\n}\nfunction balanceProperties(stateMap, key1, key2) {\n if (stateMap.has(key1)) {\n if (!stateMap.has(key2)) {\n stateMap.set(key2, stateMap.get(key1));\n }\n }\n else if (stateMap.has(key2)) {\n stateMap.set(key1, stateMap.get(key2));\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst EMPTY_INSTRUCTION_MAP = new ElementInstructionMap();\nclass TimelineAnimationEngine {\n constructor(bodyNode, _driver, _normalizer) {\n this.bodyNode = bodyNode;\n this._driver = _driver;\n this._normalizer = _normalizer;\n this._animations = new Map();\n this._playersById = new Map();\n this.players = [];\n }\n register(id, metadata) {\n const errors = [];\n const warnings = [];\n const ast = buildAnimationAst(this._driver, metadata, errors, warnings);\n if (errors.length) {\n throw registerFailed(errors);\n }\n else {\n if (warnings.length) {\n warnRegister(warnings);\n }\n this._animations.set(id, ast);\n }\n }\n _buildPlayer(i, preStyles, postStyles) {\n const element = i.element;\n const keyframes = normalizeKeyframes$1(this._driver, this._normalizer, element, i.keyframes, preStyles, postStyles);\n return this._driver.animate(element, keyframes, i.duration, i.delay, i.easing, [], true);\n }\n create(id, element, options = {}) {\n const errors = [];\n const ast = this._animations.get(id);\n let instructions;\n const autoStylesMap = new Map();\n if (ast) {\n instructions = buildAnimationTimelines(this._driver, element, ast, ENTER_CLASSNAME, LEAVE_CLASSNAME, new Map(), new Map(), options, EMPTY_INSTRUCTION_MAP, errors);\n instructions.forEach(inst => {\n const styles = getOrSetDefaultValue(autoStylesMap, inst.element, new Map());\n inst.postStyleProps.forEach(prop => styles.set(prop, null));\n });\n }\n else {\n errors.push(missingOrDestroyedAnimation());\n instructions = [];\n }\n if (errors.length) {\n throw createAnimationFailed(errors);\n }\n autoStylesMap.forEach((styles, element) => {\n styles.forEach((_, prop) => {\n styles.set(prop, this._driver.computeStyle(element, prop, AUTO_STYLE));\n });\n });\n const players = instructions.map(i => {\n const styles = autoStylesMap.get(i.element);\n return this._buildPlayer(i, new Map(), styles);\n });\n const player = optimizeGroupPlayer(players);\n this._playersById.set(id, player);\n player.onDestroy(() => this.destroy(id));\n this.players.push(player);\n return player;\n }\n destroy(id) {\n const player = this._getPlayer(id);\n player.destroy();\n this._playersById.delete(id);\n const index = this.players.indexOf(player);\n if (index >= 0) {\n this.players.splice(index, 1);\n }\n }\n _getPlayer(id) {\n const player = this._playersById.get(id);\n if (!player) {\n throw missingPlayer(id);\n }\n return player;\n }\n listen(id, element, eventName, callback) {\n // triggerName, fromState, toState are all ignored for timeline animations\n const baseEvent = makeAnimationEvent(element, '', '', '');\n listenOnPlayer(this._getPlayer(id), eventName, baseEvent, callback);\n return () => { };\n }\n command(id, element, command, args) {\n if (command == 'register') {\n this.register(id, args[0]);\n return;\n }\n if (command == 'create') {\n const options = (args[0] || {});\n this.create(id, element, options);\n return;\n }\n const player = this._getPlayer(id);\n switch (command) {\n case 'play':\n player.play();\n break;\n case 'pause':\n player.pause();\n break;\n case 'reset':\n player.reset();\n break;\n case 'restart':\n player.restart();\n break;\n case 'finish':\n player.finish();\n break;\n case 'init':\n player.init();\n break;\n case 'setPosition':\n player.setPosition(parseFloat(args[0]));\n break;\n case 'destroy':\n this.destroy(id);\n break;\n }\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst QUEUED_CLASSNAME = 'ng-animate-queued';\nconst QUEUED_SELECTOR = '.ng-animate-queued';\nconst DISABLED_CLASSNAME = 'ng-animate-disabled';\nconst DISABLED_SELECTOR = '.ng-animate-disabled';\nconst STAR_CLASSNAME = 'ng-star-inserted';\nconst STAR_SELECTOR = '.ng-star-inserted';\nconst EMPTY_PLAYER_ARRAY = [];\nconst NULL_REMOVAL_STATE = {\n namespaceId: '',\n setForRemoval: false,\n setForMove: false,\n hasAnimation: false,\n removedBeforeQueried: false\n};\nconst NULL_REMOVED_QUERIED_STATE = {\n namespaceId: '',\n setForMove: false,\n setForRemoval: false,\n hasAnimation: false,\n removedBeforeQueried: true\n};\nconst REMOVAL_FLAG = '__ng_removed';\nclass StateValue {\n constructor(input, namespaceId = '') {\n this.namespaceId = namespaceId;\n const isObj = input && input.hasOwnProperty('value');\n const value = isObj ? input['value'] : input;\n this.value = normalizeTriggerValue(value);\n if (isObj) {\n const options = copyObj(input);\n delete options['value'];\n this.options = options;\n }\n else {\n this.options = {};\n }\n if (!this.options.params) {\n this.options.params = {};\n }\n }\n get params() {\n return this.options.params;\n }\n absorbOptions(options) {\n const newParams = options.params;\n if (newParams) {\n const oldParams = this.options.params;\n Object.keys(newParams).forEach(prop => {\n if (oldParams[prop] == null) {\n oldParams[prop] = newParams[prop];\n }\n });\n }\n }\n}\nconst VOID_VALUE = 'void';\nconst DEFAULT_STATE_VALUE = new StateValue(VOID_VALUE);\nclass AnimationTransitionNamespace {\n constructor(id, hostElement, _engine) {\n this.id = id;\n this.hostElement = hostElement;\n this._engine = _engine;\n this.players = [];\n this._triggers = new Map();\n this._queue = [];\n this._elementListeners = new Map();\n this._hostClassName = 'ng-tns-' + id;\n addClass(hostElement, this._hostClassName);\n }\n listen(element, name, phase, callback) {\n if (!this._triggers.has(name)) {\n throw missingTrigger(phase, name);\n }\n if (phase == null || phase.length == 0) {\n throw missingEvent(name);\n }\n if (!isTriggerEventValid(phase)) {\n throw unsupportedTriggerEvent(phase, name);\n }\n const listeners = getOrSetDefaultValue(this._elementListeners, element, []);\n const data = { name, phase, callback };\n listeners.push(data);\n const triggersWithStates = getOrSetDefaultValue(this._engine.statesByElement, element, new Map());\n if (!triggersWithStates.has(name)) {\n addClass(element, NG_TRIGGER_CLASSNAME);\n addClass(element, NG_TRIGGER_CLASSNAME + '-' + name);\n triggersWithStates.set(name, DEFAULT_STATE_VALUE);\n }\n return () => {\n // the event listener is removed AFTER the flush has occurred such\n // that leave animations callbacks can fire (otherwise if the node\n // is removed in between then the listeners would be deregistered)\n this._engine.afterFlush(() => {\n const index = listeners.indexOf(data);\n if (index >= 0) {\n listeners.splice(index, 1);\n }\n if (!this._triggers.has(name)) {\n triggersWithStates.delete(name);\n }\n });\n };\n }\n register(name, ast) {\n if (this._triggers.has(name)) {\n // throw\n return false;\n }\n else {\n this._triggers.set(name, ast);\n return true;\n }\n }\n _getTrigger(name) {\n const trigger = this._triggers.get(name);\n if (!trigger) {\n throw unregisteredTrigger(name);\n }\n return trigger;\n }\n trigger(element, triggerName, value, defaultToFallback = true) {\n const trigger = this._getTrigger(triggerName);\n const player = new TransitionAnimationPlayer(this.id, triggerName, element);\n let triggersWithStates = this._engine.statesByElement.get(element);\n if (!triggersWithStates) {\n addClass(element, NG_TRIGGER_CLASSNAME);\n addClass(element, NG_TRIGGER_CLASSNAME + '-' + triggerName);\n this._engine.statesByElement.set(element, triggersWithStates = new Map());\n }\n let fromState = triggersWithStates.get(triggerName);\n const toState = new StateValue(value, this.id);\n const isObj = value && value.hasOwnProperty('value');\n if (!isObj && fromState) {\n toState.absorbOptions(fromState.options);\n }\n triggersWithStates.set(triggerName, toState);\n if (!fromState) {\n fromState = DEFAULT_STATE_VALUE;\n }\n const isRemoval = toState.value === VOID_VALUE;\n // normally this isn't reached by here, however, if an object expression\n // is passed in then it may be a new object each time. Comparing the value\n // is important since that will stay the same despite there being a new object.\n // The removal arc here is special cased because the same element is triggered\n // twice in the event that it contains animations on the outer/inner portions\n // of the host container\n if (!isRemoval && fromState.value === toState.value) {\n // this means that despite the value not changing, some inner params\n // have changed which means that the animation final styles need to be applied\n if (!objEquals(fromState.params, toState.params)) {\n const errors = [];\n const fromStyles = trigger.matchStyles(fromState.value, fromState.params, errors);\n const toStyles = trigger.matchStyles(toState.value, toState.params, errors);\n if (errors.length) {\n this._engine.reportError(errors);\n }\n else {\n this._engine.afterFlush(() => {\n eraseStyles(element, fromStyles);\n setStyles(element, toStyles);\n });\n }\n }\n return;\n }\n const playersOnElement = getOrSetDefaultValue(this._engine.playersByElement, element, []);\n playersOnElement.forEach(player => {\n // only remove the player if it is queued on the EXACT same trigger/namespace\n // we only also deal with queued players here because if the animation has\n // started then we want to keep the player alive until the flush happens\n // (which is where the previousPlayers are passed into the new player)\n if (player.namespaceId == this.id && player.triggerName == triggerName && player.queued) {\n player.destroy();\n }\n });\n let transition = trigger.matchTransition(fromState.value, toState.value, element, toState.params);\n let isFallbackTransition = false;\n if (!transition) {\n if (!defaultToFallback)\n return;\n transition = trigger.fallbackTransition;\n isFallbackTransition = true;\n }\n this._engine.totalQueuedPlayers++;\n this._queue.push({ element, triggerName, transition, fromState, toState, player, isFallbackTransition });\n if (!isFallbackTransition) {\n addClass(element, QUEUED_CLASSNAME);\n player.onStart(() => {\n removeClass(element, QUEUED_CLASSNAME);\n });\n }\n player.onDone(() => {\n let index = this.players.indexOf(player);\n if (index >= 0) {\n this.players.splice(index, 1);\n }\n const players = this._engine.playersByElement.get(element);\n if (players) {\n let index = players.indexOf(player);\n if (index >= 0) {\n players.splice(index, 1);\n }\n }\n });\n this.players.push(player);\n playersOnElement.push(player);\n return player;\n }\n deregister(name) {\n this._triggers.delete(name);\n this._engine.statesByElement.forEach(stateMap => stateMap.delete(name));\n this._elementListeners.forEach((listeners, element) => {\n this._elementListeners.set(element, listeners.filter(entry => {\n return entry.name != name;\n }));\n });\n }\n clearElementCache(element) {\n this._engine.statesByElement.delete(element);\n this._elementListeners.delete(element);\n const elementPlayers = this._engine.playersByElement.get(element);\n if (elementPlayers) {\n elementPlayers.forEach(player => player.destroy());\n this._engine.playersByElement.delete(element);\n }\n }\n _signalRemovalForInnerTriggers(rootElement, context) {\n const elements = this._engine.driver.query(rootElement, NG_TRIGGER_SELECTOR, true);\n // emulate a leave animation for all inner nodes within this node.\n // If there are no animations found for any of the nodes then clear the cache\n // for the element.\n elements.forEach(elm => {\n // this means that an inner remove() operation has already kicked off\n // the animation on this element...\n if (elm[REMOVAL_FLAG])\n return;\n const namespaces = this._engine.fetchNamespacesByElement(elm);\n if (namespaces.size) {\n namespaces.forEach(ns => ns.triggerLeaveAnimation(elm, context, false, true));\n }\n else {\n this.clearElementCache(elm);\n }\n });\n // If the child elements were removed along with the parent, their animations might not\n // have completed. Clear all the elements from the cache so we don't end up with a memory leak.\n this._engine.afterFlushAnimationsDone(() => elements.forEach(elm => this.clearElementCache(elm)));\n }\n triggerLeaveAnimation(element, context, destroyAfterComplete, defaultToFallback) {\n const triggerStates = this._engine.statesByElement.get(element);\n const previousTriggersValues = new Map();\n if (triggerStates) {\n const players = [];\n triggerStates.forEach((state, triggerName) => {\n previousTriggersValues.set(triggerName, state.value);\n // this check is here in the event that an element is removed\n // twice (both on the host level and the component level)\n if (this._triggers.has(triggerName)) {\n const player = this.trigger(element, triggerName, VOID_VALUE, defaultToFallback);\n if (player) {\n players.push(player);\n }\n }\n });\n if (players.length) {\n this._engine.markElementAsRemoved(this.id, element, true, context, previousTriggersValues);\n if (destroyAfterComplete) {\n optimizeGroupPlayer(players).onDone(() => this._engine.processLeaveNode(element));\n }\n return true;\n }\n }\n return false;\n }\n prepareLeaveAnimationListeners(element) {\n const listeners = this._elementListeners.get(element);\n const elementStates = this._engine.statesByElement.get(element);\n // if this statement fails then it means that the element was picked up\n // by an earlier flush (or there are no listeners at all to track the leave).\n if (listeners && elementStates) {\n const visitedTriggers = new Set();\n listeners.forEach(listener => {\n const triggerName = listener.name;\n if (visitedTriggers.has(triggerName))\n return;\n visitedTriggers.add(triggerName);\n const trigger = this._triggers.get(triggerName);\n const transition = trigger.fallbackTransition;\n const fromState = elementStates.get(triggerName) || DEFAULT_STATE_VALUE;\n const toState = new StateValue(VOID_VALUE);\n const player = new TransitionAnimationPlayer(this.id, triggerName, element);\n this._engine.totalQueuedPlayers++;\n this._queue.push({\n element,\n triggerName,\n transition,\n fromState,\n toState,\n player,\n isFallbackTransition: true\n });\n });\n }\n }\n removeNode(element, context) {\n const engine = this._engine;\n if (element.childElementCount) {\n this._signalRemovalForInnerTriggers(element, context);\n }\n // this means that a * => VOID animation was detected and kicked off\n if (this.triggerLeaveAnimation(element, context, true))\n return;\n // find the player that is animating and make sure that the\n // removal is delayed until that player has completed\n let containsPotentialParentTransition = false;\n if (engine.totalAnimations) {\n const currentPlayers = engine.players.length ? engine.playersByQueriedElement.get(element) : [];\n // when this `if statement` does not continue forward it means that\n // a previous animation query has selected the current element and\n // is animating it. In this situation want to continue forwards and\n // allow the element to be queued up for animation later.\n if (currentPlayers && currentPlayers.length) {\n containsPotentialParentTransition = true;\n }\n else {\n let parent = element;\n while (parent = parent.parentNode) {\n const triggers = engine.statesByElement.get(parent);\n if (triggers) {\n containsPotentialParentTransition = true;\n break;\n }\n }\n }\n }\n // at this stage we know that the element will either get removed\n // during flush or will be picked up by a parent query. Either way\n // we need to fire the listeners for this element when it DOES get\n // removed (once the query parent animation is done or after flush)\n this.prepareLeaveAnimationListeners(element);\n // whether or not a parent has an animation we need to delay the deferral of the leave\n // operation until we have more information (which we do after flush() has been called)\n if (containsPotentialParentTransition) {\n engine.markElementAsRemoved(this.id, element, false, context);\n }\n else {\n const removalFlag = element[REMOVAL_FLAG];\n if (!removalFlag || removalFlag === NULL_REMOVAL_STATE) {\n // we do this after the flush has occurred such\n // that the callbacks can be fired\n engine.afterFlush(() => this.clearElementCache(element));\n engine.destroyInnerAnimations(element);\n engine._onRemovalComplete(element, context);\n }\n }\n }\n insertNode(element, parent) {\n addClass(element, this._hostClassName);\n }\n drainQueuedTransitions(microtaskId) {\n const instructions = [];\n this._queue.forEach(entry => {\n const player = entry.player;\n if (player.destroyed)\n return;\n const element = entry.element;\n const listeners = this._elementListeners.get(element);\n if (listeners) {\n listeners.forEach((listener) => {\n if (listener.name == entry.triggerName) {\n const baseEvent = makeAnimationEvent(element, entry.triggerName, entry.fromState.value, entry.toState.value);\n baseEvent['_data'] = microtaskId;\n listenOnPlayer(entry.player, listener.phase, baseEvent, listener.callback);\n }\n });\n }\n if (player.markedForDestroy) {\n this._engine.afterFlush(() => {\n // now we can destroy the element properly since the event listeners have\n // been bound to the player\n player.destroy();\n });\n }\n else {\n instructions.push(entry);\n }\n });\n this._queue = [];\n return instructions.sort((a, b) => {\n // if depCount == 0 them move to front\n // otherwise if a contains b then move back\n const d0 = a.transition.ast.depCount;\n const d1 = b.transition.ast.depCount;\n if (d0 == 0 || d1 == 0) {\n return d0 - d1;\n }\n return this._engine.driver.containsElement(a.element, b.element) ? 1 : -1;\n });\n }\n destroy(context) {\n this.players.forEach(p => p.destroy());\n this._signalRemovalForInnerTriggers(this.hostElement, context);\n }\n elementContainsData(element) {\n let containsData = false;\n if (this._elementListeners.has(element))\n containsData = true;\n containsData =\n (this._queue.find(entry => entry.element === element) ? true : false) || containsData;\n return containsData;\n }\n}\nclass TransitionAnimationEngine {\n constructor(bodyNode, driver, _normalizer) {\n this.bodyNode = bodyNode;\n this.driver = driver;\n this._normalizer = _normalizer;\n this.players = [];\n this.newHostElements = new Map();\n this.playersByElement = new Map();\n this.playersByQueriedElement = new Map();\n this.statesByElement = new Map();\n this.disabledNodes = new Set();\n this.totalAnimations = 0;\n this.totalQueuedPlayers = 0;\n this._namespaceLookup = {};\n this._namespaceList = [];\n this._flushFns = [];\n this._whenQuietFns = [];\n this.namespacesByHostElement = new Map();\n this.collectedEnterElements = [];\n this.collectedLeaveElements = [];\n // this method is designed to be overridden by the code that uses this engine\n this.onRemovalComplete = (element, context) => { };\n }\n /** @internal */\n _onRemovalComplete(element, context) {\n this.onRemovalComplete(element, context);\n }\n get queuedPlayers() {\n const players = [];\n this._namespaceList.forEach(ns => {\n ns.players.forEach(player => {\n if (player.queued) {\n players.push(player);\n }\n });\n });\n return players;\n }\n createNamespace(namespaceId, hostElement) {\n const ns = new AnimationTransitionNamespace(namespaceId, hostElement, this);\n if (this.bodyNode && this.driver.containsElement(this.bodyNode, hostElement)) {\n this._balanceNamespaceList(ns, hostElement);\n }\n else {\n // defer this later until flush during when the host element has\n // been inserted so that we know exactly where to place it in\n // the namespace list\n this.newHostElements.set(hostElement, ns);\n // given that this host element is a part of the animation code, it\n // may or may not be inserted by a parent node that is of an\n // animation renderer type. If this happens then we can still have\n // access to this item when we query for :enter nodes. If the parent\n // is a renderer then the set data-structure will normalize the entry\n this.collectEnterElement(hostElement);\n }\n return this._namespaceLookup[namespaceId] = ns;\n }\n _balanceNamespaceList(ns, hostElement) {\n const namespaceList = this._namespaceList;\n const namespacesByHostElement = this.namespacesByHostElement;\n const limit = namespaceList.length - 1;\n if (limit >= 0) {\n let found = false;\n // Find the closest ancestor with an existing namespace so we can then insert `ns` after it,\n // establishing a top-down ordering of namespaces in `this._namespaceList`.\n let ancestor = this.driver.getParentElement(hostElement);\n while (ancestor) {\n const ancestorNs = namespacesByHostElement.get(ancestor);\n if (ancestorNs) {\n // An animation namespace has been registered for this ancestor, so we insert `ns`\n // right after it to establish top-down ordering of animation namespaces.\n const index = namespaceList.indexOf(ancestorNs);\n namespaceList.splice(index + 1, 0, ns);\n found = true;\n break;\n }\n ancestor = this.driver.getParentElement(ancestor);\n }\n if (!found) {\n // No namespace exists that is an ancestor of `ns`, so `ns` is inserted at the front to\n // ensure that any existing descendants are ordered after `ns`, retaining the desired\n // top-down ordering.\n namespaceList.unshift(ns);\n }\n }\n else {\n namespaceList.push(ns);\n }\n namespacesByHostElement.set(hostElement, ns);\n return ns;\n }\n register(namespaceId, hostElement) {\n let ns = this._namespaceLookup[namespaceId];\n if (!ns) {\n ns = this.createNamespace(namespaceId, hostElement);\n }\n return ns;\n }\n registerTrigger(namespaceId, name, trigger) {\n let ns = this._namespaceLookup[namespaceId];\n if (ns && ns.register(name, trigger)) {\n this.totalAnimations++;\n }\n }\n destroy(namespaceId, context) {\n if (!namespaceId)\n return;\n const ns = this._fetchNamespace(namespaceId);\n this.afterFlush(() => {\n this.namespacesByHostElement.delete(ns.hostElement);\n delete this._namespaceLookup[namespaceId];\n const index = this._namespaceList.indexOf(ns);\n if (index >= 0) {\n this._namespaceList.splice(index, 1);\n }\n });\n this.afterFlushAnimationsDone(() => ns.destroy(context));\n }\n _fetchNamespace(id) {\n return this._namespaceLookup[id];\n }\n fetchNamespacesByElement(element) {\n // normally there should only be one namespace per element, however\n // if @triggers are placed on both the component element and then\n // its host element (within the component code) then there will be\n // two namespaces returned. We use a set here to simply deduplicate\n // the namespaces in case (for the reason described above) there are multiple triggers\n const namespaces = new Set();\n const elementStates = this.statesByElement.get(element);\n if (elementStates) {\n for (let stateValue of elementStates.values()) {\n if (stateValue.namespaceId) {\n const ns = this._fetchNamespace(stateValue.namespaceId);\n if (ns) {\n namespaces.add(ns);\n }\n }\n }\n }\n return namespaces;\n }\n trigger(namespaceId, element, name, value) {\n if (isElementNode(element)) {\n const ns = this._fetchNamespace(namespaceId);\n if (ns) {\n ns.trigger(element, name, value);\n return true;\n }\n }\n return false;\n }\n insertNode(namespaceId, element, parent, insertBefore) {\n if (!isElementNode(element))\n return;\n // special case for when an element is removed and reinserted (move operation)\n // when this occurs we do not want to use the element for deletion later\n const details = element[REMOVAL_FLAG];\n if (details && details.setForRemoval) {\n details.setForRemoval = false;\n details.setForMove = true;\n const index = this.collectedLeaveElements.indexOf(element);\n if (index >= 0) {\n this.collectedLeaveElements.splice(index, 1);\n }\n }\n // in the event that the namespaceId is blank then the caller\n // code does not contain any animation code in it, but it is\n // just being called so that the node is marked as being inserted\n if (namespaceId) {\n const ns = this._fetchNamespace(namespaceId);\n // This if-statement is a workaround for router issue #21947.\n // The router sometimes hits a race condition where while a route\n // is being instantiated a new navigation arrives, triggering leave\n // animation of DOM that has not been fully initialized, until this\n // is resolved, we need to handle the scenario when DOM is not in a\n // consistent state during the animation.\n if (ns) {\n ns.insertNode(element, parent);\n }\n }\n // only *directives and host elements are inserted before\n if (insertBefore) {\n this.collectEnterElement(element);\n }\n }\n collectEnterElement(element) {\n this.collectedEnterElements.push(element);\n }\n markElementAsDisabled(element, value) {\n if (value) {\n if (!this.disabledNodes.has(element)) {\n this.disabledNodes.add(element);\n addClass(element, DISABLED_CLASSNAME);\n }\n }\n else if (this.disabledNodes.has(element)) {\n this.disabledNodes.delete(element);\n removeClass(element, DISABLED_CLASSNAME);\n }\n }\n removeNode(namespaceId, element, isHostElement, context) {\n if (isElementNode(element)) {\n const ns = namespaceId ? this._fetchNamespace(namespaceId) : null;\n if (ns) {\n ns.removeNode(element, context);\n }\n else {\n this.markElementAsRemoved(namespaceId, element, false, context);\n }\n if (isHostElement) {\n const hostNS = this.namespacesByHostElement.get(element);\n if (hostNS && hostNS.id !== namespaceId) {\n hostNS.removeNode(element, context);\n }\n }\n }\n else {\n this._onRemovalComplete(element, context);\n }\n }\n markElementAsRemoved(namespaceId, element, hasAnimation, context, previousTriggersValues) {\n this.collectedLeaveElements.push(element);\n element[REMOVAL_FLAG] = {\n namespaceId,\n setForRemoval: context,\n hasAnimation,\n removedBeforeQueried: false,\n previousTriggersValues\n };\n }\n listen(namespaceId, element, name, phase, callback) {\n if (isElementNode(element)) {\n return this._fetchNamespace(namespaceId).listen(element, name, phase, callback);\n }\n return () => { };\n }\n _buildInstruction(entry, subTimelines, enterClassName, leaveClassName, skipBuildAst) {\n return entry.transition.build(this.driver, entry.element, entry.fromState.value, entry.toState.value, enterClassName, leaveClassName, entry.fromState.options, entry.toState.options, subTimelines, skipBuildAst);\n }\n destroyInnerAnimations(containerElement) {\n let elements = this.driver.query(containerElement, NG_TRIGGER_SELECTOR, true);\n elements.forEach(element => this.destroyActiveAnimationsForElement(element));\n if (this.playersByQueriedElement.size == 0)\n return;\n elements = this.driver.query(containerElement, NG_ANIMATING_SELECTOR, true);\n elements.forEach(element => this.finishActiveQueriedAnimationOnElement(element));\n }\n destroyActiveAnimationsForElement(element) {\n const players = this.playersByElement.get(element);\n if (players) {\n players.forEach(player => {\n // special case for when an element is set for destruction, but hasn't started.\n // in this situation we want to delay the destruction until the flush occurs\n // so that any event listeners attached to the player are triggered.\n if (player.queued) {\n player.markedForDestroy = true;\n }\n else {\n player.destroy();\n }\n });\n }\n }\n finishActiveQueriedAnimationOnElement(element) {\n const players = this.playersByQueriedElement.get(element);\n if (players) {\n players.forEach(player => player.finish());\n }\n }\n whenRenderingDone() {\n return new Promise(resolve => {\n if (this.players.length) {\n return optimizeGroupPlayer(this.players).onDone(() => resolve());\n }\n else {\n resolve();\n }\n });\n }\n processLeaveNode(element) {\n const details = element[REMOVAL_FLAG];\n if (details && details.setForRemoval) {\n // this will prevent it from removing it twice\n element[REMOVAL_FLAG] = NULL_REMOVAL_STATE;\n if (details.namespaceId) {\n this.destroyInnerAnimations(element);\n const ns = this._fetchNamespace(details.namespaceId);\n if (ns) {\n ns.clearElementCache(element);\n }\n }\n this._onRemovalComplete(element, details.setForRemoval);\n }\n if (element.classList?.contains(DISABLED_CLASSNAME)) {\n this.markElementAsDisabled(element, false);\n }\n this.driver.query(element, DISABLED_SELECTOR, true).forEach(node => {\n this.markElementAsDisabled(node, false);\n });\n }\n flush(microtaskId = -1) {\n let players = [];\n if (this.newHostElements.size) {\n this.newHostElements.forEach((ns, element) => this._balanceNamespaceList(ns, element));\n this.newHostElements.clear();\n }\n if (this.totalAnimations && this.collectedEnterElements.length) {\n for (let i = 0; i < this.collectedEnterElements.length; i++) {\n const elm = this.collectedEnterElements[i];\n addClass(elm, STAR_CLASSNAME);\n }\n }\n if (this._namespaceList.length &&\n (this.totalQueuedPlayers || this.collectedLeaveElements.length)) {\n const cleanupFns = [];\n try {\n players = this._flushAnimations(cleanupFns, microtaskId);\n }\n finally {\n for (let i = 0; i < cleanupFns.length; i++) {\n cleanupFns[i]();\n }\n }\n }\n else {\n for (let i = 0; i < this.collectedLeaveElements.length; i++) {\n const element = this.collectedLeaveElements[i];\n this.processLeaveNode(element);\n }\n }\n this.totalQueuedPlayers = 0;\n this.collectedEnterElements.length = 0;\n this.collectedLeaveElements.length = 0;\n this._flushFns.forEach(fn => fn());\n this._flushFns = [];\n if (this._whenQuietFns.length) {\n // we move these over to a variable so that\n // if any new callbacks are registered in another\n // flush they do not populate the existing set\n const quietFns = this._whenQuietFns;\n this._whenQuietFns = [];\n if (players.length) {\n optimizeGroupPlayer(players).onDone(() => {\n quietFns.forEach(fn => fn());\n });\n }\n else {\n quietFns.forEach(fn => fn());\n }\n }\n }\n reportError(errors) {\n throw triggerTransitionsFailed(errors);\n }\n _flushAnimations(cleanupFns, microtaskId) {\n const subTimelines = new ElementInstructionMap();\n const skippedPlayers = [];\n const skippedPlayersMap = new Map();\n const queuedInstructions = [];\n const queriedElements = new Map();\n const allPreStyleElements = new Map();\n const allPostStyleElements = new Map();\n const disabledElementsSet = new Set();\n this.disabledNodes.forEach(node => {\n disabledElementsSet.add(node);\n const nodesThatAreDisabled = this.driver.query(node, QUEUED_SELECTOR, true);\n for (let i = 0; i < nodesThatAreDisabled.length; i++) {\n disabledElementsSet.add(nodesThatAreDisabled[i]);\n }\n });\n const bodyNode = this.bodyNode;\n const allTriggerElements = Array.from(this.statesByElement.keys());\n const enterNodeMap = buildRootMap(allTriggerElements, this.collectedEnterElements);\n // this must occur before the instructions are built below such that\n // the :enter queries match the elements (since the timeline queries\n // are fired during instruction building).\n const enterNodeMapIds = new Map();\n let i = 0;\n enterNodeMap.forEach((nodes, root) => {\n const className = ENTER_CLASSNAME + i++;\n enterNodeMapIds.set(root, className);\n nodes.forEach(node => addClass(node, className));\n });\n const allLeaveNodes = [];\n const mergedLeaveNodes = new Set();\n const leaveNodesWithoutAnimations = new Set();\n for (let i = 0; i < this.collectedLeaveElements.length; i++) {\n const element = this.collectedLeaveElements[i];\n const details = element[REMOVAL_FLAG];\n if (details && details.setForRemoval) {\n allLeaveNodes.push(element);\n mergedLeaveNodes.add(element);\n if (details.hasAnimation) {\n this.driver.query(element, STAR_SELECTOR, true).forEach(elm => mergedLeaveNodes.add(elm));\n }\n else {\n leaveNodesWithoutAnimations.add(element);\n }\n }\n }\n const leaveNodeMapIds = new Map();\n const leaveNodeMap = buildRootMap(allTriggerElements, Array.from(mergedLeaveNodes));\n leaveNodeMap.forEach((nodes, root) => {\n const className = LEAVE_CLASSNAME + i++;\n leaveNodeMapIds.set(root, className);\n nodes.forEach(node => addClass(node, className));\n });\n cleanupFns.push(() => {\n enterNodeMap.forEach((nodes, root) => {\n const className = enterNodeMapIds.get(root);\n nodes.forEach(node => removeClass(node, className));\n });\n leaveNodeMap.forEach((nodes, root) => {\n const className = leaveNodeMapIds.get(root);\n nodes.forEach(node => removeClass(node, className));\n });\n allLeaveNodes.forEach(element => {\n this.processLeaveNode(element);\n });\n });\n const allPlayers = [];\n const erroneousTransitions = [];\n for (let i = this._namespaceList.length - 1; i >= 0; i--) {\n const ns = this._namespaceList[i];\n ns.drainQueuedTransitions(microtaskId).forEach(entry => {\n const player = entry.player;\n const element = entry.element;\n allPlayers.push(player);\n if (this.collectedEnterElements.length) {\n const details = element[REMOVAL_FLAG];\n // animations for move operations (elements being removed and reinserted,\n // e.g. when the order of an *ngFor list changes) are currently not supported\n if (details && details.setForMove) {\n if (details.previousTriggersValues &&\n details.previousTriggersValues.has(entry.triggerName)) {\n const previousValue = details.previousTriggersValues.get(entry.triggerName);\n // we need to restore the previous trigger value since the element has\n // only been moved and hasn't actually left the DOM\n const triggersWithStates = this.statesByElement.get(entry.element);\n if (triggersWithStates && triggersWithStates.has(entry.triggerName)) {\n const state = triggersWithStates.get(entry.triggerName);\n state.value = previousValue;\n triggersWithStates.set(entry.triggerName, state);\n }\n }\n player.destroy();\n return;\n }\n }\n const nodeIsOrphaned = !bodyNode || !this.driver.containsElement(bodyNode, element);\n const leaveClassName = leaveNodeMapIds.get(element);\n const enterClassName = enterNodeMapIds.get(element);\n const instruction = this._buildInstruction(entry, subTimelines, enterClassName, leaveClassName, nodeIsOrphaned);\n if (instruction.errors && instruction.errors.length) {\n erroneousTransitions.push(instruction);\n return;\n }\n // even though the element may not be in the DOM, it may still\n // be added at a later point (due to the mechanics of content\n // projection and/or dynamic component insertion) therefore it's\n // important to still style the element.\n if (nodeIsOrphaned) {\n player.onStart(() => eraseStyles(element, instruction.fromStyles));\n player.onDestroy(() => setStyles(element, instruction.toStyles));\n skippedPlayers.push(player);\n return;\n }\n // if an unmatched transition is queued and ready to go\n // then it SHOULD NOT render an animation and cancel the\n // previously running animations.\n if (entry.isFallbackTransition) {\n player.onStart(() => eraseStyles(element, instruction.fromStyles));\n player.onDestroy(() => setStyles(element, instruction.toStyles));\n skippedPlayers.push(player);\n return;\n }\n // this means that if a parent animation uses this animation as a sub-trigger\n // then it will instruct the timeline builder not to add a player delay, but\n // instead stretch the first keyframe gap until the animation starts. This is\n // important in order to prevent extra initialization styles from being\n // required by the user for the animation.\n const timelines = [];\n instruction.timelines.forEach(tl => {\n tl.stretchStartingKeyframe = true;\n if (!this.disabledNodes.has(tl.element)) {\n timelines.push(tl);\n }\n });\n instruction.timelines = timelines;\n subTimelines.append(element, instruction.timelines);\n const tuple = { instruction, player, element };\n queuedInstructions.push(tuple);\n instruction.queriedElements.forEach(element => getOrSetDefaultValue(queriedElements, element, []).push(player));\n instruction.preStyleProps.forEach((stringMap, element) => {\n if (stringMap.size) {\n let setVal = allPreStyleElements.get(element);\n if (!setVal) {\n allPreStyleElements.set(element, setVal = new Set());\n }\n stringMap.forEach((_, prop) => setVal.add(prop));\n }\n });\n instruction.postStyleProps.forEach((stringMap, element) => {\n let setVal = allPostStyleElements.get(element);\n if (!setVal) {\n allPostStyleElements.set(element, setVal = new Set());\n }\n stringMap.forEach((_, prop) => setVal.add(prop));\n });\n });\n }\n if (erroneousTransitions.length) {\n const errors = [];\n erroneousTransitions.forEach(instruction => {\n errors.push(transitionFailed(instruction.triggerName, instruction.errors));\n });\n allPlayers.forEach(player => player.destroy());\n this.reportError(errors);\n }\n const allPreviousPlayersMap = new Map();\n // this map tells us which element in the DOM tree is contained by\n // which animation. Further down this map will get populated once\n // the players are built and in doing so we can use it to efficiently\n // figure out if a sub player is skipped due to a parent player having priority.\n const animationElementMap = new Map();\n queuedInstructions.forEach(entry => {\n const element = entry.element;\n if (subTimelines.has(element)) {\n animationElementMap.set(element, element);\n this._beforeAnimationBuild(entry.player.namespaceId, entry.instruction, allPreviousPlayersMap);\n }\n });\n skippedPlayers.forEach(player => {\n const element = player.element;\n const previousPlayers = this._getPreviousPlayers(element, false, player.namespaceId, player.triggerName, null);\n previousPlayers.forEach(prevPlayer => {\n getOrSetDefaultValue(allPreviousPlayersMap, element, []).push(prevPlayer);\n prevPlayer.destroy();\n });\n });\n // this is a special case for nodes that will be removed either by\n // having their own leave animations or by being queried in a container\n // that will be removed once a parent animation is complete. The idea\n // here is that * styles must be identical to ! styles because of\n // backwards compatibility (* is also filled in by default in many places).\n // Otherwise * styles will return an empty value or \"auto\" since the element\n // passed to getComputedStyle will not be visible (since * === destination)\n const replaceNodes = allLeaveNodes.filter(node => {\n return replacePostStylesAsPre(node, allPreStyleElements, allPostStyleElements);\n });\n // POST STAGE: fill the * styles\n const postStylesMap = new Map();\n const allLeaveQueriedNodes = cloakAndComputeStyles(postStylesMap, this.driver, leaveNodesWithoutAnimations, allPostStyleElements, AUTO_STYLE);\n allLeaveQueriedNodes.forEach(node => {\n if (replacePostStylesAsPre(node, allPreStyleElements, allPostStyleElements)) {\n replaceNodes.push(node);\n }\n });\n // PRE STAGE: fill the ! styles\n const preStylesMap = new Map();\n enterNodeMap.forEach((nodes, root) => {\n cloakAndComputeStyles(preStylesMap, this.driver, new Set(nodes), allPreStyleElements, ɵPRE_STYLE);\n });\n replaceNodes.forEach(node => {\n const post = postStylesMap.get(node);\n const pre = preStylesMap.get(node);\n postStylesMap.set(node, new Map([...Array.from(post?.entries() ?? []), ...Array.from(pre?.entries() ?? [])]));\n });\n const rootPlayers = [];\n const subPlayers = [];\n const NO_PARENT_ANIMATION_ELEMENT_DETECTED = {};\n queuedInstructions.forEach(entry => {\n const { element, player, instruction } = entry;\n // this means that it was never consumed by a parent animation which\n // means that it is independent and therefore should be set for animation\n if (subTimelines.has(element)) {\n if (disabledElementsSet.has(element)) {\n player.onDestroy(() => setStyles(element, instruction.toStyles));\n player.disabled = true;\n player.overrideTotalTime(instruction.totalTime);\n skippedPlayers.push(player);\n return;\n }\n // this will flow up the DOM and query the map to figure out\n // if a parent animation has priority over it. In the situation\n // that a parent is detected then it will cancel the loop. If\n // nothing is detected, or it takes a few hops to find a parent,\n // then it will fill in the missing nodes and signal them as having\n // a detected parent (or a NO_PARENT value via a special constant).\n let parentWithAnimation = NO_PARENT_ANIMATION_ELEMENT_DETECTED;\n if (animationElementMap.size > 1) {\n let elm = element;\n const parentsToAdd = [];\n while (elm = elm.parentNode) {\n const detectedParent = animationElementMap.get(elm);\n if (detectedParent) {\n parentWithAnimation = detectedParent;\n break;\n }\n parentsToAdd.push(elm);\n }\n parentsToAdd.forEach(parent => animationElementMap.set(parent, parentWithAnimation));\n }\n const innerPlayer = this._buildAnimation(player.namespaceId, instruction, allPreviousPlayersMap, skippedPlayersMap, preStylesMap, postStylesMap);\n player.setRealPlayer(innerPlayer);\n if (parentWithAnimation === NO_PARENT_ANIMATION_ELEMENT_DETECTED) {\n rootPlayers.push(player);\n }\n else {\n const parentPlayers = this.playersByElement.get(parentWithAnimation);\n if (parentPlayers && parentPlayers.length) {\n player.parentPlayer = optimizeGroupPlayer(parentPlayers);\n }\n skippedPlayers.push(player);\n }\n }\n else {\n eraseStyles(element, instruction.fromStyles);\n player.onDestroy(() => setStyles(element, instruction.toStyles));\n // there still might be a ancestor player animating this\n // element therefore we will still add it as a sub player\n // even if its animation may be disabled\n subPlayers.push(player);\n if (disabledElementsSet.has(element)) {\n skippedPlayers.push(player);\n }\n }\n });\n // find all of the sub players' corresponding inner animation players\n subPlayers.forEach(player => {\n // even if no players are found for a sub animation it\n // will still complete itself after the next tick since it's Noop\n const playersForElement = skippedPlayersMap.get(player.element);\n if (playersForElement && playersForElement.length) {\n const innerPlayer = optimizeGroupPlayer(playersForElement);\n player.setRealPlayer(innerPlayer);\n }\n });\n // the reason why we don't actually play the animation is\n // because all that a skipped player is designed to do is to\n // fire the start/done transition callback events\n skippedPlayers.forEach(player => {\n if (player.parentPlayer) {\n player.syncPlayerEvents(player.parentPlayer);\n }\n else {\n player.destroy();\n }\n });\n // run through all of the queued removals and see if they\n // were picked up by a query. If not then perform the removal\n // operation right away unless a parent animation is ongoing.\n for (let i = 0; i < allLeaveNodes.length; i++) {\n const element = allLeaveNodes[i];\n const details = element[REMOVAL_FLAG];\n removeClass(element, LEAVE_CLASSNAME);\n // this means the element has a removal animation that is being\n // taken care of and therefore the inner elements will hang around\n // until that animation is over (or the parent queried animation)\n if (details && details.hasAnimation)\n continue;\n let players = [];\n // if this element is queried or if it contains queried children\n // then we want for the element not to be removed from the page\n // until the queried animations have finished\n if (queriedElements.size) {\n let queriedPlayerResults = queriedElements.get(element);\n if (queriedPlayerResults && queriedPlayerResults.length) {\n players.push(...queriedPlayerResults);\n }\n let queriedInnerElements = this.driver.query(element, NG_ANIMATING_SELECTOR, true);\n for (let j = 0; j < queriedInnerElements.length; j++) {\n let queriedPlayers = queriedElements.get(queriedInnerElements[j]);\n if (queriedPlayers && queriedPlayers.length) {\n players.push(...queriedPlayers);\n }\n }\n }\n const activePlayers = players.filter(p => !p.destroyed);\n if (activePlayers.length) {\n removeNodesAfterAnimationDone(this, element, activePlayers);\n }\n else {\n this.processLeaveNode(element);\n }\n }\n // this is required so the cleanup method doesn't remove them\n allLeaveNodes.length = 0;\n rootPlayers.forEach(player => {\n this.players.push(player);\n player.onDone(() => {\n player.destroy();\n const index = this.players.indexOf(player);\n this.players.splice(index, 1);\n });\n player.play();\n });\n return rootPlayers;\n }\n elementContainsData(namespaceId, element) {\n let containsData = false;\n const details = element[REMOVAL_FLAG];\n if (details && details.setForRemoval)\n containsData = true;\n if (this.playersByElement.has(element))\n containsData = true;\n if (this.playersByQueriedElement.has(element))\n containsData = true;\n if (this.statesByElement.has(element))\n containsData = true;\n return this._fetchNamespace(namespaceId).elementContainsData(element) || containsData;\n }\n afterFlush(callback) {\n this._flushFns.push(callback);\n }\n afterFlushAnimationsDone(callback) {\n this._whenQuietFns.push(callback);\n }\n _getPreviousPlayers(element, isQueriedElement, namespaceId, triggerName, toStateValue) {\n let players = [];\n if (isQueriedElement) {\n const queriedElementPlayers = this.playersByQueriedElement.get(element);\n if (queriedElementPlayers) {\n players = queriedElementPlayers;\n }\n }\n else {\n const elementPlayers = this.playersByElement.get(element);\n if (elementPlayers) {\n const isRemovalAnimation = !toStateValue || toStateValue == VOID_VALUE;\n elementPlayers.forEach(player => {\n if (player.queued)\n return;\n if (!isRemovalAnimation && player.triggerName != triggerName)\n return;\n players.push(player);\n });\n }\n }\n if (namespaceId || triggerName) {\n players = players.filter(player => {\n if (namespaceId && namespaceId != player.namespaceId)\n return false;\n if (triggerName && triggerName != player.triggerName)\n return false;\n return true;\n });\n }\n return players;\n }\n _beforeAnimationBuild(namespaceId, instruction, allPreviousPlayersMap) {\n const triggerName = instruction.triggerName;\n const rootElement = instruction.element;\n // when a removal animation occurs, ALL previous players are collected\n // and destroyed (even if they are outside of the current namespace)\n const targetNameSpaceId = instruction.isRemovalTransition ? undefined : namespaceId;\n const targetTriggerName = instruction.isRemovalTransition ? undefined : triggerName;\n for (const timelineInstruction of instruction.timelines) {\n const element = timelineInstruction.element;\n const isQueriedElement = element !== rootElement;\n const players = getOrSetDefaultValue(allPreviousPlayersMap, element, []);\n const previousPlayers = this._getPreviousPlayers(element, isQueriedElement, targetNameSpaceId, targetTriggerName, instruction.toState);\n previousPlayers.forEach(player => {\n const realPlayer = player.getRealPlayer();\n if (realPlayer.beforeDestroy) {\n realPlayer.beforeDestroy();\n }\n player.destroy();\n players.push(player);\n });\n }\n // this needs to be done so that the PRE/POST styles can be\n // computed properly without interfering with the previous animation\n eraseStyles(rootElement, instruction.fromStyles);\n }\n _buildAnimation(namespaceId, instruction, allPreviousPlayersMap, skippedPlayersMap, preStylesMap, postStylesMap) {\n const triggerName = instruction.triggerName;\n const rootElement = instruction.element;\n // we first run this so that the previous animation player\n // data can be passed into the successive animation players\n const allQueriedPlayers = [];\n const allConsumedElements = new Set();\n const allSubElements = new Set();\n const allNewPlayers = instruction.timelines.map(timelineInstruction => {\n const element = timelineInstruction.element;\n allConsumedElements.add(element);\n // FIXME (matsko): make sure to-be-removed animations are removed properly\n const details = element[REMOVAL_FLAG];\n if (details && details.removedBeforeQueried)\n return new NoopAnimationPlayer(timelineInstruction.duration, timelineInstruction.delay);\n const isQueriedElement = element !== rootElement;\n const previousPlayers = flattenGroupPlayers((allPreviousPlayersMap.get(element) || EMPTY_PLAYER_ARRAY)\n .map(p => p.getRealPlayer()))\n .filter(p => {\n // the `element` is not apart of the AnimationPlayer definition, but\n // Mock/WebAnimations\n // use the element within their implementation. This will be added in Angular5 to\n // AnimationPlayer\n const pp = p;\n return pp.element ? pp.element === element : false;\n });\n const preStyles = preStylesMap.get(element);\n const postStyles = postStylesMap.get(element);\n const keyframes = normalizeKeyframes$1(this.driver, this._normalizer, element, timelineInstruction.keyframes, preStyles, postStyles);\n const player = this._buildPlayer(timelineInstruction, keyframes, previousPlayers);\n // this means that this particular player belongs to a sub trigger. It is\n // important that we match this player up with the corresponding (@trigger.listener)\n if (timelineInstruction.subTimeline && skippedPlayersMap) {\n allSubElements.add(element);\n }\n if (isQueriedElement) {\n const wrappedPlayer = new TransitionAnimationPlayer(namespaceId, triggerName, element);\n wrappedPlayer.setRealPlayer(player);\n allQueriedPlayers.push(wrappedPlayer);\n }\n return player;\n });\n allQueriedPlayers.forEach(player => {\n getOrSetDefaultValue(this.playersByQueriedElement, player.element, []).push(player);\n player.onDone(() => deleteOrUnsetInMap(this.playersByQueriedElement, player.element, player));\n });\n allConsumedElements.forEach(element => addClass(element, NG_ANIMATING_CLASSNAME));\n const player = optimizeGroupPlayer(allNewPlayers);\n player.onDestroy(() => {\n allConsumedElements.forEach(element => removeClass(element, NG_ANIMATING_CLASSNAME));\n setStyles(rootElement, instruction.toStyles);\n });\n // this basically makes all of the callbacks for sub element animations\n // be dependent on the upper players for when they finish\n allSubElements.forEach(element => {\n getOrSetDefaultValue(skippedPlayersMap, element, []).push(player);\n });\n return player;\n }\n _buildPlayer(instruction, keyframes, previousPlayers) {\n if (keyframes.length > 0) {\n return this.driver.animate(instruction.element, keyframes, instruction.duration, instruction.delay, instruction.easing, previousPlayers);\n }\n // special case for when an empty transition|definition is provided\n // ... there is no point in rendering an empty animation\n return new NoopAnimationPlayer(instruction.duration, instruction.delay);\n }\n}\nclass TransitionAnimationPlayer {\n constructor(namespaceId, triggerName, element) {\n this.namespaceId = namespaceId;\n this.triggerName = triggerName;\n this.element = element;\n this._player = new NoopAnimationPlayer();\n this._containsRealPlayer = false;\n this._queuedCallbacks = new Map();\n this.destroyed = false;\n this.markedForDestroy = false;\n this.disabled = false;\n this.queued = true;\n this.totalTime = 0;\n }\n setRealPlayer(player) {\n if (this._containsRealPlayer)\n return;\n this._player = player;\n this._queuedCallbacks.forEach((callbacks, phase) => {\n callbacks.forEach(callback => listenOnPlayer(player, phase, undefined, callback));\n });\n this._queuedCallbacks.clear();\n this._containsRealPlayer = true;\n this.overrideTotalTime(player.totalTime);\n this.queued = false;\n }\n getRealPlayer() {\n return this._player;\n }\n overrideTotalTime(totalTime) {\n this.totalTime = totalTime;\n }\n syncPlayerEvents(player) {\n const p = this._player;\n if (p.triggerCallback) {\n player.onStart(() => p.triggerCallback('start'));\n }\n player.onDone(() => this.finish());\n player.onDestroy(() => this.destroy());\n }\n _queueEvent(name, callback) {\n getOrSetDefaultValue(this._queuedCallbacks, name, []).push(callback);\n }\n onDone(fn) {\n if (this.queued) {\n this._queueEvent('done', fn);\n }\n this._player.onDone(fn);\n }\n onStart(fn) {\n if (this.queued) {\n this._queueEvent('start', fn);\n }\n this._player.onStart(fn);\n }\n onDestroy(fn) {\n if (this.queued) {\n this._queueEvent('destroy', fn);\n }\n this._player.onDestroy(fn);\n }\n init() {\n this._player.init();\n }\n hasStarted() {\n return this.queued ? false : this._player.hasStarted();\n }\n play() {\n !this.queued && this._player.play();\n }\n pause() {\n !this.queued && this._player.pause();\n }\n restart() {\n !this.queued && this._player.restart();\n }\n finish() {\n this._player.finish();\n }\n destroy() {\n this.destroyed = true;\n this._player.destroy();\n }\n reset() {\n !this.queued && this._player.reset();\n }\n setPosition(p) {\n if (!this.queued) {\n this._player.setPosition(p);\n }\n }\n getPosition() {\n return this.queued ? 0 : this._player.getPosition();\n }\n /** @internal */\n triggerCallback(phaseName) {\n const p = this._player;\n if (p.triggerCallback) {\n p.triggerCallback(phaseName);\n }\n }\n}\nfunction deleteOrUnsetInMap(map, key, value) {\n let currentValues = map.get(key);\n if (currentValues) {\n if (currentValues.length) {\n const index = currentValues.indexOf(value);\n currentValues.splice(index, 1);\n }\n if (currentValues.length == 0) {\n map.delete(key);\n }\n }\n return currentValues;\n}\nfunction normalizeTriggerValue(value) {\n // we use `!= null` here because it's the most simple\n // way to test against a \"falsy\" value without mixing\n // in empty strings or a zero value. DO NOT OPTIMIZE.\n return value != null ? value : null;\n}\nfunction isElementNode(node) {\n return node && node['nodeType'] === 1;\n}\nfunction isTriggerEventValid(eventName) {\n return eventName == 'start' || eventName == 'done';\n}\nfunction cloakElement(element, value) {\n const oldValue = element.style.display;\n element.style.display = value != null ? value : 'none';\n return oldValue;\n}\nfunction cloakAndComputeStyles(valuesMap, driver, elements, elementPropsMap, defaultStyle) {\n const cloakVals = [];\n elements.forEach(element => cloakVals.push(cloakElement(element)));\n const failedElements = [];\n elementPropsMap.forEach((props, element) => {\n const styles = new Map();\n props.forEach(prop => {\n const value = driver.computeStyle(element, prop, defaultStyle);\n styles.set(prop, value);\n // there is no easy way to detect this because a sub element could be removed\n // by a parent animation element being detached.\n if (!value || value.length == 0) {\n element[REMOVAL_FLAG] = NULL_REMOVED_QUERIED_STATE;\n failedElements.push(element);\n }\n });\n valuesMap.set(element, styles);\n });\n // we use a index variable here since Set.forEach(a, i) does not return\n // an index value for the closure (but instead just the value)\n let i = 0;\n elements.forEach(element => cloakElement(element, cloakVals[i++]));\n return failedElements;\n}\n/*\nSince the Angular renderer code will return a collection of inserted\nnodes in all areas of a DOM tree, it's up to this algorithm to figure\nout which nodes are roots for each animation @trigger.\n\nBy placing each inserted node into a Set and traversing upwards, it\nis possible to find the @trigger elements and well any direct *star\ninsertion nodes, if a @trigger root is found then the enter element\nis placed into the Map[@trigger] spot.\n */\nfunction buildRootMap(roots, nodes) {\n const rootMap = new Map();\n roots.forEach(root => rootMap.set(root, []));\n if (nodes.length == 0)\n return rootMap;\n const NULL_NODE = 1;\n const nodeSet = new Set(nodes);\n const localRootMap = new Map();\n function getRoot(node) {\n if (!node)\n return NULL_NODE;\n let root = localRootMap.get(node);\n if (root)\n return root;\n const parent = node.parentNode;\n if (rootMap.has(parent)) { // ngIf inside @trigger\n root = parent;\n }\n else if (nodeSet.has(parent)) { // ngIf inside ngIf\n root = NULL_NODE;\n }\n else { // recurse upwards\n root = getRoot(parent);\n }\n localRootMap.set(node, root);\n return root;\n }\n nodes.forEach(node => {\n const root = getRoot(node);\n if (root !== NULL_NODE) {\n rootMap.get(root).push(node);\n }\n });\n return rootMap;\n}\nfunction addClass(element, className) {\n element.classList?.add(className);\n}\nfunction removeClass(element, className) {\n element.classList?.remove(className);\n}\nfunction removeNodesAfterAnimationDone(engine, element, players) {\n optimizeGroupPlayer(players).onDone(() => engine.processLeaveNode(element));\n}\nfunction flattenGroupPlayers(players) {\n const finalPlayers = [];\n _flattenGroupPlayersRecur(players, finalPlayers);\n return finalPlayers;\n}\nfunction _flattenGroupPlayersRecur(players, finalPlayers) {\n for (let i = 0; i < players.length; i++) {\n const player = players[i];\n if (player instanceof ɵAnimationGroupPlayer) {\n _flattenGroupPlayersRecur(player.players, finalPlayers);\n }\n else {\n finalPlayers.push(player);\n }\n }\n}\nfunction objEquals(a, b) {\n const k1 = Object.keys(a);\n const k2 = Object.keys(b);\n if (k1.length != k2.length)\n return false;\n for (let i = 0; i < k1.length; i++) {\n const prop = k1[i];\n if (!b.hasOwnProperty(prop) || a[prop] !== b[prop])\n return false;\n }\n return true;\n}\nfunction replacePostStylesAsPre(element, allPreStyleElements, allPostStyleElements) {\n const postEntry = allPostStyleElements.get(element);\n if (!postEntry)\n return false;\n let preEntry = allPreStyleElements.get(element);\n if (preEntry) {\n postEntry.forEach(data => preEntry.add(data));\n }\n else {\n allPreStyleElements.set(element, postEntry);\n }\n allPostStyleElements.delete(element);\n return true;\n}\n\nclass AnimationEngine {\n constructor(bodyNode, _driver, _normalizer) {\n this.bodyNode = bodyNode;\n this._driver = _driver;\n this._normalizer = _normalizer;\n this._triggerCache = {};\n // this method is designed to be overridden by the code that uses this engine\n this.onRemovalComplete = (element, context) => { };\n this._transitionEngine = new TransitionAnimationEngine(bodyNode, _driver, _normalizer);\n this._timelineEngine = new TimelineAnimationEngine(bodyNode, _driver, _normalizer);\n this._transitionEngine.onRemovalComplete = (element, context) => this.onRemovalComplete(element, context);\n }\n registerTrigger(componentId, namespaceId, hostElement, name, metadata) {\n const cacheKey = componentId + '-' + name;\n let trigger = this._triggerCache[cacheKey];\n if (!trigger) {\n const errors = [];\n const warnings = [];\n const ast = buildAnimationAst(this._driver, metadata, errors, warnings);\n if (errors.length) {\n throw triggerBuildFailed(name, errors);\n }\n if (warnings.length) {\n warnTriggerBuild(name, warnings);\n }\n trigger = buildTrigger(name, ast, this._normalizer);\n this._triggerCache[cacheKey] = trigger;\n }\n this._transitionEngine.registerTrigger(namespaceId, name, trigger);\n }\n register(namespaceId, hostElement) {\n this._transitionEngine.register(namespaceId, hostElement);\n }\n destroy(namespaceId, context) {\n this._transitionEngine.destroy(namespaceId, context);\n }\n onInsert(namespaceId, element, parent, insertBefore) {\n this._transitionEngine.insertNode(namespaceId, element, parent, insertBefore);\n }\n onRemove(namespaceId, element, context, isHostElement) {\n this._transitionEngine.removeNode(namespaceId, element, isHostElement || false, context);\n }\n disableAnimations(element, disable) {\n this._transitionEngine.markElementAsDisabled(element, disable);\n }\n process(namespaceId, element, property, value) {\n if (property.charAt(0) == '@') {\n const [id, action] = parseTimelineCommand(property);\n const args = value;\n this._timelineEngine.command(id, element, action, args);\n }\n else {\n this._transitionEngine.trigger(namespaceId, element, property, value);\n }\n }\n listen(namespaceId, element, eventName, eventPhase, callback) {\n // @@listen\n if (eventName.charAt(0) == '@') {\n const [id, action] = parseTimelineCommand(eventName);\n return this._timelineEngine.listen(id, element, action, callback);\n }\n return this._transitionEngine.listen(namespaceId, element, eventName, eventPhase, callback);\n }\n flush(microtaskId = -1) {\n this._transitionEngine.flush(microtaskId);\n }\n get players() {\n return this._transitionEngine.players\n .concat(this._timelineEngine.players);\n }\n whenRenderingDone() {\n return this._transitionEngine.whenRenderingDone();\n }\n}\n\n/**\n * Returns an instance of `SpecialCasedStyles` if and when any special (non animateable) styles are\n * detected.\n *\n * In CSS there exist properties that cannot be animated within a keyframe animation\n * (whether it be via CSS keyframes or web-animations) and the animation implementation\n * will ignore them. This function is designed to detect those special cased styles and\n * return a container that will be executed at the start and end of the animation.\n *\n * @returns an instance of `SpecialCasedStyles` if any special styles are detected otherwise `null`\n */\nfunction packageNonAnimatableStyles(element, styles) {\n let startStyles = null;\n let endStyles = null;\n if (Array.isArray(styles) && styles.length) {\n startStyles = filterNonAnimatableStyles(styles[0]);\n if (styles.length > 1) {\n endStyles = filterNonAnimatableStyles(styles[styles.length - 1]);\n }\n }\n else if (styles instanceof Map) {\n startStyles = filterNonAnimatableStyles(styles);\n }\n return (startStyles || endStyles) ? new SpecialCasedStyles(element, startStyles, endStyles) :\n null;\n}\n/**\n * Designed to be executed during a keyframe-based animation to apply any special-cased styles.\n *\n * When started (when the `start()` method is run) then the provided `startStyles`\n * will be applied. When finished (when the `finish()` method is called) the\n * `endStyles` will be applied as well any any starting styles. Finally when\n * `destroy()` is called then all styles will be removed.\n */\nclass SpecialCasedStyles {\n constructor(_element, _startStyles, _endStyles) {\n this._element = _element;\n this._startStyles = _startStyles;\n this._endStyles = _endStyles;\n this._state = 0 /* SpecialCasedStylesState.Pending */;\n let initialStyles = SpecialCasedStyles.initialStylesByElement.get(_element);\n if (!initialStyles) {\n SpecialCasedStyles.initialStylesByElement.set(_element, initialStyles = new Map());\n }\n this._initialStyles = initialStyles;\n }\n start() {\n if (this._state < 1 /* SpecialCasedStylesState.Started */) {\n if (this._startStyles) {\n setStyles(this._element, this._startStyles, this._initialStyles);\n }\n this._state = 1 /* SpecialCasedStylesState.Started */;\n }\n }\n finish() {\n this.start();\n if (this._state < 2 /* SpecialCasedStylesState.Finished */) {\n setStyles(this._element, this._initialStyles);\n if (this._endStyles) {\n setStyles(this._element, this._endStyles);\n this._endStyles = null;\n }\n this._state = 1 /* SpecialCasedStylesState.Started */;\n }\n }\n destroy() {\n this.finish();\n if (this._state < 3 /* SpecialCasedStylesState.Destroyed */) {\n SpecialCasedStyles.initialStylesByElement.delete(this._element);\n if (this._startStyles) {\n eraseStyles(this._element, this._startStyles);\n this._endStyles = null;\n }\n if (this._endStyles) {\n eraseStyles(this._element, this._endStyles);\n this._endStyles = null;\n }\n setStyles(this._element, this._initialStyles);\n this._state = 3 /* SpecialCasedStylesState.Destroyed */;\n }\n }\n}\nSpecialCasedStyles.initialStylesByElement = ( /* @__PURE__ */new WeakMap());\nfunction filterNonAnimatableStyles(styles) {\n let result = null;\n styles.forEach((val, prop) => {\n if (isNonAnimatableStyle(prop)) {\n result = result || new Map();\n result.set(prop, val);\n }\n });\n return result;\n}\nfunction isNonAnimatableStyle(prop) {\n return prop === 'display' || prop === 'position';\n}\n\nclass WebAnimationsPlayer {\n constructor(element, keyframes, options, _specialStyles) {\n this.element = element;\n this.keyframes = keyframes;\n this.options = options;\n this._specialStyles = _specialStyles;\n this._onDoneFns = [];\n this._onStartFns = [];\n this._onDestroyFns = [];\n this._initialized = false;\n this._finished = false;\n this._started = false;\n this._destroyed = false;\n // the following original fns are persistent copies of the _onStartFns and _onDoneFns\n // and are used to reset the fns to their original values upon reset()\n // (since the _onStartFns and _onDoneFns get deleted after they are called)\n this._originalOnDoneFns = [];\n this._originalOnStartFns = [];\n this.time = 0;\n this.parentPlayer = null;\n this.currentSnapshot = new Map();\n this._duration = options['duration'];\n this._delay = options['delay'] || 0;\n this.time = this._duration + this._delay;\n }\n _onFinish() {\n if (!this._finished) {\n this._finished = true;\n this._onDoneFns.forEach(fn => fn());\n this._onDoneFns = [];\n }\n }\n init() {\n this._buildPlayer();\n this._preparePlayerBeforeStart();\n }\n _buildPlayer() {\n if (this._initialized)\n return;\n this._initialized = true;\n const keyframes = this.keyframes;\n this.domPlayer =\n this._triggerWebAnimation(this.element, keyframes, this.options);\n this._finalKeyframe = keyframes.length ? keyframes[keyframes.length - 1] : new Map();\n this.domPlayer.addEventListener('finish', () => this._onFinish());\n }\n _preparePlayerBeforeStart() {\n // this is required so that the player doesn't start to animate right away\n if (this._delay) {\n this._resetDomPlayerState();\n }\n else {\n this.domPlayer.pause();\n }\n }\n _convertKeyframesToObject(keyframes) {\n const kfs = [];\n keyframes.forEach(frame => {\n kfs.push(Object.fromEntries(frame));\n });\n return kfs;\n }\n /** @internal */\n _triggerWebAnimation(element, keyframes, options) {\n // jscompiler doesn't seem to know animate is a native property because it's not fully\n // supported yet across common browsers (we polyfill it for Edge/Safari) [CL #143630929]\n return element['animate'](this._convertKeyframesToObject(keyframes), options);\n }\n onStart(fn) {\n this._originalOnStartFns.push(fn);\n this._onStartFns.push(fn);\n }\n onDone(fn) {\n this._originalOnDoneFns.push(fn);\n this._onDoneFns.push(fn);\n }\n onDestroy(fn) {\n this._onDestroyFns.push(fn);\n }\n play() {\n this._buildPlayer();\n if (!this.hasStarted()) {\n this._onStartFns.forEach(fn => fn());\n this._onStartFns = [];\n this._started = true;\n if (this._specialStyles) {\n this._specialStyles.start();\n }\n }\n this.domPlayer.play();\n }\n pause() {\n this.init();\n this.domPlayer.pause();\n }\n finish() {\n this.init();\n if (this._specialStyles) {\n this._specialStyles.finish();\n }\n this._onFinish();\n this.domPlayer.finish();\n }\n reset() {\n this._resetDomPlayerState();\n this._destroyed = false;\n this._finished = false;\n this._started = false;\n this._onStartFns = this._originalOnStartFns;\n this._onDoneFns = this._originalOnDoneFns;\n }\n _resetDomPlayerState() {\n if (this.domPlayer) {\n this.domPlayer.cancel();\n }\n }\n restart() {\n this.reset();\n this.play();\n }\n hasStarted() {\n return this._started;\n }\n destroy() {\n if (!this._destroyed) {\n this._destroyed = true;\n this._resetDomPlayerState();\n this._onFinish();\n if (this._specialStyles) {\n this._specialStyles.destroy();\n }\n this._onDestroyFns.forEach(fn => fn());\n this._onDestroyFns = [];\n }\n }\n setPosition(p) {\n if (this.domPlayer === undefined) {\n this.init();\n }\n this.domPlayer.currentTime = p * this.time;\n }\n getPosition() {\n return this.domPlayer.currentTime / this.time;\n }\n get totalTime() {\n return this._delay + this._duration;\n }\n beforeDestroy() {\n const styles = new Map();\n if (this.hasStarted()) {\n // note: this code is invoked only when the `play` function was called prior to this\n // (thus `hasStarted` returns true), this implies that the code that initializes\n // `_finalKeyframe` has also been executed and the non-null assertion can be safely used here\n const finalKeyframe = this._finalKeyframe;\n finalKeyframe.forEach((val, prop) => {\n if (prop !== 'offset') {\n styles.set(prop, this._finished ? val : computeStyle(this.element, prop));\n }\n });\n }\n this.currentSnapshot = styles;\n }\n /** @internal */\n triggerCallback(phaseName) {\n const methods = phaseName === 'start' ? this._onStartFns : this._onDoneFns;\n methods.forEach(fn => fn());\n methods.length = 0;\n }\n}\n\nclass WebAnimationsDriver {\n validateStyleProperty(prop) {\n // Perform actual validation in dev mode only, in prod mode this check is a noop.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n return validateStyleProperty(prop);\n }\n return true;\n }\n validateAnimatableStyleProperty(prop) {\n // Perform actual validation in dev mode only, in prod mode this check is a noop.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n const cssProp = camelCaseToDashCase(prop);\n return validateWebAnimatableStyleProperty(cssProp);\n }\n return true;\n }\n matchesElement(_element, _selector) {\n // This method is deprecated and no longer in use so we return false.\n return false;\n }\n containsElement(elm1, elm2) {\n return containsElement(elm1, elm2);\n }\n getParentElement(element) {\n return getParentElement(element);\n }\n query(element, selector, multi) {\n return invokeQuery(element, selector, multi);\n }\n computeStyle(element, prop, defaultValue) {\n return window.getComputedStyle(element)[prop];\n }\n animate(element, keyframes, duration, delay, easing, previousPlayers = []) {\n const fill = delay == 0 ? 'both' : 'forwards';\n const playerOptions = { duration, delay, fill };\n // we check for this to avoid having a null|undefined value be present\n // for the easing (which results in an error for certain browsers #9752)\n if (easing) {\n playerOptions['easing'] = easing;\n }\n const previousStyles = new Map();\n const previousWebAnimationPlayers = previousPlayers.filter(player => player instanceof WebAnimationsPlayer);\n if (allowPreviousPlayerStylesMerge(duration, delay)) {\n previousWebAnimationPlayers.forEach(player => {\n player.currentSnapshot.forEach((val, prop) => previousStyles.set(prop, val));\n });\n }\n let _keyframes = normalizeKeyframes(keyframes).map(styles => copyStyles(styles));\n _keyframes = balancePreviousStylesIntoKeyframes(element, _keyframes, previousStyles);\n const specialStyles = packageNonAnimatableStyles(element, _keyframes);\n return new WebAnimationsPlayer(element, _keyframes, playerOptions, specialStyles);\n }\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { AnimationDriver, Animation as ɵAnimation, AnimationEngine as ɵAnimationEngine, AnimationStyleNormalizer as ɵAnimationStyleNormalizer, NoopAnimationDriver as ɵNoopAnimationDriver, NoopAnimationStyleNormalizer as ɵNoopAnimationStyleNormalizer, WebAnimationsDriver as ɵWebAnimationsDriver, WebAnimationsPlayer as ɵWebAnimationsPlayer, WebAnimationsStyleNormalizer as ɵWebAnimationsStyleNormalizer, allowPreviousPlayerStylesMerge as ɵallowPreviousPlayerStylesMerge, containsElement as ɵcontainsElement, getParentElement as ɵgetParentElement, invokeQuery as ɵinvokeQuery, normalizeKeyframes as ɵnormalizeKeyframes, validateStyleProperty as ɵvalidateStyleProperty };\n","/**\n * @license Angular v14.3.0\n * (c) 2010-2022 Google LLC. https://angular.io/\n * License: MIT\n */\n\nimport * as i0 from '@angular/core';\nimport { ViewEncapsulation, Injectable, Inject, RendererFactory2, NgZone, ANIMATION_MODULE_TYPE, NgModule } from '@angular/core';\nexport { ANIMATION_MODULE_TYPE } from '@angular/core';\nimport { ɵDomRendererFactory2, BrowserModule } from '@angular/platform-browser';\nimport { AnimationBuilder, sequence, AnimationFactory } from '@angular/animations';\nimport * as i1 from '@angular/animations/browser';\nimport { ɵAnimationEngine, ɵWebAnimationsStyleNormalizer, ɵAnimationStyleNormalizer, AnimationDriver, ɵWebAnimationsDriver, ɵNoopAnimationDriver } from '@angular/animations/browser';\nimport { DOCUMENT } from '@angular/common';\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass BrowserAnimationBuilder extends AnimationBuilder {\n constructor(rootRenderer, doc) {\n super();\n this._nextAnimationId = 0;\n const typeData = { id: '0', encapsulation: ViewEncapsulation.None, styles: [], data: { animation: [] } };\n this._renderer = rootRenderer.createRenderer(doc.body, typeData);\n }\n build(animation) {\n const id = this._nextAnimationId.toString();\n this._nextAnimationId++;\n const entry = Array.isArray(animation) ? sequence(animation) : animation;\n issueAnimationCommand(this._renderer, null, id, 'register', [entry]);\n return new BrowserAnimationFactory(id, this._renderer);\n }\n}\nBrowserAnimationBuilder.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationBuilder, deps: [{ token: i0.RendererFactory2 }, { token: DOCUMENT }], target: i0.ɵɵFactoryTarget.Injectable });\nBrowserAnimationBuilder.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationBuilder });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationBuilder, decorators: [{\n type: Injectable\n }], ctorParameters: function () { return [{ type: i0.RendererFactory2 }, { type: undefined, decorators: [{\n type: Inject,\n args: [DOCUMENT]\n }] }]; } });\nclass BrowserAnimationFactory extends AnimationFactory {\n constructor(_id, _renderer) {\n super();\n this._id = _id;\n this._renderer = _renderer;\n }\n create(element, options) {\n return new RendererAnimationPlayer(this._id, element, options || {}, this._renderer);\n }\n}\nclass RendererAnimationPlayer {\n constructor(id, element, options, _renderer) {\n this.id = id;\n this.element = element;\n this._renderer = _renderer;\n this.parentPlayer = null;\n this._started = false;\n this.totalTime = 0;\n this._command('create', options);\n }\n _listen(eventName, callback) {\n return this._renderer.listen(this.element, `@@${this.id}:${eventName}`, callback);\n }\n _command(command, ...args) {\n return issueAnimationCommand(this._renderer, this.element, this.id, command, args);\n }\n onDone(fn) {\n this._listen('done', fn);\n }\n onStart(fn) {\n this._listen('start', fn);\n }\n onDestroy(fn) {\n this._listen('destroy', fn);\n }\n init() {\n this._command('init');\n }\n hasStarted() {\n return this._started;\n }\n play() {\n this._command('play');\n this._started = true;\n }\n pause() {\n this._command('pause');\n }\n restart() {\n this._command('restart');\n }\n finish() {\n this._command('finish');\n }\n destroy() {\n this._command('destroy');\n }\n reset() {\n this._command('reset');\n this._started = false;\n }\n setPosition(p) {\n this._command('setPosition', p);\n }\n getPosition() {\n return this._renderer.engine.players[+this.id]?.getPosition() ?? 0;\n }\n}\nfunction issueAnimationCommand(renderer, element, id, command, args) {\n return renderer.setProperty(element, `@@${id}:${command}`, args);\n}\n\nconst ANIMATION_PREFIX = '@';\nconst DISABLE_ANIMATIONS_FLAG = '@.disabled';\nclass AnimationRendererFactory {\n constructor(delegate, engine, _zone) {\n this.delegate = delegate;\n this.engine = engine;\n this._zone = _zone;\n this._currentId = 0;\n this._microtaskId = 1;\n this._animationCallbacksBuffer = [];\n this._rendererCache = new Map();\n this._cdRecurDepth = 0;\n this.promise = Promise.resolve(0);\n engine.onRemovalComplete = (element, delegate) => {\n // Note: if a component element has a leave animation, and a host leave animation,\n // the view engine will call `removeChild` for the parent\n // component renderer as well as for the child component renderer.\n // Therefore, we need to check if we already removed the element.\n const parentNode = delegate?.parentNode(element);\n if (parentNode) {\n delegate.removeChild(parentNode, element);\n }\n };\n }\n createRenderer(hostElement, type) {\n const EMPTY_NAMESPACE_ID = '';\n // cache the delegates to find out which cached delegate can\n // be used by which cached renderer\n const delegate = this.delegate.createRenderer(hostElement, type);\n if (!hostElement || !type || !type.data || !type.data['animation']) {\n let renderer = this._rendererCache.get(delegate);\n if (!renderer) {\n // Ensure that the renderer is removed from the cache on destroy\n // since it may contain references to detached DOM nodes.\n const onRendererDestroy = () => this._rendererCache.delete(delegate);\n renderer =\n new BaseAnimationRenderer(EMPTY_NAMESPACE_ID, delegate, this.engine, onRendererDestroy);\n // only cache this result when the base renderer is used\n this._rendererCache.set(delegate, renderer);\n }\n return renderer;\n }\n const componentId = type.id;\n const namespaceId = type.id + '-' + this._currentId;\n this._currentId++;\n this.engine.register(namespaceId, hostElement);\n const registerTrigger = (trigger) => {\n if (Array.isArray(trigger)) {\n trigger.forEach(registerTrigger);\n }\n else {\n this.engine.registerTrigger(componentId, namespaceId, hostElement, trigger.name, trigger);\n }\n };\n const animationTriggers = type.data['animation'];\n animationTriggers.forEach(registerTrigger);\n return new AnimationRenderer(this, namespaceId, delegate, this.engine);\n }\n begin() {\n this._cdRecurDepth++;\n if (this.delegate.begin) {\n this.delegate.begin();\n }\n }\n _scheduleCountTask() {\n // always use promise to schedule microtask instead of use Zone\n this.promise.then(() => {\n this._microtaskId++;\n });\n }\n /** @internal */\n scheduleListenerCallback(count, fn, data) {\n if (count >= 0 && count < this._microtaskId) {\n this._zone.run(() => fn(data));\n return;\n }\n if (this._animationCallbacksBuffer.length == 0) {\n Promise.resolve(null).then(() => {\n this._zone.run(() => {\n this._animationCallbacksBuffer.forEach(tuple => {\n const [fn, data] = tuple;\n fn(data);\n });\n this._animationCallbacksBuffer = [];\n });\n });\n }\n this._animationCallbacksBuffer.push([fn, data]);\n }\n end() {\n this._cdRecurDepth--;\n // this is to prevent animations from running twice when an inner\n // component does CD when a parent component instead has inserted it\n if (this._cdRecurDepth == 0) {\n this._zone.runOutsideAngular(() => {\n this._scheduleCountTask();\n this.engine.flush(this._microtaskId);\n });\n }\n if (this.delegate.end) {\n this.delegate.end();\n }\n }\n whenRenderingDone() {\n return this.engine.whenRenderingDone();\n }\n}\nAnimationRendererFactory.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: AnimationRendererFactory, deps: [{ token: i0.RendererFactory2 }, { token: i1.ɵAnimationEngine }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Injectable });\nAnimationRendererFactory.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: AnimationRendererFactory });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: AnimationRendererFactory, decorators: [{\n type: Injectable\n }], ctorParameters: function () { return [{ type: i0.RendererFactory2 }, { type: i1.ɵAnimationEngine }, { type: i0.NgZone }]; } });\nclass BaseAnimationRenderer {\n constructor(namespaceId, delegate, engine, _onDestroy) {\n this.namespaceId = namespaceId;\n this.delegate = delegate;\n this.engine = engine;\n this._onDestroy = _onDestroy;\n this.destroyNode = this.delegate.destroyNode ? (n) => delegate.destroyNode(n) : null;\n }\n get data() {\n return this.delegate.data;\n }\n destroy() {\n this.engine.destroy(this.namespaceId, this.delegate);\n this.delegate.destroy();\n this._onDestroy?.();\n }\n createElement(name, namespace) {\n return this.delegate.createElement(name, namespace);\n }\n createComment(value) {\n return this.delegate.createComment(value);\n }\n createText(value) {\n return this.delegate.createText(value);\n }\n appendChild(parent, newChild) {\n this.delegate.appendChild(parent, newChild);\n this.engine.onInsert(this.namespaceId, newChild, parent, false);\n }\n insertBefore(parent, newChild, refChild, isMove = true) {\n this.delegate.insertBefore(parent, newChild, refChild);\n // If `isMove` true than we should animate this insert.\n this.engine.onInsert(this.namespaceId, newChild, parent, isMove);\n }\n removeChild(parent, oldChild, isHostElement) {\n this.engine.onRemove(this.namespaceId, oldChild, this.delegate, isHostElement);\n }\n selectRootElement(selectorOrNode, preserveContent) {\n return this.delegate.selectRootElement(selectorOrNode, preserveContent);\n }\n parentNode(node) {\n return this.delegate.parentNode(node);\n }\n nextSibling(node) {\n return this.delegate.nextSibling(node);\n }\n setAttribute(el, name, value, namespace) {\n this.delegate.setAttribute(el, name, value, namespace);\n }\n removeAttribute(el, name, namespace) {\n this.delegate.removeAttribute(el, name, namespace);\n }\n addClass(el, name) {\n this.delegate.addClass(el, name);\n }\n removeClass(el, name) {\n this.delegate.removeClass(el, name);\n }\n setStyle(el, style, value, flags) {\n this.delegate.setStyle(el, style, value, flags);\n }\n removeStyle(el, style, flags) {\n this.delegate.removeStyle(el, style, flags);\n }\n setProperty(el, name, value) {\n if (name.charAt(0) == ANIMATION_PREFIX && name == DISABLE_ANIMATIONS_FLAG) {\n this.disableAnimations(el, !!value);\n }\n else {\n this.delegate.setProperty(el, name, value);\n }\n }\n setValue(node, value) {\n this.delegate.setValue(node, value);\n }\n listen(target, eventName, callback) {\n return this.delegate.listen(target, eventName, callback);\n }\n disableAnimations(element, value) {\n this.engine.disableAnimations(element, value);\n }\n}\nclass AnimationRenderer extends BaseAnimationRenderer {\n constructor(factory, namespaceId, delegate, engine, onDestroy) {\n super(namespaceId, delegate, engine, onDestroy);\n this.factory = factory;\n this.namespaceId = namespaceId;\n }\n setProperty(el, name, value) {\n if (name.charAt(0) == ANIMATION_PREFIX) {\n if (name.charAt(1) == '.' && name == DISABLE_ANIMATIONS_FLAG) {\n value = value === undefined ? true : !!value;\n this.disableAnimations(el, value);\n }\n else {\n this.engine.process(this.namespaceId, el, name.slice(1), value);\n }\n }\n else {\n this.delegate.setProperty(el, name, value);\n }\n }\n listen(target, eventName, callback) {\n if (eventName.charAt(0) == ANIMATION_PREFIX) {\n const element = resolveElementFromTarget(target);\n let name = eventName.slice(1);\n let phase = '';\n // @listener.phase is for trigger animation callbacks\n // @@listener is for animation builder callbacks\n if (name.charAt(0) != ANIMATION_PREFIX) {\n [name, phase] = parseTriggerCallbackName(name);\n }\n return this.engine.listen(this.namespaceId, element, name, phase, event => {\n const countId = event['_data'] || -1;\n this.factory.scheduleListenerCallback(countId, callback, event);\n });\n }\n return this.delegate.listen(target, eventName, callback);\n }\n}\nfunction resolveElementFromTarget(target) {\n switch (target) {\n case 'body':\n return document.body;\n case 'document':\n return document;\n case 'window':\n return window;\n default:\n return target;\n }\n}\nfunction parseTriggerCallbackName(triggerName) {\n const dotIndex = triggerName.indexOf('.');\n const trigger = triggerName.substring(0, dotIndex);\n const phase = triggerName.slice(dotIndex + 1);\n return [trigger, phase];\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass InjectableAnimationEngine extends ɵAnimationEngine {\n // The `ApplicationRef` is injected here explicitly to force the dependency ordering.\n // Since the `ApplicationRef` should be created earlier before the `AnimationEngine`, they\n // both have `ngOnDestroy` hooks and `flush()` must be called after all views are destroyed.\n constructor(doc, driver, normalizer, appRef) {\n super(doc.body, driver, normalizer);\n }\n ngOnDestroy() {\n this.flush();\n }\n}\nInjectableAnimationEngine.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: InjectableAnimationEngine, deps: [{ token: DOCUMENT }, { token: i1.AnimationDriver }, { token: i1.ɵAnimationStyleNormalizer }, { token: i0.ApplicationRef }], target: i0.ɵɵFactoryTarget.Injectable });\nInjectableAnimationEngine.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: InjectableAnimationEngine });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: InjectableAnimationEngine, decorators: [{\n type: Injectable\n }], ctorParameters: function () { return [{ type: undefined, decorators: [{\n type: Inject,\n args: [DOCUMENT]\n }] }, { type: i1.AnimationDriver }, { type: i1.ɵAnimationStyleNormalizer }, { type: i0.ApplicationRef }]; } });\nfunction instantiateDefaultStyleNormalizer() {\n return new ɵWebAnimationsStyleNormalizer();\n}\nfunction instantiateRendererFactory(renderer, engine, zone) {\n return new AnimationRendererFactory(renderer, engine, zone);\n}\nconst SHARED_ANIMATION_PROVIDERS = [\n { provide: AnimationBuilder, useClass: BrowserAnimationBuilder },\n { provide: ɵAnimationStyleNormalizer, useFactory: instantiateDefaultStyleNormalizer },\n { provide: ɵAnimationEngine, useClass: InjectableAnimationEngine }, {\n provide: RendererFactory2,\n useFactory: instantiateRendererFactory,\n deps: [ɵDomRendererFactory2, ɵAnimationEngine, NgZone]\n }\n];\n/**\n * Separate providers from the actual module so that we can do a local modification in Google3 to\n * include them in the BrowserModule.\n */\nconst BROWSER_ANIMATIONS_PROVIDERS = [\n { provide: AnimationDriver, useFactory: () => new ɵWebAnimationsDriver() },\n { provide: ANIMATION_MODULE_TYPE, useValue: 'BrowserAnimations' }, ...SHARED_ANIMATION_PROVIDERS\n];\n/**\n * Separate providers from the actual module so that we can do a local modification in Google3 to\n * include them in the BrowserTestingModule.\n */\nconst BROWSER_NOOP_ANIMATIONS_PROVIDERS = [\n { provide: AnimationDriver, useClass: ɵNoopAnimationDriver },\n { provide: ANIMATION_MODULE_TYPE, useValue: 'NoopAnimations' }, ...SHARED_ANIMATION_PROVIDERS\n];\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * Exports `BrowserModule` with additional [dependency-injection providers](guide/glossary#provider)\n * for use with animations. See [Animations](guide/animations).\n * @publicApi\n */\nclass BrowserAnimationsModule {\n /**\n * Configures the module based on the specified object.\n *\n * @param config Object used to configure the behavior of the `BrowserAnimationsModule`.\n * @see `BrowserAnimationsModuleConfig`\n *\n * @usageNotes\n * When registering the `BrowserAnimationsModule`, you can use the `withConfig`\n * function as follows:\n * ```\n * @NgModule({\n * imports: [BrowserAnimationsModule.withConfig(config)]\n * })\n * class MyNgModule {}\n * ```\n */\n static withConfig(config) {\n return {\n ngModule: BrowserAnimationsModule,\n providers: config.disableAnimations ? BROWSER_NOOP_ANIMATIONS_PROVIDERS :\n BROWSER_ANIMATIONS_PROVIDERS\n };\n }\n}\nBrowserAnimationsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });\nBrowserAnimationsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: \"14.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationsModule, exports: [BrowserModule] });\nBrowserAnimationsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationsModule, providers: BROWSER_ANIMATIONS_PROVIDERS, imports: [BrowserModule] });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: BrowserAnimationsModule, decorators: [{\n type: NgModule,\n args: [{\n exports: [BrowserModule],\n providers: BROWSER_ANIMATIONS_PROVIDERS,\n }]\n }] });\n/**\n * Returns the set of [dependency-injection providers](guide/glossary#provider)\n * to enable animations in an application. See [animations guide](guide/animations)\n * to learn more about animations in Angular.\n *\n * @usageNotes\n *\n * The function is useful when you want to enable animations in an application\n * bootstrapped using the `bootstrapApplication` function. In this scenario there\n * is no need to import the `BrowserAnimationsModule` NgModule at all, just add\n * providers returned by this function to the `providers` list as show below.\n *\n * ```typescript\n * bootstrapApplication(RootComponent, {\n * providers: [\n * provideAnimations()\n * ]\n * });\n * ```\n *\n * @publicApi\n * @developerPreview\n */\nfunction provideAnimations() {\n // Return a copy to prevent changes to the original array in case any in-place\n // alterations are performed to the `provideAnimations` call results in app code.\n return [...BROWSER_ANIMATIONS_PROVIDERS];\n}\n/**\n * A null player that must be imported to allow disabling of animations.\n * @publicApi\n */\nclass NoopAnimationsModule {\n}\nNoopAnimationsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });\nNoopAnimationsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: \"14.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationsModule, exports: [BrowserModule] });\nNoopAnimationsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationsModule, providers: BROWSER_NOOP_ANIMATIONS_PROVIDERS, imports: [BrowserModule] });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.3.0\", ngImport: i0, type: NoopAnimationsModule, decorators: [{\n type: NgModule,\n args: [{\n exports: [BrowserModule],\n providers: BROWSER_NOOP_ANIMATIONS_PROVIDERS,\n }]\n }] });\n/**\n * Returns the set of [dependency-injection providers](guide/glossary#provider)\n * to disable animations in an application. See [animations guide](guide/animations)\n * to learn more about animations in Angular.\n *\n * @usageNotes\n *\n * The function is useful when you want to bootstrap an application using\n * the `bootstrapApplication` function, but you need to disable animations\n * (for example, when running tests).\n *\n * ```typescript\n * bootstrapApplication(RootComponent, {\n * providers: [\n * provideNoopAnimations()\n * ]\n * });\n * ```\n *\n * @publicApi\n * @developerPreview\n */\nfunction provideNoopAnimations() {\n // Return a copy to prevent changes to the original array in case any in-place\n // alterations are performed to the `provideNoopAnimations` call results in app code.\n return [...BROWSER_NOOP_ANIMATIONS_PROVIDERS];\n}\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { BrowserAnimationsModule, NoopAnimationsModule, provideAnimations, provideNoopAnimations, AnimationRenderer as ɵAnimationRenderer, AnimationRendererFactory as ɵAnimationRendererFactory, BrowserAnimationBuilder as ɵBrowserAnimationBuilder, BrowserAnimationFactory as ɵBrowserAnimationFactory, InjectableAnimationEngine as ɵInjectableAnimationEngine };\n","\n \n \n \n \n {{ 'pages.header.label_search' | translate }}\n {{ searchedTerm }}\n {{ 'pages.header.label_everywhere' | translate }}\n \n \n \n \n \n\n \n
  • \n \n \n\n
    \n
    \n {{ result.firstname + ' ' + result.lastname }}\n \n \n ({{ result.medias[0].code }}{{ result.medias.length === 1 ? '' : '...' }})\n \n
    \n
    \n \n \n
    \n
  • \n \n\n\n\n \n \n \n\n","import { Component, Inject, OnInit } from '@angular/core';\nimport { UntypedFormBuilder } from '@angular/forms';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { Router } from '@angular/router';\nimport { Customer } from '@app/modules/customers/models/customer';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { debounceTime, firstValueFrom } from 'rxjs';\n\nexport interface DialogData {}\n\n@Component({\n selector: 'tu-search-customer-modal',\n templateUrl: './search-customer-modal.component.html',\n})\nexport class SearchCustomerModalComponent implements OnInit {\n constructor(\n public router: Router,\n public modalRef: MatDialogRef,\n public customersService: CustomersService,\n private fb: UntypedFormBuilder,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n public searchForm = this.fb.group({\n search: [null],\n });\n\n public loading: boolean = false;\n public heroicons = heroicons;\n public displayedResult: Customer[] = [];\n public searchedTerm = '';\n public isLoading = false;\n\n public ngOnInit() {\n this.searchForm.controls.search.valueChanges\n .pipe(debounceTime(window['__CONFIG__']?.features?.globalSearchPreview ? 600 : 0))\n .subscribe(async (value) => {\n this.searchedTerm = value;\n if (value.length < 3) return;\n\n if (!window['__CONFIG__']?.features?.globalSearchPreview) {\n return;\n }\n\n this.isLoading = true;\n const customers = await firstValueFrom(this.customersService.searchCustomer(value));\n\n this.isLoading = false;\n return (this.displayedResult = customers.slice(0, 5));\n });\n }\n\n public async onSubmitSearchCustomers() {\n this.router.navigate(['/customers/search', { term: this.searchedTerm }]);\n\n this.closeModal();\n }\n\n public closeModal(): void {\n this.modalRef.close();\n }\n}\n","import { Injectable } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport { SubmissionService } from '../../submissions/services/submission.service';\nimport { OrdersService } from '@app/modules/orders/services/orders.service';\n\n@Injectable()\nexport class BadgeService {\n\n public orderToProcess = new Subject();\n\n constructor(private submissionService: SubmissionService, private orderService: OrdersService) {}\n\n /*private _badge: { [menu: string]: number } = {};\n\n\tset badge(total: number) { this._badge[window.location.pathname] = total; }\n\tget badge(): number { return this._badge[window.location.pathname]; }*/\n\n public getUnsubmittedDocument(): Observable {\n return this.submissionService.getUnsubmitted();\n }\n\n public async updateNbOrderToProcess() {\n const toProcess = await this.orderService.updateNbOrderToProcess();\n this.orderToProcess.next(toProcess);\n }\n}\n","import { HttpBackend, HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { AdminlabsComponent } from '@app/models/adminlabs';\nimport { Observable, map } from 'rxjs';\nimport { environment as env } from '@env/environment';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\n\n@Injectable()\nexport class AdminlabsService {\n private http: HttpClient;\n\n constructor(handle: HttpBackend, private authService: AuthService) {\n this.http = new HttpClient(handle);\n }\n\n public getComponentList(componentIdList: string[]): Observable {\n return this.http\n .get(`${env.config.adminlabsURL}/getComponents`, {\n headers: {\n authorization: `Bearer ${this.authService.token}`,\n },\n })\n .pipe(\n map((componentList) => {\n return componentList.filter(({ id }) => componentIdList.includes(id));\n })\n );\n }\n}\n","\n
  • \n {{ breadcrumb.data.title | translate }}\n {{ breadcrumb.data.title | translate }}\n
  • \n
    \n","import { Component } from '@angular/core';\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\nimport { filter } from 'rxjs/operators';\n\n@Component({\n selector: 'tu-breadcrumb',\n templateUrl: './breadcrumb.component.html',\n})\nexport class BreadcrumbComponent {\n // Will be used to create the breadcrumb\n public breadcrumbs: Array;\n\n constructor(private router: Router, private route: ActivatedRoute) {\n this.router.events.pipe(filter((event) => event instanceof NavigationEnd)).subscribe(() => {\n this.breadcrumbs = [];\n\n let currentRoute = this.route.root;\n let url = '';\n let routeStep = 0;\n\n do {\n const childrenRoutes = currentRoute.children;\n currentRoute = null;\n\n for (const childRoute of childrenRoutes) {\n if (childRoute.outlet !== 'primary') continue;\n\n currentRoute = childRoute;\n\n if (!childRoute.snapshot.data['title']) continue;\n\n routeStep++;\n\n // Compute breadcrumb step URL\n const urlExtension = '/' + this.router.url.split('/')[routeStep];\n url = !url ? urlExtension : url + urlExtension;\n\n // Add breadscrumb step data, i.e. step title and URL\n this.breadcrumbs.push({\n data: childRoute.snapshot.data,\n url,\n });\n }\n } while (currentRoute);\n });\n }\n}\n","\n
    \n

    {{ popup?.title }}

    \n

    {{ popup?.content }}

    \n \n
    \n \n
    \n
    \n\n","import { Component, ElementRef, Renderer2, ViewChild } from '@angular/core';\nimport { ModalService } from '../../services/modal.service';\n\n@Component({\n selector: 'tu-modal',\n templateUrl: './modal.component.html',\n styleUrls: ['./modal.component.scss'],\n})\nexport class ModalComponent {\n @ViewChild('answer', { static: true }) protected answersEl: ElementRef;\n public opened = false;\n\n public currValue: string;\n\n public popup: {\n title: string;\n content: string;\n input?: InputOptions;\n options: Option[];\n };\n\n constructor(private renderer: Renderer2, private modalService: ModalService) {\n this.modalService.registerComponent(this);\n }\n\n public async open(title: string, content: string, options: Option[]): Promise {\n this.popup = { title, content, options };\n this.opened = true;\n\n const p = new Promise((resolve, reject) => {\n this.renderer.listen(this.answersEl.nativeElement, 'click', (event) => {\n if (!event.target.value) return;\n const res = event.target.value;\n\n if (this.popup.options.find((o) => o.value === res)) resolve(res);\n else reject(new Error('Invalid choice'));\n\n this.opened = false;\n });\n });\n return await p;\n }\n\n public async prompt(\n title: string,\n content: string,\n input: InputOptions,\n options: Option[]\n ): Promise<{ choice: string; value: string }> {\n this.currValue = input.value || '';\n this.popup = { title, content, input, options };\n this.opened = true;\n\n const p = new Promise<{ choice: string; value: string }>((resolve, reject) => {\n this.renderer.listen(this.answersEl.nativeElement, 'click', (event) => {\n if (!event.target.value) return;\n const choice = event.target.value;\n\n if (this.popup.options.find((o) => o.value === choice))\n resolve({ choice, value: this.currValue });\n else reject(new Error('Invalid choice'));\n\n this.opened = false;\n });\n });\n return await p;\n }\n}\n\nexport interface Option {\n label: string;\n value: string;\n}\n\nexport interface InputOptions {\n type?: 'text' | 'time' | 'date' | 'email' | 'image';\n placeholder?: string;\n value?: string;\n src?: string;\n}\n","\n \n
    \n
    \n
    \n \n \n \n\n \n \n Status : {{ 'pages.list_menu.' + backOfficeState | lowercase | translate }}\n

    \n \n
    \n\n \n Menu\n \n \n
    \n\n
      \n
    • \n \n \n \n \n  {{ 'pages.header.search_client' | translate }}\n \n \n \n
    • \n \n \n \n  {{ 'pages.header.documents' | translate }}\n \n \n \n \n
    \n \n \n \n  \n {{ 'pages.header.documents' | translate }}\n \n \n \n  \n {{ 'pages.header.graphic_documents' | translate }}\n \n \n \n
  • \n \n \n  {{ userLogin }}\n \n \n \n \n \n \n \n  {{ 'pages.account_management.my_account' | translate }}\n \n \n \n  {{ 'pages.auth.logout' | translate }}\n \n \n
  • \n
  • \n \n \n \n \n \n \n \n \n \n \n \n
  • \n
  • \n \n {{ 'pages.changelog.title' | translate }}\n \n \n
  • \n \n \n \n\n \n \n\n \n \n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.home' | translate }}\n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.sales' | translate }}\n \n \n {{ orderToProcess }}\n \n \n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.subscriptions' | translate }}\n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.school_sales' | translate }}\n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.clients' | translate }}\n \n \n \n
  • \n\n \n \n \n \n\n \n  {{ 'pages.list_menu.usage_menu' | translate }} \n \n \n\n \n \n\n
    \n \n \n \n\n \n  {{ 'pages.list_menu.validations' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.titles' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.geolocations' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.controls' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.colors' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.booking' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.dash' | translate }}\n \n \n \n
    \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.legal_information_menu' | translate }} \n \n\n \n \n\n
    \n \n \n \n\n \n  {{ 'pages.list_menu.t_amp_cs' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.legal_information' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.privacy_policy' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.contact' | translate }}\n \n \n \n
    \n \n \n \n \n \n\n \n  {{ 'pages.list_menu.settings_menu' | translate }} \n \n \n \n \n
    \n \n \n \n\n \n  {{ 'pages.list_menu.payment' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.park' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.flyers' | translate }}\n \n \n \n {{ 'otherslabels.btn_new' | translate }}\n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.vehicles' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.trips' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.providers' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.graphical' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.users' | translate }}\n \n \n \n\n \n \n \n \n\n \n  {{ 'pages.agency.title' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.journey.title' | translate }}\n \n \n\n \n {{ 'otherslabels.btn_new' | translate }}\n \n \n\n \n \n \n\n \n  {{ 'pages.maintenance_mode_list.title' | translate }}\n \n \n \n
    \n \n\n \n \n \n \n\n \n  {{ 'pages.list_menu.networks_menu' | translate }} \n \n \n \n \n\n
    \n \n \n \n\n \n  {{ 'pages.list_menu.create_network' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.networks' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.gtfs_import' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.regions_applications' | translate }}\n \n \n \n
    \n \n\n \n \n \n \n\n \n  {{ 'pages.list_menu.shop_menu' | translate }} \n \n \n \n \n
    \n \n \n \n\n \n  {{ 'pages.list_menu.categories' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.v2_title' | translate }}\n
    \n  {{ 'pages.list_menu.blueprint' | translate }}\n
    \n
    \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.v1_title' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.products' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.scheduling' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.family' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.fields' | translate }}\n \n \n \n
    \n \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.submission' | translate }} \n \n \n \n {{ badgeSubmission < 10 ? badgeSubmission : '+10' }}\n \n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  {{ 'pages.list_menu.reports' | translate }}\n \n \n \n
  • \n\n
  • \n \n \n \n \n  {{ 'pages.list_menu.logs' | translate }}\n \n \n \n
  • \n\n \n \n \n \n\n \n  Tap & Protect \n \n \n \n \n\n
    \n \n \n \n\n \n  {{ 'pages.list_menu.tap_csv_imports' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.tap_csv_exports' | translate }}\n \n \n \n\n \n \n \n \n  {{ 'pages.list_menu.tap_transactions_details' | translate }}\n \n \n \n\n \n \n \n\n \n  {{ 'pages.list_menu.tap_dashboard' | translate }}\n \n \n \n
    \n \n\n \n \n \n \n\n \n  {{ 'pages.list_menu.validators' | translate }} \n \n \n \n \n\n \n \n \n \n\n \n  {{ 'pages.validators.machines' | translate }}\n \n \n \n \n\n \n \n \n \n\n \n  {{ 'pages.validators_validations.validator_events' | translate }}\n \n \n \n \n \n\n
  • \n \n \n \n\n \n  \n {{ 'pages.provisioning.provisioning' | translate }}\n \n \n \n
  • \n\n
  • \n \n \n \n\n \n  \n {{ 'Support' }}\n \n \n \n
  • \n \n \n\n
    \n
    \n \n
    \n\n \n \n \n
    \n\n \n v{{ appVersion }}\n \n AirwebPass © 2016 - {{ year }}\n Airweb\n \n \n\n \n\n","import { ChangeDetectorRef, Component, OnInit } from '@angular/core';\nimport { MatDialog, MatDialogRef, MatDialogState } from '@angular/material/dialog';\nimport { NavigationStart, Router } from '@angular/router';\nimport { AppComponent } from '@app/app.component';\nimport { Customer } from '@app/modules/customers/models/customer';\nimport { SearchCustomerModalComponent } from '@app/modules/shared/components/search-customer-modal/search-customer-modal.component';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { BadgeService } from '@app/modules/shared/services/badge.service';\nimport { ColorService } from '@app/modules/shared/services/color.service';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { environment as env } from '@env/environment';\nimport { NgbTypeaheadConfig } from '@ng-bootstrap/ng-bootstrap';\nimport { TranslateService } from '@ngx-translate/core';\nimport { filter } from 'rxjs';\nimport pkg from '../../../package.json';\nimport { AdminlabsService } from '@app/services/adminlabs.service';\nimport { AdminlabsComponentState } from '@app/models/adminlabs';\nimport { UpKeepService } from '@app/modules/shared/services/up-keep.service';\n\nconst MOBILE_MEDIA_QUERY = '(max-width: 767px)';\nconst BACK_OFFICE_STATE_KEY = 'back-office-state';\n\n@Component({\n selector: 'tu-app-layout',\n templateUrl: './app-layout.component.html',\n styleUrls: ['./app-layout.component.scss'],\n})\nexport class AppLayoutComponent implements OnInit {\n public openedDropdowns: Record = {};\n private _isSidebarToggled = true;\n public isMobile = false;\n public heroicons = heroicons;\n public appVersion = pkg.version + (env.production ? '' : '-dev');\n public airwebStatusUrl = env.config.airwebStatusUrl;\n public closeResult: string;\n public userLogin: string;\n public parent: AppComponent;\n public width = document.documentElement.clientWidth;\n public customers: Customer[];\n public langsTab = [\n // HACK: Hardcoded flags\n { locale: 'fr', label: 'FR', flag: 'fr' },\n { locale: 'en', label: 'EN', flag: 'gb' },\n { locale: 'es', label: 'ES', flag: 'es' },\n { locale: 'it', label: 'IT', flag: 'it' },\n ];\n public selectedLang = null;\n public flags = null;\n\n public year = new Date().getFullYear();\n\n public od: boolean;\n\n public access_reports = false;\n public badgeSubmission = null;\n private _badgeToProcess = 0;\n\n public backOfficeState: AdminlabsComponentState = 'operational';\n\n private customerModalRef: MatDialogRef | null = null;\n\n constructor(\n private authService: AuthService,\n public config: NgbTypeaheadConfig,\n public router: Router,\n public translate: TranslateService,\n public badgeService: BadgeService,\n public colorService: ColorService,\n public modal: MatDialog,\n private readonly changeDetector: ChangeDetectorRef,\n private adminlabsService: AdminlabsService,\n private upKeep: UpKeepService\n ) {\n const cachedLang = localStorage.getItem('userPreferedLang');\n translate.addLangs(this.langsTab.map((_) => _.locale));\n const browserLang = cachedLang || translate.getBrowserLang();\n const locale = browserLang.match(/en|fr/) ? browserLang : 'en';\n this.selectedLang = this.langsTab.find((l) => l.locale === locale);\n translate.setDefaultLang(locale);\n translate.use(locale);\n }\n\n get shouldShowGlobalSearch(): boolean {\n return window['__CONFIG__']?.features?.globalSearch ?? true;\n }\n\n get shouldShowDashboard(): boolean {\n return window['__CONFIG__']?.features?.dashboard ?? true;\n }\n\n public get sidebarStyle() {\n const style = this.colorService.sidebarColor;\n const background = `--sidebar-bg : ${style.primary}; `;\n const text = `--sidebar-text : ${style.text}; `;\n const selected = `--sidebar-bg-active : ${style.selected}; `;\n const hover = `--sidebar-bg-hover : ${style.hover}; `;\n const dropdown = `--sidebar-bg-dropdown : ${style.dropdown}; `;\n\n return background + text + selected + hover + dropdown;\n }\n\n public get hasShopRestricted() {\n //Maybe edit permission name EDIT_TEXT_PRODUCT to EDIT_RESTRICTIVE_TEXT_PRODUCT\n //Check if EDIT_ITEM_PRODUCT_RESTRICTED is not the same as EDIT_ITEMS_PRODUCT_RESTRICTIVE\n const restricted_permissions = [\n 'EDIT_ITEM_PRODUCT_RESTRICTED',\n 'EDIT_ITEM_PRODUCT_RESTRICTED_EDITOR',\n 'EDIT_ITEM_PRODUCT_RESTRICTED_MODIFICATOR',\n 'EDIT_ITEM_PRODUCT_RESTRICTED_CREATOR',\n ];\n for (const permission of restricted_permissions) {\n if (this.authService.hasPermission(permission)) {\n return true;\n }\n }\n return false;\n }\n\n public get isBlueprintAccess() {\n if (this.authService.hasPermission('MANAGE_PLATFORM_BLUEPRINT')) return true;\n if (this.authService.isSuperAdmin) return true;\n return this.hasShopRestricted;\n }\n\n public get lastBackOfficeState(): AdminlabsComponentState {\n const state = localStorage.getItem(BACK_OFFICE_STATE_KEY);\n\n if (['operational', 'degradedPerformance', 'partialOutage', 'majorOutage'].includes(state)) {\n return state as AdminlabsComponentState;\n }\n\n return 'operational';\n }\n\n get isSidebarToggled(): boolean {\n return this._isSidebarToggled;\n }\n\n set isSidebarToggled(toggle: boolean) {\n this._isSidebarToggled = toggle;\n\n const body = document.querySelector('body');\n\n if (toggle) {\n body.classList.remove('small-sidebar');\n } else {\n body.classList.add('small-sidebar');\n }\n }\n\n public get orderToProcess() {\n return this._badgeToProcess > 10 ? '+10' : this._badgeToProcess;\n }\n\n async ngOnInit() {\n this.verifyMaintenanceAlert();\n this.badgeService.getUnsubmittedDocument().subscribe((e) => (this.badgeSubmission = e));\n this.badgeService.orderToProcess.subscribe((toProcess) => {\n this._badgeToProcess = toProcess;\n });\n this.userLogin = this.authService.username;\n this.isMobile = window.matchMedia(MOBILE_MEDIA_QUERY).matches;\n this.isSidebarToggled = !this.isMobile;\n\n window.onresize = () => {\n this.isMobile = window.matchMedia(MOBILE_MEDIA_QUERY).matches;\n };\n\n document.addEventListener('keydown', (event) => {\n if ((event.ctrlKey || event.metaKey) && event.code === 'KeyB') {\n // Toggle sidebar\n this.toggleSidebar();\n }\n\n if (\n (event.ctrlKey || event.metaKey) &&\n event.code === 'KeyK' &&\n this.shouldShowGlobalSearch &&\n !this.router.isActive('/customers/search', true)\n ) {\n // Global customer search\n this.openSearchCustomerModal();\n }\n });\n\n this.authService.networks.forEach((n) => {\n if (n.payment_company === 'LEMONWAY') this.access_reports = true;\n });\n\n if (this.authService.networks.length === 1) {\n this.od = this.authService.networks[0].od;\n } else this.od = true;\n\n // We listen to route change and close the sidebar\n // if we navigate to a new page\n this.router.events.pipe(filter((event) => event instanceof NavigationStart)).subscribe(() => {\n this.verifyMaintenanceAlert();\n\n this.badgeService.updateNbOrderToProcess(); //Dirty => call for each url change. But need with multiple operator ?\n if (!this.isMobile) return;\n this.isSidebarToggled = false;\n });\n\n this.router.events.subscribe(() => {\n if (this.customerModalRef) {\n this.customerModalRef.close();\n }\n });\n\n this.adminlabsService\n .getComponentList(env.config.adminlabsBackOfficeComponentIds)\n .subscribe((componentList) => {\n const stateList = componentList.map(({ state }) => state);\n\n if (stateList.includes('majorOutage')) {\n this.backOfficeState = 'majorOutage';\n } else if (stateList.includes('partialOutage')) {\n this.backOfficeState = 'partialOutage';\n } else if (stateList.includes('degradedPerformance')) {\n this.backOfficeState = 'degradedPerformance';\n } else {\n this.backOfficeState = 'operational';\n }\n });\n this.badgeService.updateNbOrderToProcess();\n }\n\n public openSearchCustomerModal() {\n if (this.customerModalRef && this.customerModalRef.getState() === MatDialogState.OPEN) {\n this.customerModalRef.close();\n return;\n }\n\n // I don't even...\n // https://stackoverflow.com/a/58866045\n this.changeDetector.detach();\n\n this.customerModalRef = this.modal.open(SearchCustomerModalComponent, {\n width: '600px',\n position: { top: '10vh' },\n maxWidth: '100vw',\n maxHeight: '80vh',\n });\n\n // I don't even...\n this.customerModalRef.afterClosed().subscribe(() => {\n this.changeDetector.reattach();\n });\n }\n\n public setDefaultLang(lang) {\n this.translate.use(lang);\n this.selectedLang = this.langsTab.find((l) => l.locale === lang);\n localStorage.setItem('userPreferedLang', this.selectedLang.locale);\n }\n\n public storeBackofficeState(): void {\n localStorage.setItem(BACK_OFFICE_STATE_KEY, this.backOfficeState);\n }\n\n private async verifyMaintenanceAlert(): Promise {\n const alert = await this.upKeep.getMaintenanceAlert();\n\n if (alert) {\n this.router.navigate(['/maintenance']);\n }\n }\n\n public async logout(): Promise {\n this.openedDropdowns.account = false;\n\n await this.authService.logout();\n\n const newLoginUrl = new URL(env.config.newAuthenticationUrl);\n newLoginUrl.searchParams.set('redirect', location.href);\n\n location.href = newLoginUrl.toString();\n }\n\n public toggleSidebar(): void {\n this.isSidebarToggled = !this.isSidebarToggled;\n }\n}\n","import { FocusKeyManager } from '@angular/cdk/a11y';\nimport * as i1 from '@angular/cdk/bidi';\nimport { BidiModule } from '@angular/cdk/bidi';\nimport { coerceBooleanProperty, coerceNumberProperty } from '@angular/cdk/coercion';\nimport { hasModifierKey, SPACE, ENTER } from '@angular/cdk/keycodes';\nimport * as i0 from '@angular/core';\nimport { Directive, InjectionToken, EventEmitter, forwardRef, TemplateRef, Component, ViewEncapsulation, ChangeDetectionStrategy, Inject, Optional, ContentChild, ViewChild, Input, Output, QueryList, ContentChildren, NgModule } from '@angular/core';\nimport { _getFocusedElementPierceShadowDom } from '@angular/cdk/platform';\nimport { Subject, of } from 'rxjs';\nimport { startWith, takeUntil } from 'rxjs/operators';\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass CdkStepHeader {\n constructor(_elementRef) {\n this._elementRef = _elementRef;\n }\n /** Focuses the step header. */\n focus() {\n this._elementRef.nativeElement.focus();\n }\n}\nCdkStepHeader.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepHeader, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });\nCdkStepHeader.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStepHeader, selector: \"[cdkStepHeader]\", host: { attributes: { \"role\": \"tab\" } }, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepHeader, decorators: [{\n type: Directive,\n args: [{\n selector: '[cdkStepHeader]',\n host: {\n 'role': 'tab',\n },\n }]\n }], ctorParameters: function () { return [{ type: i0.ElementRef }]; } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass CdkStepLabel {\n constructor(/** @docs-private */ template) {\n this.template = template;\n }\n}\nCdkStepLabel.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepLabel, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });\nCdkStepLabel.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStepLabel, selector: \"[cdkStepLabel]\", ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepLabel, decorators: [{\n type: Directive,\n args: [{\n selector: '[cdkStepLabel]',\n }]\n }], ctorParameters: function () { return [{ type: i0.TemplateRef }]; } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/** Used to generate unique ID for each stepper component. */\nlet nextId = 0;\n/** Change event emitted on selection changes. */\nclass StepperSelectionEvent {\n}\n/** Enum to represent the different states of the steps. */\nconst STEP_STATE = {\n NUMBER: 'number',\n EDIT: 'edit',\n DONE: 'done',\n ERROR: 'error',\n};\n/** InjectionToken that can be used to specify the global stepper options. */\nconst STEPPER_GLOBAL_OPTIONS = new InjectionToken('STEPPER_GLOBAL_OPTIONS');\nclass CdkStep {\n constructor(_stepper, stepperOptions) {\n this._stepper = _stepper;\n /** Whether user has attempted to move away from the step. */\n this.interacted = false;\n /** Emits when the user has attempted to move away from the step. */\n this.interactedStream = new EventEmitter();\n this._editable = true;\n this._optional = false;\n this._completedOverride = null;\n this._customError = null;\n this._stepperOptions = stepperOptions ? stepperOptions : {};\n this._displayDefaultIndicatorType = this._stepperOptions.displayDefaultIndicatorType !== false;\n }\n /** Whether the user can return to this step once it has been marked as completed. */\n get editable() {\n return this._editable;\n }\n set editable(value) {\n this._editable = coerceBooleanProperty(value);\n }\n /** Whether the completion of step is optional. */\n get optional() {\n return this._optional;\n }\n set optional(value) {\n this._optional = coerceBooleanProperty(value);\n }\n /** Whether step is marked as completed. */\n get completed() {\n return this._completedOverride == null ? this._getDefaultCompleted() : this._completedOverride;\n }\n set completed(value) {\n this._completedOverride = coerceBooleanProperty(value);\n }\n _getDefaultCompleted() {\n return this.stepControl ? this.stepControl.valid && this.interacted : this.interacted;\n }\n /** Whether step has an error. */\n get hasError() {\n return this._customError == null ? this._getDefaultError() : this._customError;\n }\n set hasError(value) {\n this._customError = coerceBooleanProperty(value);\n }\n _getDefaultError() {\n return this.stepControl && this.stepControl.invalid && this.interacted;\n }\n /** Selects this step component. */\n select() {\n this._stepper.selected = this;\n }\n /** Resets the step to its initial state. Note that this includes resetting form data. */\n reset() {\n this.interacted = false;\n if (this._completedOverride != null) {\n this._completedOverride = false;\n }\n if (this._customError != null) {\n this._customError = false;\n }\n if (this.stepControl) {\n this.stepControl.reset();\n }\n }\n ngOnChanges() {\n // Since basically all inputs of the MatStep get proxied through the view down to the\n // underlying MatStepHeader, we have to make sure that change detection runs correctly.\n this._stepper._stateChanged();\n }\n _markAsInteracted() {\n if (!this.interacted) {\n this.interacted = true;\n this.interactedStream.emit(this);\n }\n }\n /** Determines whether the error state can be shown. */\n _showError() {\n // We want to show the error state either if the user opted into/out of it using the\n // global options, or if they've explicitly set it through the `hasError` input.\n return this._stepperOptions.showError ?? this._customError != null;\n }\n}\nCdkStep.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStep, deps: [{ token: forwardRef(() => CdkStepper) }, { token: STEPPER_GLOBAL_OPTIONS, optional: true }], target: i0.ɵɵFactoryTarget.Component });\nCdkStep.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStep, selector: \"cdk-step\", inputs: { stepControl: \"stepControl\", label: \"label\", errorMessage: \"errorMessage\", ariaLabel: [\"aria-label\", \"ariaLabel\"], ariaLabelledby: [\"aria-labelledby\", \"ariaLabelledby\"], state: \"state\", editable: \"editable\", optional: \"optional\", completed: \"completed\", hasError: \"hasError\" }, outputs: { interactedStream: \"interacted\" }, queries: [{ propertyName: \"stepLabel\", first: true, predicate: CdkStepLabel, descendants: true }], viewQueries: [{ propertyName: \"content\", first: true, predicate: TemplateRef, descendants: true, static: true }], exportAs: [\"cdkStep\"], usesOnChanges: true, ngImport: i0, template: '', isInline: true, changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStep, decorators: [{\n type: Component,\n args: [{\n selector: 'cdk-step',\n exportAs: 'cdkStep',\n template: '',\n encapsulation: ViewEncapsulation.None,\n changeDetection: ChangeDetectionStrategy.OnPush,\n }]\n }], ctorParameters: function () { return [{ type: CdkStepper, decorators: [{\n type: Inject,\n args: [forwardRef(() => CdkStepper)]\n }] }, { type: undefined, decorators: [{\n type: Optional\n }, {\n type: Inject,\n args: [STEPPER_GLOBAL_OPTIONS]\n }] }]; }, propDecorators: { stepLabel: [{\n type: ContentChild,\n args: [CdkStepLabel]\n }], content: [{\n type: ViewChild,\n args: [TemplateRef, { static: true }]\n }], stepControl: [{\n type: Input\n }], interactedStream: [{\n type: Output,\n args: ['interacted']\n }], label: [{\n type: Input\n }], errorMessage: [{\n type: Input\n }], ariaLabel: [{\n type: Input,\n args: ['aria-label']\n }], ariaLabelledby: [{\n type: Input,\n args: ['aria-labelledby']\n }], state: [{\n type: Input\n }], editable: [{\n type: Input\n }], optional: [{\n type: Input\n }], completed: [{\n type: Input\n }], hasError: [{\n type: Input\n }] } });\nclass CdkStepper {\n constructor(_dir, _changeDetectorRef, _elementRef) {\n this._dir = _dir;\n this._changeDetectorRef = _changeDetectorRef;\n this._elementRef = _elementRef;\n /** Emits when the component is destroyed. */\n this._destroyed = new Subject();\n /** Steps that belong to the current stepper, excluding ones from nested steppers. */\n this.steps = new QueryList();\n /** List of step headers sorted based on their DOM order. */\n this._sortedHeaders = new QueryList();\n this._linear = false;\n this._selectedIndex = 0;\n /** Event emitted when the selected step has changed. */\n this.selectionChange = new EventEmitter();\n this._orientation = 'horizontal';\n this._groupId = nextId++;\n }\n /** Whether the validity of previous steps should be checked or not. */\n get linear() {\n return this._linear;\n }\n set linear(value) {\n this._linear = coerceBooleanProperty(value);\n }\n /** The index of the selected step. */\n get selectedIndex() {\n return this._selectedIndex;\n }\n set selectedIndex(index) {\n const newIndex = coerceNumberProperty(index);\n if (this.steps && this._steps) {\n // Ensure that the index can't be out of bounds.\n if (!this._isValidIndex(newIndex) && (typeof ngDevMode === 'undefined' || ngDevMode)) {\n throw Error('cdkStepper: Cannot assign out-of-bounds value to `selectedIndex`.');\n }\n this.selected?._markAsInteracted();\n if (this._selectedIndex !== newIndex &&\n !this._anyControlsInvalidOrPending(newIndex) &&\n (newIndex >= this._selectedIndex || this.steps.toArray()[newIndex].editable)) {\n this._updateSelectedItemIndex(newIndex);\n }\n }\n else {\n this._selectedIndex = newIndex;\n }\n }\n /** The step that is selected. */\n get selected() {\n return this.steps ? this.steps.toArray()[this.selectedIndex] : undefined;\n }\n set selected(step) {\n this.selectedIndex = step && this.steps ? this.steps.toArray().indexOf(step) : -1;\n }\n /** Orientation of the stepper. */\n get orientation() {\n return this._orientation;\n }\n set orientation(value) {\n // This is a protected method so that `MatStepper` can hook into it.\n this._orientation = value;\n if (this._keyManager) {\n this._keyManager.withVerticalOrientation(value === 'vertical');\n }\n }\n ngAfterContentInit() {\n this._steps.changes\n .pipe(startWith(this._steps), takeUntil(this._destroyed))\n .subscribe((steps) => {\n this.steps.reset(steps.filter(step => step._stepper === this));\n this.steps.notifyOnChanges();\n });\n }\n ngAfterViewInit() {\n // If the step headers are defined outside of the `ngFor` that renders the steps, like in the\n // Material stepper, they won't appear in the `QueryList` in the same order as they're\n // rendered in the DOM which will lead to incorrect keyboard navigation. We need to sort\n // them manually to ensure that they're correct. Alternatively, we can change the Material\n // template to inline the headers in the `ngFor`, but that'll result in a lot of\n // code duplication. See #23539.\n this._stepHeader.changes\n .pipe(startWith(this._stepHeader), takeUntil(this._destroyed))\n .subscribe((headers) => {\n this._sortedHeaders.reset(headers.toArray().sort((a, b) => {\n const documentPosition = a._elementRef.nativeElement.compareDocumentPosition(b._elementRef.nativeElement);\n // `compareDocumentPosition` returns a bitmask so we have to use a bitwise operator.\n // https://developer.mozilla.org/en-US/docs/Web/API/Node/compareDocumentPosition\n // tslint:disable-next-line:no-bitwise\n return documentPosition & Node.DOCUMENT_POSITION_FOLLOWING ? -1 : 1;\n }));\n this._sortedHeaders.notifyOnChanges();\n });\n // Note that while the step headers are content children by default, any components that\n // extend this one might have them as view children. We initialize the keyboard handling in\n // AfterViewInit so we're guaranteed for both view and content children to be defined.\n this._keyManager = new FocusKeyManager(this._sortedHeaders)\n .withWrap()\n .withHomeAndEnd()\n .withVerticalOrientation(this._orientation === 'vertical');\n (this._dir ? this._dir.change : of())\n .pipe(startWith(this._layoutDirection()), takeUntil(this._destroyed))\n .subscribe(direction => this._keyManager.withHorizontalOrientation(direction));\n this._keyManager.updateActiveItem(this._selectedIndex);\n // No need to `takeUntil` here, because we're the ones destroying `steps`.\n this.steps.changes.subscribe(() => {\n if (!this.selected) {\n this._selectedIndex = Math.max(this._selectedIndex - 1, 0);\n }\n });\n // The logic which asserts that the selected index is within bounds doesn't run before the\n // steps are initialized, because we don't how many steps there are yet so we may have an\n // invalid index on init. If that's the case, auto-correct to the default so we don't throw.\n if (!this._isValidIndex(this._selectedIndex)) {\n this._selectedIndex = 0;\n }\n }\n ngOnDestroy() {\n this.steps.destroy();\n this._sortedHeaders.destroy();\n this._destroyed.next();\n this._destroyed.complete();\n }\n /** Selects and focuses the next step in list. */\n next() {\n this.selectedIndex = Math.min(this._selectedIndex + 1, this.steps.length - 1);\n }\n /** Selects and focuses the previous step in list. */\n previous() {\n this.selectedIndex = Math.max(this._selectedIndex - 1, 0);\n }\n /** Resets the stepper to its initial state. Note that this includes clearing form data. */\n reset() {\n this._updateSelectedItemIndex(0);\n this.steps.forEach(step => step.reset());\n this._stateChanged();\n }\n /** Returns a unique id for each step label element. */\n _getStepLabelId(i) {\n return `cdk-step-label-${this._groupId}-${i}`;\n }\n /** Returns unique id for each step content element. */\n _getStepContentId(i) {\n return `cdk-step-content-${this._groupId}-${i}`;\n }\n /** Marks the component to be change detected. */\n _stateChanged() {\n this._changeDetectorRef.markForCheck();\n }\n /** Returns position state of the step with the given index. */\n _getAnimationDirection(index) {\n const position = index - this._selectedIndex;\n if (position < 0) {\n return this._layoutDirection() === 'rtl' ? 'next' : 'previous';\n }\n else if (position > 0) {\n return this._layoutDirection() === 'rtl' ? 'previous' : 'next';\n }\n return 'current';\n }\n /** Returns the type of icon to be displayed. */\n _getIndicatorType(index, state = STEP_STATE.NUMBER) {\n const step = this.steps.toArray()[index];\n const isCurrentStep = this._isCurrentStep(index);\n return step._displayDefaultIndicatorType\n ? this._getDefaultIndicatorLogic(step, isCurrentStep)\n : this._getGuidelineLogic(step, isCurrentStep, state);\n }\n _getDefaultIndicatorLogic(step, isCurrentStep) {\n if (step._showError() && step.hasError && !isCurrentStep) {\n return STEP_STATE.ERROR;\n }\n else if (!step.completed || isCurrentStep) {\n return STEP_STATE.NUMBER;\n }\n else {\n return step.editable ? STEP_STATE.EDIT : STEP_STATE.DONE;\n }\n }\n _getGuidelineLogic(step, isCurrentStep, state = STEP_STATE.NUMBER) {\n if (step._showError() && step.hasError && !isCurrentStep) {\n return STEP_STATE.ERROR;\n }\n else if (step.completed && !isCurrentStep) {\n return STEP_STATE.DONE;\n }\n else if (step.completed && isCurrentStep) {\n return state;\n }\n else if (step.editable && isCurrentStep) {\n return STEP_STATE.EDIT;\n }\n else {\n return state;\n }\n }\n _isCurrentStep(index) {\n return this._selectedIndex === index;\n }\n /** Returns the index of the currently-focused step header. */\n _getFocusIndex() {\n return this._keyManager ? this._keyManager.activeItemIndex : this._selectedIndex;\n }\n _updateSelectedItemIndex(newIndex) {\n const stepsArray = this.steps.toArray();\n this.selectionChange.emit({\n selectedIndex: newIndex,\n previouslySelectedIndex: this._selectedIndex,\n selectedStep: stepsArray[newIndex],\n previouslySelectedStep: stepsArray[this._selectedIndex],\n });\n // If focus is inside the stepper, move it to the next header, otherwise it may become\n // lost when the active step content is hidden. We can't be more granular with the check\n // (e.g. checking whether focus is inside the active step), because we don't have a\n // reference to the elements that are rendering out the content.\n this._containsFocus()\n ? this._keyManager.setActiveItem(newIndex)\n : this._keyManager.updateActiveItem(newIndex);\n this._selectedIndex = newIndex;\n this._stateChanged();\n }\n _onKeydown(event) {\n const hasModifier = hasModifierKey(event);\n const keyCode = event.keyCode;\n const manager = this._keyManager;\n if (manager.activeItemIndex != null &&\n !hasModifier &&\n (keyCode === SPACE || keyCode === ENTER)) {\n this.selectedIndex = manager.activeItemIndex;\n event.preventDefault();\n }\n else {\n manager.onKeydown(event);\n }\n }\n _anyControlsInvalidOrPending(index) {\n if (this._linear && index >= 0) {\n return this.steps\n .toArray()\n .slice(0, index)\n .some(step => {\n const control = step.stepControl;\n const isIncomplete = control\n ? control.invalid || control.pending || !step.interacted\n : !step.completed;\n return isIncomplete && !step.optional && !step._completedOverride;\n });\n }\n return false;\n }\n _layoutDirection() {\n return this._dir && this._dir.value === 'rtl' ? 'rtl' : 'ltr';\n }\n /** Checks whether the stepper contains the focused element. */\n _containsFocus() {\n const stepperElement = this._elementRef.nativeElement;\n const focusedElement = _getFocusedElementPierceShadowDom();\n return stepperElement === focusedElement || stepperElement.contains(focusedElement);\n }\n /** Checks whether the passed-in index is a valid step index. */\n _isValidIndex(index) {\n return index > -1 && (!this.steps || index < this.steps.length);\n }\n}\nCdkStepper.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepper, deps: [{ token: i1.Directionality, optional: true }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });\nCdkStepper.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStepper, selector: \"[cdkStepper]\", inputs: { linear: \"linear\", selectedIndex: \"selectedIndex\", selected: \"selected\", orientation: \"orientation\" }, outputs: { selectionChange: \"selectionChange\" }, queries: [{ propertyName: \"_steps\", predicate: CdkStep, descendants: true }, { propertyName: \"_stepHeader\", predicate: CdkStepHeader, descendants: true }], exportAs: [\"cdkStepper\"], ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepper, decorators: [{\n type: Directive,\n args: [{\n selector: '[cdkStepper]',\n exportAs: 'cdkStepper',\n }]\n }], ctorParameters: function () { return [{ type: i1.Directionality, decorators: [{\n type: Optional\n }] }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { _steps: [{\n type: ContentChildren,\n args: [CdkStep, { descendants: true }]\n }], _stepHeader: [{\n type: ContentChildren,\n args: [CdkStepHeader, { descendants: true }]\n }], linear: [{\n type: Input\n }], selectedIndex: [{\n type: Input\n }], selected: [{\n type: Input\n }], selectionChange: [{\n type: Output\n }], orientation: [{\n type: Input\n }] } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/** Button that moves to the next step in a stepper workflow. */\nclass CdkStepperNext {\n constructor(_stepper) {\n this._stepper = _stepper;\n /** Type of the next button. Defaults to \"submit\" if not specified. */\n this.type = 'submit';\n }\n}\nCdkStepperNext.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperNext, deps: [{ token: CdkStepper }], target: i0.ɵɵFactoryTarget.Directive });\nCdkStepperNext.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStepperNext, selector: \"button[cdkStepperNext]\", inputs: { type: \"type\" }, host: { listeners: { \"click\": \"_stepper.next()\" }, properties: { \"type\": \"type\" } }, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperNext, decorators: [{\n type: Directive,\n args: [{\n selector: 'button[cdkStepperNext]',\n host: {\n '[type]': 'type',\n '(click)': '_stepper.next()',\n },\n }]\n }], ctorParameters: function () { return [{ type: CdkStepper }]; }, propDecorators: { type: [{\n type: Input\n }] } });\n/** Button that moves to the previous step in a stepper workflow. */\nclass CdkStepperPrevious {\n constructor(_stepper) {\n this._stepper = _stepper;\n /** Type of the previous button. Defaults to \"button\" if not specified. */\n this.type = 'button';\n }\n}\nCdkStepperPrevious.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperPrevious, deps: [{ token: CdkStepper }], target: i0.ɵɵFactoryTarget.Directive });\nCdkStepperPrevious.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: CdkStepperPrevious, selector: \"button[cdkStepperPrevious]\", inputs: { type: \"type\" }, host: { listeners: { \"click\": \"_stepper.previous()\" }, properties: { \"type\": \"type\" } }, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperPrevious, decorators: [{\n type: Directive,\n args: [{\n selector: 'button[cdkStepperPrevious]',\n host: {\n '[type]': 'type',\n '(click)': '_stepper.previous()',\n },\n }]\n }], ctorParameters: function () { return [{ type: CdkStepper }]; }, propDecorators: { type: [{\n type: Input\n }] } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass CdkStepperModule {\n}\nCdkStepperModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });\nCdkStepperModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: \"14.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperModule, declarations: [CdkStep,\n CdkStepper,\n CdkStepHeader,\n CdkStepLabel,\n CdkStepperNext,\n CdkStepperPrevious], imports: [BidiModule], exports: [CdkStep, CdkStepper, CdkStepHeader, CdkStepLabel, CdkStepperNext, CdkStepperPrevious] });\nCdkStepperModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperModule, imports: [BidiModule] });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: CdkStepperModule, decorators: [{\n type: NgModule,\n args: [{\n imports: [BidiModule],\n exports: [CdkStep, CdkStepper, CdkStepHeader, CdkStepLabel, CdkStepperNext, CdkStepperPrevious],\n declarations: [\n CdkStep,\n CdkStepper,\n CdkStepHeader,\n CdkStepLabel,\n CdkStepperNext,\n CdkStepperPrevious,\n ],\n }]\n }] });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { CdkStep, CdkStepHeader, CdkStepLabel, CdkStepper, CdkStepperModule, CdkStepperNext, CdkStepperPrevious, STEPPER_GLOBAL_OPTIONS, STEP_STATE, StepperSelectionEvent };\n","import * as i2$1 from '@angular/cdk/portal';\nimport { TemplatePortal, PortalModule } from '@angular/cdk/portal';\nimport { CdkStepLabel, CdkStepHeader, CdkStep, STEPPER_GLOBAL_OPTIONS, CdkStepper, CdkStepperNext, CdkStepperPrevious, CdkStepperModule } from '@angular/cdk/stepper';\nimport * as i3 from '@angular/common';\nimport { CommonModule } from '@angular/common';\nimport * as i0 from '@angular/core';\nimport { Directive, Injectable, Optional, SkipSelf, Component, ViewEncapsulation, ChangeDetectionStrategy, Input, forwardRef, Inject, ContentChild, QueryList, EventEmitter, ViewChildren, ContentChildren, Output, NgModule } from '@angular/core';\nimport { MatButtonModule } from '@angular/material/button';\nimport * as i1 from '@angular/material/core';\nimport { mixinColor, ErrorStateMatcher, MatCommonModule, MatRippleModule } from '@angular/material/core';\nimport * as i4 from '@angular/material/icon';\nimport { MatIconModule } from '@angular/material/icon';\nimport * as i2 from '@angular/cdk/a11y';\nimport { Subject, Subscription } from 'rxjs';\nimport * as i3$1 from '@angular/cdk/bidi';\nimport { switchMap, map, startWith, takeUntil, distinctUntilChanged } from 'rxjs/operators';\nimport { trigger, state, style, transition, animate } from '@angular/animations';\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass MatStepLabel extends CdkStepLabel {\n}\nMatStepLabel.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepLabel, deps: null, target: i0.ɵɵFactoryTarget.Directive });\nMatStepLabel.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepLabel, selector: \"[matStepLabel]\", usesInheritance: true, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepLabel, decorators: [{\n type: Directive,\n args: [{\n selector: '[matStepLabel]',\n }]\n }] });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/** Stepper data that is required for internationalization. */\nclass MatStepperIntl {\n constructor() {\n /**\n * Stream that emits whenever the labels here are changed. Use this to notify\n * components if the labels have changed after initialization.\n */\n this.changes = new Subject();\n /** Label that is rendered below optional steps. */\n this.optionalLabel = 'Optional';\n /** Label that is used to indicate step as completed to screen readers. */\n this.completedLabel = 'Completed';\n /** Label that is used to indicate step as editable to screen readers. */\n this.editableLabel = 'Editable';\n }\n}\nMatStepperIntl.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperIntl, deps: [], target: i0.ɵɵFactoryTarget.Injectable });\nMatStepperIntl.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperIntl, providedIn: 'root' });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperIntl, decorators: [{\n type: Injectable,\n args: [{ providedIn: 'root' }]\n }] });\n/** @docs-private */\nfunction MAT_STEPPER_INTL_PROVIDER_FACTORY(parentIntl) {\n return parentIntl || new MatStepperIntl();\n}\n/** @docs-private */\nconst MAT_STEPPER_INTL_PROVIDER = {\n provide: MatStepperIntl,\n deps: [[new Optional(), new SkipSelf(), MatStepperIntl]],\n useFactory: MAT_STEPPER_INTL_PROVIDER_FACTORY,\n};\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n// Boilerplate for applying mixins to MatStepHeader.\n/** @docs-private */\nconst _MatStepHeaderBase = mixinColor(class MatStepHeaderBase extends CdkStepHeader {\n constructor(elementRef) {\n super(elementRef);\n }\n}, 'primary');\nclass MatStepHeader extends _MatStepHeaderBase {\n constructor(_intl, _focusMonitor, _elementRef, changeDetectorRef) {\n super(_elementRef);\n this._intl = _intl;\n this._focusMonitor = _focusMonitor;\n this._intlSubscription = _intl.changes.subscribe(() => changeDetectorRef.markForCheck());\n }\n ngAfterViewInit() {\n this._focusMonitor.monitor(this._elementRef, true);\n }\n ngOnDestroy() {\n this._intlSubscription.unsubscribe();\n this._focusMonitor.stopMonitoring(this._elementRef);\n }\n /** Focuses the step header. */\n focus(origin, options) {\n if (origin) {\n this._focusMonitor.focusVia(this._elementRef, origin, options);\n }\n else {\n this._elementRef.nativeElement.focus(options);\n }\n }\n /** Returns string label of given step if it is a text label. */\n _stringLabel() {\n return this.label instanceof MatStepLabel ? null : this.label;\n }\n /** Returns MatStepLabel if the label of given step is a template label. */\n _templateLabel() {\n return this.label instanceof MatStepLabel ? this.label : null;\n }\n /** Returns the host HTML element. */\n _getHostElement() {\n return this._elementRef.nativeElement;\n }\n /** Template context variables that are exposed to the `matStepperIcon` instances. */\n _getIconContext() {\n return {\n index: this.index,\n active: this.active,\n optional: this.optional,\n };\n }\n _getDefaultTextForState(state) {\n if (state == 'number') {\n return `${this.index + 1}`;\n }\n if (state == 'edit') {\n return 'create';\n }\n if (state == 'error') {\n return 'warning';\n }\n return state;\n }\n}\nMatStepHeader.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepHeader, deps: [{ token: MatStepperIntl }, { token: i2.FocusMonitor }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });\nMatStepHeader.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepHeader, selector: \"mat-step-header\", inputs: { color: \"color\", state: \"state\", label: \"label\", errorMessage: \"errorMessage\", iconOverrides: \"iconOverrides\", index: \"index\", selected: \"selected\", active: \"active\", optional: \"optional\", disableRipple: \"disableRipple\" }, host: { attributes: { \"role\": \"tab\" }, classAttribute: \"mat-step-header\" }, usesInheritance: true, ngImport: i0, template: \"
    \\n\\n
    \\n
    \\n \\n \\n {{_getDefaultTextForState(state)}}\\n {{_intl.completedLabel}}\\n {{_intl.editableLabel}}\\n {{_getDefaultTextForState(state)}}\\n \\n
    \\n
    \\n
    \\n \\n
    \\n \\n
    \\n \\n
    {{label}}
    \\n\\n
    {{_intl.optionalLabel}}
    \\n
    {{errorMessage}}
    \\n
    \\n\\n\", styles: [\".mat-step-header{overflow:hidden;outline:none;cursor:pointer;position:relative;box-sizing:content-box;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mat-step-header:focus .mat-focus-indicator::before{content:\\\"\\\"}.cdk-high-contrast-active .mat-step-header{outline:solid 1px}.cdk-high-contrast-active .mat-step-header[aria-selected=true] .mat-step-label{text-decoration:underline}.cdk-high-contrast-active .mat-step-header[aria-disabled=true]{outline-color:GrayText}.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-label,.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-icon,.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-optional{color:GrayText}.mat-step-optional,.mat-step-sub-label-error{font-size:12px}.mat-step-icon{border-radius:50%;height:24px;width:24px;flex-shrink:0;position:relative}.mat-step-icon-content{position:absolute;top:50%;left:50%;transform:translate(-50%, -50%);display:flex}.mat-step-icon .mat-icon{font-size:16px;height:16px;width:16px}.mat-step-icon-state-error .mat-icon{font-size:24px;height:24px;width:24px}.mat-step-label{display:inline-block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;min-width:50px;vertical-align:middle}.mat-step-text-label{text-overflow:ellipsis;overflow:hidden}.mat-step-header .mat-step-header-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none}\"], dependencies: [{ kind: \"directive\", type: i3.NgIf, selector: \"[ngIf]\", inputs: [\"ngIf\", \"ngIfThen\", \"ngIfElse\"] }, { kind: \"directive\", type: i3.NgTemplateOutlet, selector: \"[ngTemplateOutlet]\", inputs: [\"ngTemplateOutletContext\", \"ngTemplateOutlet\", \"ngTemplateOutletInjector\"] }, { kind: \"directive\", type: i3.NgSwitch, selector: \"[ngSwitch]\", inputs: [\"ngSwitch\"] }, { kind: \"directive\", type: i3.NgSwitchCase, selector: \"[ngSwitchCase]\", inputs: [\"ngSwitchCase\"] }, { kind: \"directive\", type: i3.NgSwitchDefault, selector: \"[ngSwitchDefault]\" }, { kind: \"component\", type: i4.MatIcon, selector: \"mat-icon\", inputs: [\"color\", \"inline\", \"svgIcon\", \"fontSet\", \"fontIcon\"], exportAs: [\"matIcon\"] }, { kind: \"directive\", type: i1.MatRipple, selector: \"[mat-ripple], [matRipple]\", inputs: [\"matRippleColor\", \"matRippleUnbounded\", \"matRippleCentered\", \"matRippleRadius\", \"matRippleAnimation\", \"matRippleDisabled\", \"matRippleTrigger\"], exportAs: [\"matRipple\"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepHeader, decorators: [{\n type: Component,\n args: [{ selector: 'mat-step-header', inputs: ['color'], host: {\n 'class': 'mat-step-header',\n 'role': 'tab',\n }, encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: \"
    \\n\\n
    \\n
    \\n \\n \\n {{_getDefaultTextForState(state)}}\\n {{_intl.completedLabel}}\\n {{_intl.editableLabel}}\\n {{_getDefaultTextForState(state)}}\\n \\n
    \\n
    \\n
    \\n \\n
    \\n \\n
    \\n \\n
    {{label}}
    \\n\\n
    {{_intl.optionalLabel}}
    \\n
    {{errorMessage}}
    \\n
    \\n\\n\", styles: [\".mat-step-header{overflow:hidden;outline:none;cursor:pointer;position:relative;box-sizing:content-box;-webkit-tap-highlight-color:rgba(0,0,0,0)}.mat-step-header:focus .mat-focus-indicator::before{content:\\\"\\\"}.cdk-high-contrast-active .mat-step-header{outline:solid 1px}.cdk-high-contrast-active .mat-step-header[aria-selected=true] .mat-step-label{text-decoration:underline}.cdk-high-contrast-active .mat-step-header[aria-disabled=true]{outline-color:GrayText}.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-label,.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-icon,.cdk-high-contrast-active .mat-step-header[aria-disabled=true] .mat-step-optional{color:GrayText}.mat-step-optional,.mat-step-sub-label-error{font-size:12px}.mat-step-icon{border-radius:50%;height:24px;width:24px;flex-shrink:0;position:relative}.mat-step-icon-content{position:absolute;top:50%;left:50%;transform:translate(-50%, -50%);display:flex}.mat-step-icon .mat-icon{font-size:16px;height:16px;width:16px}.mat-step-icon-state-error .mat-icon{font-size:24px;height:24px;width:24px}.mat-step-label{display:inline-block;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;min-width:50px;vertical-align:middle}.mat-step-text-label{text-overflow:ellipsis;overflow:hidden}.mat-step-header .mat-step-header-ripple{top:0;left:0;right:0;bottom:0;position:absolute;pointer-events:none}\"] }]\n }], ctorParameters: function () { return [{ type: MatStepperIntl }, { type: i2.FocusMonitor }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { state: [{\n type: Input\n }], label: [{\n type: Input\n }], errorMessage: [{\n type: Input\n }], iconOverrides: [{\n type: Input\n }], index: [{\n type: Input\n }], selected: [{\n type: Input\n }], active: [{\n type: Input\n }], optional: [{\n type: Input\n }], disableRipple: [{\n type: Input\n }] } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nconst DEFAULT_HORIZONTAL_ANIMATION_DURATION = '500ms';\nconst DEFAULT_VERTICAL_ANIMATION_DURATION = '225ms';\n/**\n * Animations used by the Material steppers.\n * @docs-private\n */\nconst matStepperAnimations = {\n /** Animation that transitions the step along the X axis in a horizontal stepper. */\n horizontalStepTransition: trigger('horizontalStepTransition', [\n state('previous', style({ transform: 'translate3d(-100%, 0, 0)', visibility: 'hidden' })),\n // Transition to `inherit`, rather than `visible`,\n // because visibility on a child element the one from the parent,\n // making this element focusable inside of a `hidden` element.\n state('current', style({ transform: 'none', visibility: 'inherit' })),\n state('next', style({ transform: 'translate3d(100%, 0, 0)', visibility: 'hidden' })),\n transition('* => *', animate('{{animationDuration}} cubic-bezier(0.35, 0, 0.25, 1)'), {\n params: { 'animationDuration': DEFAULT_HORIZONTAL_ANIMATION_DURATION },\n }),\n ]),\n /** Animation that transitions the step along the Y axis in a vertical stepper. */\n verticalStepTransition: trigger('verticalStepTransition', [\n state('previous', style({ height: '0px', visibility: 'hidden' })),\n state('next', style({ height: '0px', visibility: 'hidden' })),\n // Transition to `inherit`, rather than `visible`,\n // because visibility on a child element the one from the parent,\n // making this element focusable inside of a `hidden` element.\n state('current', style({ height: '*', visibility: 'inherit' })),\n transition('* <=> current', animate('{{animationDuration}} cubic-bezier(0.4, 0.0, 0.2, 1)'), {\n params: { 'animationDuration': DEFAULT_VERTICAL_ANIMATION_DURATION },\n }),\n ]),\n};\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * Template to be used to override the icons inside the step header.\n */\nclass MatStepperIcon {\n constructor(templateRef) {\n this.templateRef = templateRef;\n }\n}\nMatStepperIcon.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperIcon, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });\nMatStepperIcon.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepperIcon, selector: \"ng-template[matStepperIcon]\", inputs: { name: [\"matStepperIcon\", \"name\"] }, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperIcon, decorators: [{\n type: Directive,\n args: [{\n selector: 'ng-template[matStepperIcon]',\n }]\n }], ctorParameters: function () { return [{ type: i0.TemplateRef }]; }, propDecorators: { name: [{\n type: Input,\n args: ['matStepperIcon']\n }] } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/**\n * Content for a `mat-step` that will be rendered lazily.\n */\nclass MatStepContent {\n constructor(_template) {\n this._template = _template;\n }\n}\nMatStepContent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepContent, deps: [{ token: i0.TemplateRef }], target: i0.ɵɵFactoryTarget.Directive });\nMatStepContent.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepContent, selector: \"ng-template[matStepContent]\", ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepContent, decorators: [{\n type: Directive,\n args: [{\n selector: 'ng-template[matStepContent]',\n }]\n }], ctorParameters: function () { return [{ type: i0.TemplateRef }]; } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass MatStep extends CdkStep {\n constructor(stepper, _errorStateMatcher, _viewContainerRef, stepperOptions) {\n super(stepper, stepperOptions);\n this._errorStateMatcher = _errorStateMatcher;\n this._viewContainerRef = _viewContainerRef;\n this._isSelected = Subscription.EMPTY;\n }\n ngAfterContentInit() {\n this._isSelected = this._stepper.steps.changes\n .pipe(switchMap(() => {\n return this._stepper.selectionChange.pipe(map(event => event.selectedStep === this), startWith(this._stepper.selected === this));\n }))\n .subscribe(isSelected => {\n if (isSelected && this._lazyContent && !this._portal) {\n this._portal = new TemplatePortal(this._lazyContent._template, this._viewContainerRef);\n }\n });\n }\n ngOnDestroy() {\n this._isSelected.unsubscribe();\n }\n /** Custom error state matcher that additionally checks for validity of interacted form. */\n isErrorState(control, form) {\n const originalErrorState = this._errorStateMatcher.isErrorState(control, form);\n // Custom error state checks for the validity of form that is not submitted or touched\n // since user can trigger a form change by calling for another step without directly\n // interacting with the current form.\n const customErrorState = !!(control && control.invalid && this.interacted);\n return originalErrorState || customErrorState;\n }\n}\nMatStep.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStep, deps: [{ token: forwardRef(() => MatStepper) }, { token: i1.ErrorStateMatcher, skipSelf: true }, { token: i0.ViewContainerRef }, { token: STEPPER_GLOBAL_OPTIONS, optional: true }], target: i0.ɵɵFactoryTarget.Component });\nMatStep.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStep, selector: \"mat-step\", inputs: { color: \"color\" }, providers: [\n { provide: ErrorStateMatcher, useExisting: MatStep },\n { provide: CdkStep, useExisting: MatStep },\n ], queries: [{ propertyName: \"stepLabel\", first: true, predicate: MatStepLabel, descendants: true }, { propertyName: \"_lazyContent\", first: true, predicate: MatStepContent, descendants: true }], exportAs: [\"matStep\"], usesInheritance: true, ngImport: i0, template: \"\\n \\n \\n\\n\", dependencies: [{ kind: \"directive\", type: i2$1.CdkPortalOutlet, selector: \"[cdkPortalOutlet]\", inputs: [\"cdkPortalOutlet\"], outputs: [\"attached\"], exportAs: [\"cdkPortalOutlet\"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStep, decorators: [{\n type: Component,\n args: [{ selector: 'mat-step', providers: [\n { provide: ErrorStateMatcher, useExisting: MatStep },\n { provide: CdkStep, useExisting: MatStep },\n ], encapsulation: ViewEncapsulation.None, exportAs: 'matStep', changeDetection: ChangeDetectionStrategy.OnPush, template: \"\\n \\n \\n\\n\" }]\n }], ctorParameters: function () { return [{ type: MatStepper, decorators: [{\n type: Inject,\n args: [forwardRef(() => MatStepper)]\n }] }, { type: i1.ErrorStateMatcher, decorators: [{\n type: SkipSelf\n }] }, { type: i0.ViewContainerRef }, { type: undefined, decorators: [{\n type: Optional\n }, {\n type: Inject,\n args: [STEPPER_GLOBAL_OPTIONS]\n }] }]; }, propDecorators: { stepLabel: [{\n type: ContentChild,\n args: [MatStepLabel]\n }], color: [{\n type: Input\n }], _lazyContent: [{\n type: ContentChild,\n args: [MatStepContent, { static: false }]\n }] } });\nclass MatStepper extends CdkStepper {\n constructor(dir, changeDetectorRef, elementRef) {\n super(dir, changeDetectorRef, elementRef);\n /** Steps that belong to the current stepper, excluding ones from nested steppers. */\n this.steps = new QueryList();\n /** Event emitted when the current step is done transitioning in. */\n this.animationDone = new EventEmitter();\n /**\n * Whether the label should display in bottom or end position.\n * Only applies in the `horizontal` orientation.\n */\n this.labelPosition = 'end';\n /**\n * Position of the stepper's header.\n * Only applies in the `horizontal` orientation.\n */\n this.headerPosition = 'top';\n /** Consumer-specified template-refs to be used to override the header icons. */\n this._iconOverrides = {};\n /** Stream of animation `done` events when the body expands/collapses. */\n this._animationDone = new Subject();\n this._animationDuration = '';\n const nodeName = elementRef.nativeElement.nodeName.toLowerCase();\n this.orientation = nodeName === 'mat-vertical-stepper' ? 'vertical' : 'horizontal';\n }\n /** Duration for the animation. Will be normalized to milliseconds if no units are set. */\n get animationDuration() {\n return this._animationDuration;\n }\n set animationDuration(value) {\n this._animationDuration = /^\\d+$/.test(value) ? value + 'ms' : value;\n }\n ngAfterContentInit() {\n super.ngAfterContentInit();\n this._icons.forEach(({ name, templateRef }) => (this._iconOverrides[name] = templateRef));\n // Mark the component for change detection whenever the content children query changes\n this.steps.changes.pipe(takeUntil(this._destroyed)).subscribe(() => {\n this._stateChanged();\n });\n this._animationDone\n .pipe(\n // This needs a `distinctUntilChanged` in order to avoid emitting the same event twice due\n // to a bug in animations where the `.done` callback gets invoked twice on some browsers.\n // See https://github.com/angular/angular/issues/24084\n distinctUntilChanged((x, y) => x.fromState === y.fromState && x.toState === y.toState), takeUntil(this._destroyed))\n .subscribe(event => {\n if (event.toState === 'current') {\n this.animationDone.emit();\n }\n });\n }\n _stepIsNavigable(index, step) {\n return step.completed || this.selectedIndex === index || !this.linear;\n }\n _getAnimationDuration() {\n if (this.animationDuration) {\n return this.animationDuration;\n }\n return this.orientation === 'horizontal'\n ? DEFAULT_HORIZONTAL_ANIMATION_DURATION\n : DEFAULT_VERTICAL_ANIMATION_DURATION;\n }\n}\nMatStepper.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepper, deps: [{ token: i3$1.Directionality, optional: true }, { token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });\nMatStepper.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepper, selector: \"mat-stepper, mat-vertical-stepper, mat-horizontal-stepper, [matStepper]\", inputs: { selectedIndex: \"selectedIndex\", disableRipple: \"disableRipple\", color: \"color\", labelPosition: \"labelPosition\", headerPosition: \"headerPosition\", animationDuration: \"animationDuration\" }, outputs: { animationDone: \"animationDone\" }, host: { attributes: { \"role\": \"tablist\" }, properties: { \"class.mat-stepper-horizontal\": \"orientation === \\\"horizontal\\\"\", \"class.mat-stepper-vertical\": \"orientation === \\\"vertical\\\"\", \"class.mat-stepper-label-position-end\": \"orientation === \\\"horizontal\\\" && labelPosition == \\\"end\\\"\", \"class.mat-stepper-label-position-bottom\": \"orientation === \\\"horizontal\\\" && labelPosition == \\\"bottom\\\"\", \"class.mat-stepper-header-position-bottom\": \"headerPosition === \\\"bottom\\\"\", \"attr.aria-orientation\": \"orientation\" } }, providers: [{ provide: CdkStepper, useExisting: MatStepper }], queries: [{ propertyName: \"_steps\", predicate: MatStep, descendants: true }, { propertyName: \"_icons\", predicate: MatStepperIcon, descendants: true }], viewQueries: [{ propertyName: \"_stepHeader\", predicate: MatStepHeader, descendants: true }], exportAs: [\"matStepper\", \"matVerticalStepper\", \"matHorizontalStepper\"], usesInheritance: true, ngImport: i0, template: \"\\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\", styles: [\".mat-stepper-vertical,.mat-stepper-horizontal{display:block}.mat-horizontal-stepper-header-container{white-space:nowrap;display:flex;align-items:center}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header-container{align-items:flex-start}.mat-stepper-header-position-bottom .mat-horizontal-stepper-header-container{order:1}.mat-stepper-horizontal-line{border-top-width:1px;border-top-style:solid;flex:auto;height:0;margin:0 -16px;min-width:32px}.mat-stepper-label-position-bottom .mat-stepper-horizontal-line{margin:0;min-width:0;position:relative}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::before,.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::after,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::after{border-top-width:1px;border-top-style:solid;content:\\\"\\\";display:inline-block;height:0;position:absolute;width:calc(50% - 20px)}.mat-horizontal-stepper-header{display:flex;height:72px;overflow:hidden;align-items:center;padding:0 24px}.mat-horizontal-stepper-header .mat-step-icon{margin-right:8px;flex:none}[dir=rtl] .mat-horizontal-stepper-header .mat-step-icon{margin-right:0;margin-left:8px}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header{box-sizing:border-box;flex-direction:column;height:auto}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::after,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::after{right:0}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::before{left:0}[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:last-child::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:first-child::after{display:none}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header .mat-step-icon{margin-right:0;margin-left:0}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header .mat-step-label{padding:16px 0 0 0;text-align:center;width:100%}.mat-vertical-stepper-header{display:flex;align-items:center;height:24px}.mat-vertical-stepper-header .mat-step-icon{margin-right:12px}[dir=rtl] .mat-vertical-stepper-header .mat-step-icon{margin-right:0;margin-left:12px}.mat-horizontal-stepper-wrapper{display:flex;flex-direction:column}.mat-horizontal-stepper-content{outline:0}.mat-horizontal-stepper-content.mat-horizontal-stepper-content-inactive{height:0;overflow:hidden}.mat-horizontal-stepper-content:not(.mat-horizontal-stepper-content-inactive){visibility:inherit !important}.mat-horizontal-content-container{overflow:hidden;padding:0 24px 24px 24px}.cdk-high-contrast-active .mat-horizontal-content-container{outline:solid 1px}.mat-stepper-header-position-bottom .mat-horizontal-content-container{padding:24px 24px 0 24px}.mat-vertical-content-container{margin-left:36px;border:0;position:relative}.cdk-high-contrast-active .mat-vertical-content-container{outline:solid 1px}[dir=rtl] .mat-vertical-content-container{margin-left:0;margin-right:36px}.mat-stepper-vertical-line::before{content:\\\"\\\";position:absolute;left:0;border-left-width:1px;border-left-style:solid}[dir=rtl] .mat-stepper-vertical-line::before{left:auto;right:0}.mat-vertical-stepper-content{overflow:hidden;outline:0}.mat-vertical-stepper-content:not(.mat-vertical-stepper-content-inactive){visibility:inherit !important}.mat-vertical-content{padding:0 24px 24px 24px}.mat-step:last-child .mat-vertical-content-container{border:none}\"], dependencies: [{ kind: \"directive\", type: i3.NgForOf, selector: \"[ngFor][ngForOf]\", inputs: [\"ngForOf\", \"ngForTrackBy\", \"ngForTemplate\"] }, { kind: \"directive\", type: i3.NgIf, selector: \"[ngIf]\", inputs: [\"ngIf\", \"ngIfThen\", \"ngIfElse\"] }, { kind: \"directive\", type: i3.NgTemplateOutlet, selector: \"[ngTemplateOutlet]\", inputs: [\"ngTemplateOutletContext\", \"ngTemplateOutlet\", \"ngTemplateOutletInjector\"] }, { kind: \"directive\", type: i3.NgSwitch, selector: \"[ngSwitch]\", inputs: [\"ngSwitch\"] }, { kind: \"directive\", type: i3.NgSwitchCase, selector: \"[ngSwitchCase]\", inputs: [\"ngSwitchCase\"] }, { kind: \"component\", type: MatStepHeader, selector: \"mat-step-header\", inputs: [\"color\", \"state\", \"label\", \"errorMessage\", \"iconOverrides\", \"index\", \"selected\", \"active\", \"optional\", \"disableRipple\"] }], animations: [\n matStepperAnimations.horizontalStepTransition,\n matStepperAnimations.verticalStepTransition,\n ], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepper, decorators: [{\n type: Component,\n args: [{ selector: 'mat-stepper, mat-vertical-stepper, mat-horizontal-stepper, [matStepper]', exportAs: 'matStepper, matVerticalStepper, matHorizontalStepper', inputs: ['selectedIndex'], host: {\n '[class.mat-stepper-horizontal]': 'orientation === \"horizontal\"',\n '[class.mat-stepper-vertical]': 'orientation === \"vertical\"',\n '[class.mat-stepper-label-position-end]': 'orientation === \"horizontal\" && labelPosition == \"end\"',\n '[class.mat-stepper-label-position-bottom]': 'orientation === \"horizontal\" && labelPosition == \"bottom\"',\n '[class.mat-stepper-header-position-bottom]': 'headerPosition === \"bottom\"',\n '[attr.aria-orientation]': 'orientation',\n 'role': 'tablist',\n }, animations: [\n matStepperAnimations.horizontalStepTransition,\n matStepperAnimations.verticalStepTransition,\n ], providers: [{ provide: CdkStepper, useExisting: MatStepper }], encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: \"\\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\", styles: [\".mat-stepper-vertical,.mat-stepper-horizontal{display:block}.mat-horizontal-stepper-header-container{white-space:nowrap;display:flex;align-items:center}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header-container{align-items:flex-start}.mat-stepper-header-position-bottom .mat-horizontal-stepper-header-container{order:1}.mat-stepper-horizontal-line{border-top-width:1px;border-top-style:solid;flex:auto;height:0;margin:0 -16px;min-width:32px}.mat-stepper-label-position-bottom .mat-stepper-horizontal-line{margin:0;min-width:0;position:relative}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::before,.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::after,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::after{border-top-width:1px;border-top-style:solid;content:\\\"\\\";display:inline-block;height:0;position:absolute;width:calc(50% - 20px)}.mat-horizontal-stepper-header{display:flex;height:72px;overflow:hidden;align-items:center;padding:0 24px}.mat-horizontal-stepper-header .mat-step-icon{margin-right:8px;flex:none}[dir=rtl] .mat-horizontal-stepper-header .mat-step-icon{margin-right:0;margin-left:8px}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header{box-sizing:border-box;flex-direction:column;height:auto}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::after,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::after{right:0}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:first-child)::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:not(:last-child)::before{left:0}[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:last-child::before,[dir=rtl] .mat-stepper-label-position-bottom .mat-horizontal-stepper-header:first-child::after{display:none}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header .mat-step-icon{margin-right:0;margin-left:0}.mat-stepper-label-position-bottom .mat-horizontal-stepper-header .mat-step-label{padding:16px 0 0 0;text-align:center;width:100%}.mat-vertical-stepper-header{display:flex;align-items:center;height:24px}.mat-vertical-stepper-header .mat-step-icon{margin-right:12px}[dir=rtl] .mat-vertical-stepper-header .mat-step-icon{margin-right:0;margin-left:12px}.mat-horizontal-stepper-wrapper{display:flex;flex-direction:column}.mat-horizontal-stepper-content{outline:0}.mat-horizontal-stepper-content.mat-horizontal-stepper-content-inactive{height:0;overflow:hidden}.mat-horizontal-stepper-content:not(.mat-horizontal-stepper-content-inactive){visibility:inherit !important}.mat-horizontal-content-container{overflow:hidden;padding:0 24px 24px 24px}.cdk-high-contrast-active .mat-horizontal-content-container{outline:solid 1px}.mat-stepper-header-position-bottom .mat-horizontal-content-container{padding:24px 24px 0 24px}.mat-vertical-content-container{margin-left:36px;border:0;position:relative}.cdk-high-contrast-active .mat-vertical-content-container{outline:solid 1px}[dir=rtl] .mat-vertical-content-container{margin-left:0;margin-right:36px}.mat-stepper-vertical-line::before{content:\\\"\\\";position:absolute;left:0;border-left-width:1px;border-left-style:solid}[dir=rtl] .mat-stepper-vertical-line::before{left:auto;right:0}.mat-vertical-stepper-content{overflow:hidden;outline:0}.mat-vertical-stepper-content:not(.mat-vertical-stepper-content-inactive){visibility:inherit !important}.mat-vertical-content{padding:0 24px 24px 24px}.mat-step:last-child .mat-vertical-content-container{border:none}\"] }]\n }], ctorParameters: function () { return [{ type: i3$1.Directionality, decorators: [{\n type: Optional\n }] }, { type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { _stepHeader: [{\n type: ViewChildren,\n args: [MatStepHeader]\n }], _steps: [{\n type: ContentChildren,\n args: [MatStep, { descendants: true }]\n }], _icons: [{\n type: ContentChildren,\n args: [MatStepperIcon, { descendants: true }]\n }], animationDone: [{\n type: Output\n }], disableRipple: [{\n type: Input\n }], color: [{\n type: Input\n }], labelPosition: [{\n type: Input\n }], headerPosition: [{\n type: Input\n }], animationDuration: [{\n type: Input\n }] } });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n/** Button that moves to the next step in a stepper workflow. */\nclass MatStepperNext extends CdkStepperNext {\n}\nMatStepperNext.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperNext, deps: null, target: i0.ɵɵFactoryTarget.Directive });\nMatStepperNext.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepperNext, selector: \"button[matStepperNext]\", inputs: { type: \"type\" }, host: { properties: { \"type\": \"type\" }, classAttribute: \"mat-stepper-next\" }, usesInheritance: true, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperNext, decorators: [{\n type: Directive,\n args: [{\n selector: 'button[matStepperNext]',\n host: {\n 'class': 'mat-stepper-next',\n '[type]': 'type',\n },\n inputs: ['type'],\n }]\n }] });\n/** Button that moves to the previous step in a stepper workflow. */\nclass MatStepperPrevious extends CdkStepperPrevious {\n}\nMatStepperPrevious.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperPrevious, deps: null, target: i0.ɵɵFactoryTarget.Directive });\nMatStepperPrevious.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: \"14.0.0\", version: \"14.2.0\", type: MatStepperPrevious, selector: \"button[matStepperPrevious]\", inputs: { type: \"type\" }, host: { properties: { \"type\": \"type\" }, classAttribute: \"mat-stepper-previous\" }, usesInheritance: true, ngImport: i0 });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperPrevious, decorators: [{\n type: Directive,\n args: [{\n selector: 'button[matStepperPrevious]',\n host: {\n 'class': 'mat-stepper-previous',\n '[type]': 'type',\n },\n inputs: ['type'],\n }]\n }] });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nclass MatStepperModule {\n}\nMatStepperModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });\nMatStepperModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: \"14.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperModule, declarations: [MatStep,\n MatStepLabel,\n MatStepper,\n MatStepperNext,\n MatStepperPrevious,\n MatStepHeader,\n MatStepperIcon,\n MatStepContent], imports: [MatCommonModule,\n CommonModule,\n PortalModule,\n MatButtonModule,\n CdkStepperModule,\n MatIconModule,\n MatRippleModule], exports: [MatCommonModule,\n MatStep,\n MatStepLabel,\n MatStepper,\n MatStepperNext,\n MatStepperPrevious,\n MatStepHeader,\n MatStepperIcon,\n MatStepContent] });\nMatStepperModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperModule, providers: [MAT_STEPPER_INTL_PROVIDER, ErrorStateMatcher], imports: [MatCommonModule,\n CommonModule,\n PortalModule,\n MatButtonModule,\n CdkStepperModule,\n MatIconModule,\n MatRippleModule, MatCommonModule] });\ni0.ɵɵngDeclareClassMetadata({ minVersion: \"12.0.0\", version: \"14.2.0\", ngImport: i0, type: MatStepperModule, decorators: [{\n type: NgModule,\n args: [{\n imports: [\n MatCommonModule,\n CommonModule,\n PortalModule,\n MatButtonModule,\n CdkStepperModule,\n MatIconModule,\n MatRippleModule,\n ],\n exports: [\n MatCommonModule,\n MatStep,\n MatStepLabel,\n MatStepper,\n MatStepperNext,\n MatStepperPrevious,\n MatStepHeader,\n MatStepperIcon,\n MatStepContent,\n ],\n declarations: [\n MatStep,\n MatStepLabel,\n MatStepper,\n MatStepperNext,\n MatStepperPrevious,\n MatStepHeader,\n MatStepperIcon,\n MatStepContent,\n ],\n providers: [MAT_STEPPER_INTL_PROVIDER, ErrorStateMatcher],\n }]\n }] });\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { MAT_STEPPER_INTL_PROVIDER, MAT_STEPPER_INTL_PROVIDER_FACTORY, MatStep, MatStepContent, MatStepHeader, MatStepLabel, MatStepper, MatStepperIcon, MatStepperIntl, MatStepperModule, MatStepperNext, MatStepperPrevious, matStepperAnimations };\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { Ticket } from '@app/modules/shared/models/ticket';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, Observable, throwError } from 'rxjs';\nimport { catchError, map } from 'rxjs/operators';\nimport { Stop } from '../models/od';\n\n@Injectable()\nexport class TicketService {\n constructor(private http: HttpClient, private authService: AuthService, private router: Router) {}\n\n public cancelTicket(ticket_id: number) {\n return firstValueFrom(\n this.http.post(env.config.feedRoot + 'Ticket/cancelTicket.json', { ticket_id }).pipe(\n map(({ response: r }) => {\n return r.status === 'ok';\n })\n )\n );\n }\n\n public reactivateTicket(ticket_id: number) {\n const http$ = this.http\n .post(env.config.feedRoot + 'Ticket/reactivateTicket', { ticket_id })\n .pipe(\n map(({ response }) => {\n return response.status === 'ok';\n })\n );\n\n return firstValueFrom(http$);\n }\n\n public generateTicketsV1(\n user_id: string,\n item_id: string,\n product_id: string,\n quantity: number,\n user_photo: string,\n dematerialized: string,\n od = null,\n start_validity_date = null,\n end_validity_date = null,\n generationReason = null,\n generationComment: string | null = null\n ) {\n return this.http\n .post(env.config.feedRoot + 'Ticket/generateTicket.json', {\n user_id,\n item_id,\n product_id,\n quantity,\n user_photo,\n dematerialized,\n od,\n start_validity_date,\n end_validity_date,\n generationReason,\n generationComment,\n })\n .pipe(\n map(({ response: r }) => {\n return (r?.tickets as Ticket[]) || [];\n })\n );\n }\n\n public generateTicketsV2(\n generationReason: string,\n blueprintId: string,\n customerId: string,\n networkId: string,\n startDate = null,\n endDate = null,\n generationComment: string | null = null\n ) {\n return this.http\n .post(env.config.feedRoot + 'Wallet/createContract.json', {\n generationReason,\n blueprintId,\n customerId,\n networkId,\n startDate,\n endDate,\n generationComment,\n })\n .pipe(\n map(({ response: r }) => {\n return r?.contract || null;\n })\n );\n }\n\n public getValidations(): Observable {\n return this.http.get(env.config.feedRoot + 'TicketValidation/getValidation.json').pipe(\n map(({ response: r }) => {\n return r.validation;\n }),\n catchError(() => throwError('Erreur serveur !'))\n );\n }\n\n public getODStops(productId: string, originStop: string = null): Observable {\n const params: {\n product_id: string;\n origin_stop?: string;\n } = {\n product_id: productId,\n };\n\n if (originStop) {\n params.origin_stop = originStop;\n }\n\n return this.http\n .get>(env.config.feedRoot + 'Transit/getODStops.json', {\n params,\n })\n .pipe(\n map(({ response }) => {\n return response.stops;\n })\n );\n }\n\n public getODTrips(productId: string, originStop: string, destinationStop: string) {\n return this.http\n .get(env.config.feedRoot + `Transit/ODTrips.json`, {\n params: {\n productId,\n originStop,\n destinationStop,\n },\n })\n .pipe(\n map(({ response }) => {\n return response.trips;\n })\n );\n }\n}\n","
    {{ title }}
    \n\n
    \n \n

    {{ 'pages.customer_details.anonymization_modal_message_1' | translate }}

    \n

    \n {{ 'pages.customer_details.anonymization_modal_message_2' | translate }}\n {{ isCustomerEmailAvailable ? data.customerEmail : data.customerId }}\n {{ 'pages.customer_details.anonymization_modal_message_3' | translate }}\n

    \n
    \n \n \n
    \n \n

    \n {{ 'pages.customer_details.forbidden_anonymize' | translate }}\n

    \n

    \n \n {{ 'pages.customer_details.ask_for_delete' | translate }}\n \n

    \n
    \n
    \n\n
    \n \n\n \n \n {{ !loading ? ('otherslabels.btn_confirm' | translate) : ('otherslabels.loading' | translate) }}\n \n
    \n\n\n","import { Component, Inject } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { firstValueFrom } from 'rxjs';\nimport { CustomersService } from '../../../services/customers.service';\n\nexport interface DialogData {\n customerId: number;\n customerEmail: string;\n customerAnonymized: string;\n}\n\n@Component({\n selector: 'tu-anonymization-modal',\n templateUrl: './anonymization-modal.component.html',\n providers: [CustomersService],\n})\nexport class AnonymizationModalComponent {\n public anonymizeInput: string;\n public loading = false;\n public isForbidden = false;\n\n public notificationOptions = {\n timeOut: 4000,\n showProgressBar: true,\n pauseOnHover: true,\n clickToClose: false,\n };\n\n public get isDisabled() {\n if (!this.data.customerEmail) {\n const isId = this.anonymizeInput === String(this.data.customerId);\n return !isId;\n }\n\n const isEmail = this.anonymizeInput === this.data.customerEmail;\n return !isEmail;\n }\n public get title() {\n return this.translate.instant(\n !this.isForbidden\n ? this.translate.instant('pages.customer_details.anonymization_modal_title')\n : this.translate.instant('pages.customer_details.user_anonymization_failed')\n );\n }\n\n constructor(\n public modalRef: MatDialogRef,\n private customersService: CustomersService,\n private readonly translate: TranslateService,\n private notification: NotificationsService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n public get isCustomerEmailAvailable() {\n return Boolean(this.data.customerEmail);\n }\n\n public onCancel(): void {\n if (this.loading || this.data.customerAnonymized) {\n return;\n }\n\n this.modalRef.close();\n }\n\n public async confirmAnonymisation() {\n if (this.loading || this.data.customerAnonymized) {\n return;\n }\n\n this.anonymizeInput = '';\n this.loading = true;\n\n try {\n const { response } = await this.customersService.anonymizeCustomer(\n this.data.customerId.toString()\n );\n\n //OK\n if (response?.errors.length === 0) {\n const notification = this.notification.success(\n this.translate.instant('pages.customer_details.user_anonymized')\n );\n await firstValueFrom(notification.timeoutEnd);\n location.reload();\n return;\n }\n //KO\n this.notification.error(\n this.translate.instant('pages.customer_details.user_anonymization_failed')\n );\n } catch ({ error }) {\n if (error.code === 'CUSTOMER_ANONYMIZATION_FORBIDDEN') {\n this.isForbidden = true;\n this.notification.error(\n this.translate.instant('pages.customer_details.user_anonymization_forbidden'),\n this.translate.instant('pages.customer_details.user_anonymization_forbidden_message')\n );\n return;\n }\n this.notification.error(\n this.translate.instant('pages.customer_details.user_anonymization_failed')\n );\n } finally {\n this.loading = false;\n }\n }\n}\n","
    {{ 'pages.customer_details.edit_metadata' | translate }}e
    \n\n
    \n
    \n \n \n\n \n \n {{ attributeValue }}\n \n \n\n
    \n \n \n
    \n\n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n\n \n
    \n \n
    \n\n\n","import { Component, EventEmitter, Inject, OnInit, Output } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { CustomersService } from '../../../services/customers.service';\n\nexport interface DialogData {\n customerId: string;\n label: string;\n value: any;\n attributeId: number | string;\n attributeType: string;\n attributeValues: (string | boolean)[];\n}\n\n@Component({\n selector: 'tu-attributes-modal',\n templateUrl: './attributes-modal.component.html',\n providers: [CustomersService],\n})\nexport class AttributesModalComponent implements OnInit {\n @Output() updateField = new EventEmitter();\n\n public attributeInput: any;\n public loading = false;\n\n public notificationOptions = {\n timeOut: 3000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n constructor(\n public modalRef: MatDialogRef,\n private customersService: CustomersService,\n private notification: NotificationsService,\n private readonly translate: TranslateService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n ngOnInit() {\n this.attributeInput = this.data.value;\n }\n\n public closeModal(): void {\n this.modalRef.close();\n }\n\n public onSelectValue(value: boolean): void {\n this.attributeInput = value;\n }\n\n public async confirmAttributeEdit() {\n try {\n if (this.loading || this.attributeInput === this.data.value) return;\n\n this.loading = true;\n\n const result = await this.customersService.setCustomerAttribute(\n this.data.customerId,\n this.data.attributeId,\n this.attributeInput\n );\n\n const attributes = await this.customersService.getCustomerAttributes(this.data.customerId);\n\n if (result.error) {\n throw new Error();\n } else {\n this.updateField.emit(attributes);\n this.closeModal();\n this.notification.success(\n this.translate.instant('pages.customer_details.metadata_changed')\n );\n }\n } catch {\n this.notification.error(\n this.translate.instant('pages.customer_details.metadata_change_fail')\n );\n } finally {\n this.loading = false;\n }\n }\n}\n","
    \n \n {{ 'otherslabels.alert_ask_photo' | translate }}\n \n \n {{ 'otherslabels.label_choose_photo' | translate }}\n \n
    \n
    \n
    \n
    \n \n \n {{\n userPhoto.uri\n ? ('pages.customer_image_modal.change_image_button' | translate)\n : ('pages.customer_image_modal.new_image_button' | translate)\n }}\n \n \n \n \n {{ 'pages.customer_image_modal.crop_image_button' | translate }}\n \n \n {{ 'pages.customer_image_modal.cancel_crop_button' | translate }}\n \n
    \n \n \n
    \n
    \n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n \n {{\n !loading ? ('otherslabels.btn_save' | translate) : ('otherslabels.loading' | translate)\n }}\n \n \n
    \n
    \n
    \n\n","import {\n Component,\n ElementRef,\n EventEmitter,\n Inject,\n Input,\n OnInit,\n Output,\n ViewChild,\n} from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { Customer } from '@app/modules/customers/models/customer';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { firstValueFrom } from 'rxjs';\nimport { CustomersService } from '../../../services/customers.service';\n\n//https://stackoverflow.com/questions/4998908/convert-data-uri-to-file-then-append-to-formdata\nfunction dataURItoBlob(dataURI: string) {\n // convert base64/URLEncoded data component to raw binary data held in a string\n let byteString: string;\n if (dataURI.split(',')[0].indexOf('base64') >= 0) byteString = atob(dataURI.split(',')[1]);\n else byteString = unescape(dataURI.split(',')[1]);\n // separate out the mime component\n const mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0];\n // write the bytes of the string to a typed array\n const intArray = new Uint8Array(byteString.length);\n for (let i = 0; i < byteString.length; i++) {\n intArray[i] = byteString.charCodeAt(i);\n }\n return new Blob([intArray], { type: mimeString });\n}\n\nexport interface DialogData {\n customerId: string;\n picture: string;\n isTicketPicture: boolean;\n}\n@Component({\n selector: 'tu-avatar-update-modal',\n templateUrl: './avatar-update-modal.component.html',\n})\nexport class ImageUpdateModalComponent implements OnInit {\n @Input('withTitle')\n public withTitle = true;\n\n @Input('customerData')\n public customerData: Customer = null;\n\n @Output() updateField = new EventEmitter();\n\n @ViewChild('image', { static: false })\n public imageElement: ElementRef;\n\n public userPhoto = {\n file: null as File | null,\n uri: '',\n isModified: false,\n croppedUri: '',\n isCropping: false,\n };\n\n public loading: boolean = false;\n\n public notificationOptions = {\n timeOut: 3000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n constructor(\n public modalRef: MatDialogRef,\n private customersService: CustomersService,\n private translateService: TranslateService,\n private notification: NotificationsService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n ngOnInit() {\n this.userPhoto.uri = this.customerData ? this.customerData.picture : this.data.picture;\n }\n\n public closeModal(pictureUri?: string): void {\n this.modalRef.close(pictureUri);\n }\n\n public onChangeCroppedImageUri(newCroppedImageUri: string) {\n this.userPhoto.croppedUri = newCroppedImageUri;\n }\n\n public onClickToggleImageCrop(isCropping: boolean) {\n this.userPhoto.isCropping = isCropping;\n this.userPhoto.isModified = isCropping || Boolean(this.userPhoto.file);\n }\n\n public onClickToggleUpload() {\n this.userPhoto.isCropping = false;\n }\n\n public onFileSelect(event) {\n const file = event.target.files?.[0];\n if (!file) return;\n\n let reader = new FileReader();\n\n reader.addEventListener(\n 'load',\n () => {\n this.userPhoto.uri = reader.result as string;\n },\n false\n );\n reader.readAsDataURL(file);\n this.userPhoto.file = file;\n this.userPhoto.isModified = true;\n }\n\n public async onClickSubmitUserPhoto() {\n if (this.data.isTicketPicture && !this.userPhoto.isModified)\n this.closeModal(this.userPhoto.uri);\n\n if (!this.userPhoto.isModified) return;\n\n try {\n this.loading = true;\n\n if (this.userPhoto.isCropping) {\n const blob = dataURItoBlob(this.userPhoto.croppedUri);\n const file = new File([blob], this.userPhoto.croppedUri);\n this.userPhoto.file = file;\n this.userPhoto.uri = this.userPhoto.croppedUri;\n }\n\n const customerId = this.customerData ? this.customerData.customer_id : this.data.customerId;\n\n const response = await firstValueFrom(\n this.customersService.uploadImage(customerId, this.userPhoto.file)\n );\n\n if (response.success && 'url' in response) {\n this.userPhoto.uri = response.url;\n\n if (!this.data.isTicketPicture) {\n this.notification.success(\n this.translateService.instant(\n 'pages.customer_details.update_avatar_notification_title'\n ),\n this.translateService.instant(\n 'pages.customer_details.update_avatar_notification_content'\n )\n );\n }\n }\n\n if (response.error) {\n throw new Error();\n } else {\n const customerId = this.customerData ? this.customerData.customer_id : this.data.customerId;\n\n const customer = await this.customersService.getCustomerDetails(customerId);\n\n this.updateField.emit(customer);\n this.notification.success(\n this.translateService.instant(`pages.customer_details.information_changes_success`)\n );\n\n if (!this.customerData) {\n this.closeModal(this.userPhoto.uri);\n }\n }\n } catch (err) {\n console.error(err);\n this.notification.error(\n this.translateService.instant('pages.customer_details.generic_error')\n );\n } finally {\n this.loading = false;\n }\n }\n}\n","
    \n {{ 'pages.customer_details.update_personal_information' | translate }}\n
    \n\n
    \n {{ 'pages.customer_details.personnal_info_fc_info' | translate }}\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 {{ errorMessage }}\n
    \n
    \n\n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n\n \n
    \n \n
    \n\n\n","import { Component, EventEmitter, Inject, OnInit, Output } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { Address } from '@app/modules/shared/models/address';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { DateTime } from 'luxon';\nimport { CustomersService } from '../../../services/customers.service';\n\nexport interface DialogData {\n isCustomerParent: boolean;\n customerId: string;\n firstname: string;\n lastname: string;\n birthday: string;\n email: string;\n phone: string;\n address: Partial
    ;\n hasFranceConnectProvider: boolean;\n}\n\n@Component({\n selector: 'tu-information-modal',\n templateUrl: './information-modal.component.html',\n providers: [CustomersService],\n})\nexport class InformationModalComponent implements OnInit {\n @Output() updateField = new EventEmitter();\n\n public firstnameInput: string | null;\n public lastnameInput: string | null;\n public birthdayInput: string | null;\n public emailInput: string | null;\n public phoneInput: string | null;\n public streetNumberInput: string | null;\n public routeInput: string | null;\n public zipCodeInput: string | null;\n public cityInput: string | null;\n public countryInput: string | null;\n\n public loading: boolean = false;\n public errorMessage: string | null = null;\n\n public notificationOptions = {\n timeOut: 3000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n constructor(\n public modalRef: MatDialogRef,\n public translateService: TranslateService,\n private customersService: CustomersService,\n private notification: NotificationsService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n ngOnInit() {\n this.firstnameInput = this.data.firstname;\n this.lastnameInput = this.data.lastname;\n\n if (this.data.birthday) {\n const birthday = DateTime.fromISO(this.data.birthday).setZone('UTC');\n this.birthdayInput = birthday.toFormat('yyyy-MM-dd');\n } else {\n this.birthdayInput = null;\n }\n\n this.emailInput = this.data.email;\n this.phoneInput = this.data.phone;\n this.streetNumberInput = this.data.address?.streetNumber;\n this.routeInput = this.data.address?.route;\n this.zipCodeInput = this.data.address?.zipCode;\n this.cityInput = this.data.address?.city;\n this.countryInput = this.data.address?.country;\n }\n\n public closeModal(): void {\n this.modalRef.close();\n }\n\n public async confirmInformationEdit() {\n this.errorMessage = null;\n\n let newBirthday = this.birthdayInput\n ? DateTime.fromISO(this.birthdayInput, { zone: 'UTC' })\n : null;\n\n if (newBirthday) {\n const isBirthdayValid = newBirthday.isValid;\n\n if (!isBirthdayValid) {\n this.errorMessage = this.translateService.instant(\n `pages.customer_details.incorrect_birthday_format`\n );\n return;\n }\n }\n\n const hasSameFirstname = this.data.firstname === this.firstnameInput;\n const hasSameLastname = this.data.lastname === this.lastnameInput;\n\n // Compare timestamps\n const hasSameBirthday = newBirthday\n ? newBirthday.toMillis() === DateTime.fromISO(this.data.birthday).toMillis()\n : true;\n\n const hasSameEmail = this.data.email === this.emailInput;\n const hasSamePhone = this.data.phone === this.phoneInput;\n const hasSameAddress =\n this.data.address?.streetNumber === this.streetNumberInput &&\n this.data.address?.route === this.routeInput &&\n this.data.address?.zipCode === this.zipCodeInput &&\n this.data.address?.city === this.cityInput &&\n this.data.address?.country === this.countryInput;\n\n const hasSameInfos =\n hasSameFirstname &&\n hasSameLastname &&\n hasSameBirthday &&\n hasSameEmail &&\n hasSamePhone &&\n hasSameAddress;\n\n if (hasSameInfos) {\n this.errorMessage = this.translateService.instant(\n `pages.customer_details.no_information_changes`\n );\n return;\n }\n\n try {\n this.loading = true;\n\n const customerAddress: Address = {\n streetNumber: this.streetNumberInput,\n route: this.routeInput,\n city: this.cityInput,\n zipCode: this.zipCodeInput,\n country: this.countryInput,\n };\n\n const body: Record = {\n phone: this.phoneInput,\n address: customerAddress,\n };\n\n if (newBirthday) {\n body.birthday = newBirthday.toISO();\n }\n\n if (!this.data.hasFranceConnectProvider) {\n body.firstname = this.firstnameInput;\n body.lastname = this.lastnameInput;\n body.email = this.emailInput;\n }\n\n const result = await this.customersService.updateInfo(this.data.customerId, body);\n\n if (result.error) {\n throw new Error();\n } else {\n const customer = await this.customersService.getCustomerDetails(this.data.customerId);\n\n this.updateField.emit(customer);\n this.notification.success(\n this.translateService.instant(`pages.customer_details.information_changes_success`)\n );\n this.closeModal();\n }\n } catch {\n this.notification.error(\n this.translateService.instant(`pages.customer_details.information_changes_fail`)\n );\n } finally {\n this.loading = false;\n }\n }\n}\n","import { Component, Inject } from '@angular/core';\nimport { MAT_DIALOG_DATA } from '@angular/material/dialog';\n\nexport interface DialogData {\n title: string;\n text: string;\n}\n\n/**\n * @title Dialog Overview\n */\n@Component({\n templateUrl: 'customer-lost-phone-modal.component.html',\n})\nexport class CustomerLostPhoneModalComponent {\n constructor(@Inject(MAT_DIALOG_DATA) public data: DialogData) {}\n}\n","

    \n {{ data.title }}\n

    \n
    \n

    \n
    \n
    \n \n \n
    \n","import { Component, EventEmitter, Inject, Output } from '@angular/core';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\n\nexport interface DialogData {\n networkId: number;\n customerId: number;\n}\n\n@Component({\n selector: 'tu-network-dissociation-modal',\n templateUrl: './network-dissociation-modal.component.html',\n providers: [CustomersService],\n})\nexport class NetworkDissociationModalComponent {\n @Output() updateField = new EventEmitter();\n\n public loading = false;\n\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n constructor(\n public modalRef: MatDialogRef,\n private customersService: CustomersService,\n private notification: NotificationsService,\n private translate: TranslateService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n public closeModal(): void {\n this.modalRef.close();\n }\n\n public async confirmNetworkDissociationDelete() {\n this.loading = true;\n\n try {\n const { networkIds } = await this.customersService.unlinkNetwork(\n this.data.customerId,\n this.data.networkId\n );\n\n this.updateField.emit(networkIds);\n\n this.notification.success(\n this.translate.instant('pages.customer_details.network_dissociation_success_notification')\n );\n\n this.closeModal();\n } catch ({ error }) {\n switch (error.code) {\n case 'NETWORK_FORBIDDEN':\n this.notification.error(\n this.translate.instant('pages.customer_details.network_forbidden')\n );\n break;\n case 'CUSTOMER_NOT_FOUND':\n this.notification.error(\n this.translate.instant('pages.customer_details.customer_not_found')\n );\n break;\n case 'CUSTOMER_NETWORK_ASSOCIATION_NOT_FOUND':\n this.notification.error(\n this.translate.instant('pages.customer_details.network_association_not_found')\n );\n break;\n default:\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n }\n } finally {\n this.loading = false;\n }\n }\n}\n","
    \n {{ 'pages.customer_details.network_dissociation_modal_title' | translate }}\n
    \n\n
    \n

    \n {{ 'pages.customer_details.network_dissociation_modal_description' | translate }}\n

    \n\n
    \n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n\n \n
    \n
    \n
    \n\n\n","import parse from \"../parse/index.js\";\nimport isValid from \"../isValid/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name isMatch\n * @category Common Helpers\n * @summary validates the date string against given formats\n *\n * @description\n * Return the true if given date is string correct against the given format else\n * will return false.\n *\n * > ⚠️ Please note that the `format` tokens differ from Moment.js and other libraries.\n * > See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * The characters in the format string wrapped between two single quotes characters (') are escaped.\n * Two single quotes in a row, whether inside or outside a quoted sequence, represent a 'real' single quote.\n *\n * Format of the format string is based on Unicode Technical Standard #35:\n * https://www.unicode.org/reports/tr35/tr35-dates.html#Date_Field_Symbol_Table\n * with a few additions (see note 5 below the table).\n *\n * Not all tokens are compatible. Combinations that don't make sense or could lead to bugs are prohibited\n * and will throw `RangeError`. For example usage of 24-hour format token with AM/PM token will throw an exception:\n *\n * ```javascript\n * isMatch('23 AM', 'HH a')\n * //=> RangeError: The format string mustn't contain `HH` and `a` at the same time\n * ```\n *\n * See the compatibility table: https://docs.google.com/spreadsheets/d/e/2PACX-1vQOPU3xUhplll6dyoMmVUXHKl_8CRDs6_ueLmex3SoqwhuolkuN3O05l4rqx5h1dKX8eb46Ul-CCSrq/pubhtml?gid=0&single=true\n *\n * Accepted format string patterns:\n * | Unit |Prior| Pattern | Result examples | Notes |\n * |---------------------------------|-----|---------|-----------------------------------|-------|\n * | Era | 140 | G..GGG | AD, BC | |\n * | | | GGGG | Anno Domini, Before Christ | 2 |\n * | | | GGGGG | A, B | |\n * | Calendar year | 130 | y | 44, 1, 1900, 2017, 9999 | 4 |\n * | | | yo | 44th, 1st, 1900th, 9999999th | 4,5 |\n * | | | yy | 44, 01, 00, 17 | 4 |\n * | | | yyy | 044, 001, 123, 999 | 4 |\n * | | | yyyy | 0044, 0001, 1900, 2017 | 4 |\n * | | | yyyyy | ... | 2,4 |\n * | Local week-numbering year | 130 | Y | 44, 1, 1900, 2017, 9000 | 4 |\n * | | | Yo | 44th, 1st, 1900th, 9999999th | 4,5 |\n * | | | YY | 44, 01, 00, 17 | 4,6 |\n * | | | YYY | 044, 001, 123, 999 | 4 |\n * | | | YYYY | 0044, 0001, 1900, 2017 | 4,6 |\n * | | | YYYYY | ... | 2,4 |\n * | ISO week-numbering year | 130 | R | -43, 1, 1900, 2017, 9999, -9999 | 4,5 |\n * | | | RR | -43, 01, 00, 17 | 4,5 |\n * | | | RRR | -043, 001, 123, 999, -999 | 4,5 |\n * | | | RRRR | -0043, 0001, 2017, 9999, -9999 | 4,5 |\n * | | | RRRRR | ... | 2,4,5 |\n * | Extended year | 130 | u | -43, 1, 1900, 2017, 9999, -999 | 4 |\n * | | | uu | -43, 01, 99, -99 | 4 |\n * | | | uuu | -043, 001, 123, 999, -999 | 4 |\n * | | | uuuu | -0043, 0001, 2017, 9999, -9999 | 4 |\n * | | | uuuuu | ... | 2,4 |\n * | Quarter (formatting) | 120 | Q | 1, 2, 3, 4 | |\n * | | | Qo | 1st, 2nd, 3rd, 4th | 5 |\n * | | | QQ | 01, 02, 03, 04 | |\n * | | | QQQ | Q1, Q2, Q3, Q4 | |\n * | | | QQQQ | 1st quarter, 2nd quarter, ... | 2 |\n * | | | QQQQQ | 1, 2, 3, 4 | 4 |\n * | Quarter (stand-alone) | 120 | q | 1, 2, 3, 4 | |\n * | | | qo | 1st, 2nd, 3rd, 4th | 5 |\n * | | | qq | 01, 02, 03, 04 | |\n * | | | qqq | Q1, Q2, Q3, Q4 | |\n * | | | qqqq | 1st quarter, 2nd quarter, ... | 2 |\n * | | | qqqqq | 1, 2, 3, 4 | 3 |\n * | Month (formatting) | 110 | M | 1, 2, ..., 12 | |\n * | | | Mo | 1st, 2nd, ..., 12th | 5 |\n * | | | MM | 01, 02, ..., 12 | |\n * | | | MMM | Jan, Feb, ..., Dec | |\n * | | | MMMM | January, February, ..., December | 2 |\n * | | | MMMMM | J, F, ..., D | |\n * | Month (stand-alone) | 110 | L | 1, 2, ..., 12 | |\n * | | | Lo | 1st, 2nd, ..., 12th | 5 |\n * | | | LL | 01, 02, ..., 12 | |\n * | | | LLL | Jan, Feb, ..., Dec | |\n * | | | LLLL | January, February, ..., December | 2 |\n * | | | LLLLL | J, F, ..., D | |\n * | Local week of year | 100 | w | 1, 2, ..., 53 | |\n * | | | wo | 1st, 2nd, ..., 53th | 5 |\n * | | | ww | 01, 02, ..., 53 | |\n * | ISO week of year | 100 | I | 1, 2, ..., 53 | 5 |\n * | | | Io | 1st, 2nd, ..., 53th | 5 |\n * | | | II | 01, 02, ..., 53 | 5 |\n * | Day of month | 90 | d | 1, 2, ..., 31 | |\n * | | | do | 1st, 2nd, ..., 31st | 5 |\n * | | | dd | 01, 02, ..., 31 | |\n * | Day of year | 90 | D | 1, 2, ..., 365, 366 | 7 |\n * | | | Do | 1st, 2nd, ..., 365th, 366th | 5 |\n * | | | DD | 01, 02, ..., 365, 366 | 7 |\n * | | | DDD | 001, 002, ..., 365, 366 | |\n * | | | DDDD | ... | 2 |\n * | Day of week (formatting) | 90 | E..EEE | Mon, Tue, Wed, ..., Su | |\n * | | | EEEE | Monday, Tuesday, ..., Sunday | 2 |\n * | | | EEEEE | M, T, W, T, F, S, S | |\n * | | | EEEEEE | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | ISO day of week (formatting) | 90 | i | 1, 2, 3, ..., 7 | 5 |\n * | | | io | 1st, 2nd, ..., 7th | 5 |\n * | | | ii | 01, 02, ..., 07 | 5 |\n * | | | iii | Mon, Tue, Wed, ..., Su | 5 |\n * | | | iiii | Monday, Tuesday, ..., Sunday | 2,5 |\n * | | | iiiii | M, T, W, T, F, S, S | 5 |\n * | | | iiiiii | Mo, Tu, We, Th, Fr, Sa, Su | 5 |\n * | Local day of week (formatting) | 90 | e | 2, 3, 4, ..., 1 | |\n * | | | eo | 2nd, 3rd, ..., 1st | 5 |\n * | | | ee | 02, 03, ..., 01 | |\n * | | | eee | Mon, Tue, Wed, ..., Su | |\n * | | | eeee | Monday, Tuesday, ..., Sunday | 2 |\n * | | | eeeee | M, T, W, T, F, S, S | |\n * | | | eeeeee | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | Local day of week (stand-alone) | 90 | c | 2, 3, 4, ..., 1 | |\n * | | | co | 2nd, 3rd, ..., 1st | 5 |\n * | | | cc | 02, 03, ..., 01 | |\n * | | | ccc | Mon, Tue, Wed, ..., Su | |\n * | | | cccc | Monday, Tuesday, ..., Sunday | 2 |\n * | | | ccccc | M, T, W, T, F, S, S | |\n * | | | cccccc | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | AM, PM | 80 | a..aaa | AM, PM | |\n * | | | aaaa | a.m., p.m. | 2 |\n * | | | aaaaa | a, p | |\n * | AM, PM, noon, midnight | 80 | b..bbb | AM, PM, noon, midnight | |\n * | | | bbbb | a.m., p.m., noon, midnight | 2 |\n * | | | bbbbb | a, p, n, mi | |\n * | Flexible day period | 80 | B..BBB | at night, in the morning, ... | |\n * | | | BBBB | at night, in the morning, ... | 2 |\n * | | | BBBBB | at night, in the morning, ... | |\n * | Hour [1-12] | 70 | h | 1, 2, ..., 11, 12 | |\n * | | | ho | 1st, 2nd, ..., 11th, 12th | 5 |\n * | | | hh | 01, 02, ..., 11, 12 | |\n * | Hour [0-23] | 70 | H | 0, 1, 2, ..., 23 | |\n * | | | Ho | 0th, 1st, 2nd, ..., 23rd | 5 |\n * | | | HH | 00, 01, 02, ..., 23 | |\n * | Hour [0-11] | 70 | K | 1, 2, ..., 11, 0 | |\n * | | | Ko | 1st, 2nd, ..., 11th, 0th | 5 |\n * | | | KK | 01, 02, ..., 11, 00 | |\n * | Hour [1-24] | 70 | k | 24, 1, 2, ..., 23 | |\n * | | | ko | 24th, 1st, 2nd, ..., 23rd | 5 |\n * | | | kk | 24, 01, 02, ..., 23 | |\n * | Minute | 60 | m | 0, 1, ..., 59 | |\n * | | | mo | 0th, 1st, ..., 59th | 5 |\n * | | | mm | 00, 01, ..., 59 | |\n * | Second | 50 | s | 0, 1, ..., 59 | |\n * | | | so | 0th, 1st, ..., 59th | 5 |\n * | | | ss | 00, 01, ..., 59 | |\n * | Seconds timestamp | 40 | t | 512969520 | |\n * | | | tt | ... | 2 |\n * | Fraction of second | 30 | S | 0, 1, ..., 9 | |\n * | | | SS | 00, 01, ..., 99 | |\n * | | | SSS | 000, 001, ..., 999 | |\n * | | | SSSS | ... | 2 |\n * | Milliseconds timestamp | 20 | T | 512969520900 | |\n * | | | TT | ... | 2 |\n * | Timezone (ISO-8601 w/ Z) | 10 | X | -08, +0530, Z | |\n * | | | XX | -0800, +0530, Z | |\n * | | | XXX | -08:00, +05:30, Z | |\n * | | | XXXX | -0800, +0530, Z, +123456 | 2 |\n * | | | XXXXX | -08:00, +05:30, Z, +12:34:56 | |\n * | Timezone (ISO-8601 w/o Z) | 10 | x | -08, +0530, +00 | |\n * | | | xx | -0800, +0530, +0000 | |\n * | | | xxx | -08:00, +05:30, +00:00 | 2 |\n * | | | xxxx | -0800, +0530, +0000, +123456 | |\n * | | | xxxxx | -08:00, +05:30, +00:00, +12:34:56 | |\n * | Long localized date | NA | P | 05/29/1453 | 5,8 |\n * | | | PP | May 29, 1453 | |\n * | | | PPP | May 29th, 1453 | |\n * | | | PPPP | Sunday, May 29th, 1453 | 2,5,8 |\n * | Long localized time | NA | p | 12:00 AM | 5,8 |\n * | | | pp | 12:00:00 AM | |\n * | Combination of date and time | NA | Pp | 05/29/1453, 12:00 AM | |\n * | | | PPpp | May 29, 1453, 12:00:00 AM | |\n * | | | PPPpp | May 29th, 1453 at ... | |\n * | | | PPPPpp | Sunday, May 29th, 1453 at ... | 2,5,8 |\n * Notes:\n * 1. \"Formatting\" units (e.g. formatting quarter) in the default en-US locale\n * are the same as \"stand-alone\" units, but are different in some languages.\n * \"Formatting\" units are declined according to the rules of the language\n * in the context of a date. \"Stand-alone\" units are always nominative singular.\n * In `format` function, they will produce different result:\n *\n * `format(new Date(2017, 10, 6), 'do LLLL', {locale: cs}) //=> '6. listopad'`\n *\n * `format(new Date(2017, 10, 6), 'do MMMM', {locale: cs}) //=> '6. listopadu'`\n *\n * `isMatch` will try to match both formatting and stand-alone units interchangably.\n *\n * 2. Any sequence of the identical letters is a pattern, unless it is escaped by\n * the single quote characters (see below).\n * If the sequence is longer than listed in table:\n * - for numerical units (`yyyyyyyy`) `isMatch` will try to match a number\n * as wide as the sequence\n * - for text units (`MMMMMMMM`) `isMatch` will try to match the widest variation of the unit.\n * These variations are marked with \"2\" in the last column of the table.\n *\n * 3. `QQQQQ` and `qqqqq` could be not strictly numerical in some locales.\n * These tokens represent the shortest form of the quarter.\n *\n * 4. The main difference between `y` and `u` patterns are B.C. years:\n *\n * | Year | `y` | `u` |\n * |------|-----|-----|\n * | AC 1 | 1 | 1 |\n * | BC 1 | 1 | 0 |\n * | BC 2 | 2 | -1 |\n *\n * Also `yy` will try to guess the century of two digit year by proximity with `referenceDate`:\n *\n * `isMatch('50', 'yy') //=> true`\n *\n * `isMatch('75', 'yy') //=> true`\n *\n * while `uu` will use the year as is:\n *\n * `isMatch('50', 'uu') //=> true`\n *\n * `isMatch('75', 'uu') //=> true`\n *\n * The same difference is true for local and ISO week-numbering years (`Y` and `R`),\n * except local week-numbering years are dependent on `options.weekStartsOn`\n * and `options.firstWeekContainsDate` (compare [setISOWeekYear]{@link https://date-fns.org/docs/setISOWeekYear}\n * and [setWeekYear]{@link https://date-fns.org/docs/setWeekYear}).\n *\n * 5. These patterns are not in the Unicode Technical Standard #35:\n * - `i`: ISO day of week\n * - `I`: ISO week of year\n * - `R`: ISO week-numbering year\n * - `o`: ordinal number modifier\n * - `P`: long localized date\n * - `p`: long localized time\n *\n * 6. `YY` and `YYYY` tokens represent week-numbering years but they are often confused with years.\n * You should enable `options.useAdditionalWeekYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * 7. `D` and `DD` tokens represent days of the year but they are ofthen confused with days of the month.\n * You should enable `options.useAdditionalDayOfYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * 8. `P+` tokens do not have a defined priority since they are merely aliases to other tokens based\n * on the given locale.\n *\n * using `en-US` locale: `P` => `MM/dd/yyyy`\n * using `en-US` locale: `p` => `hh:mm a`\n * using `pt-BR` locale: `P` => `dd/MM/yyyy`\n * using `pt-BR` locale: `p` => `HH:mm`\n *\n * Values will be checked in the descending order of its unit's priority.\n * Units of an equal priority overwrite each other in the order of appearance.\n *\n * If no values of higher priority are matched (e.g. when matching string 'January 1st' without a year),\n * the values will be taken from today's using `new Date()` date which works as a context of parsing.\n *\n * The result may vary by locale.\n *\n * If `formatString` matches with `dateString` but does not provides tokens, `referenceDate` will be returned.\n *\n *\n *\n * @param {String} dateString - the date string to verify\n * @param {String} formatString - the string of tokens\n * @param {Object} [options] - an object with options.\n * @param {Locale} [options.locale=defaultLocale] - the locale object. See [Locale]{@link https://date-fns.org/docs/Locale}\n * @param {0|1|2|3|4|5|6} [options.weekStartsOn=0] - the index of the first day of the week (0 - Sunday)\n * @param {1|2|3|4|5|6|7} [options.firstWeekContainsDate=1] - the day of January, which is always in the first week of the year\n * @param {Boolean} [options.useAdditionalWeekYearTokens=false] - if true, allows usage of the week-numbering year tokens `YY` and `YYYY`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @param {Boolean} [options.useAdditionalDayOfYearTokens=false] - if true, allows usage of the day of year tokens `D` and `DD`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @returns {Boolean}\n * @throws {TypeError} 2 arguments required\n * @throws {RangeError} `options.weekStartsOn` must be between 0 and 6\n * @throws {RangeError} `options.firstWeekContainsDate` must be between 1 and 7\n * @throws {RangeError} `options.locale` must contain `match` property\n * @throws {RangeError} use `yyyy` instead of `YYYY` for formatting years; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `yy` instead of `YY` for formatting years; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `d` instead of `D` for formatting days of the month; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `dd` instead of `DD` for formatting days of the month; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} format string contains an unescaped latin alphabet character\n *\n * @example\n * // Match 11 February 2014 from middle-endian format:\n * const result = isMatch('02/11/2014', 'MM/dd/yyyy')\n * //=> true\n *\n * @example\n * // Match 28th of February in Esperanto locale in the context of 2010 year:\n * import eo from 'date-fns/locale/eo'\n * const result = isMatch('28-a de februaro', \"do 'de' MMMM\", {\n * locale: eo\n * })\n * //=> true\n */\nexport default function isMatch(dateString, formatString, options) {\n requiredArgs(2, arguments);\n return isValid(parse(dateString, formatString, new Date(), options));\n}","\n\n\n
    \n {{\n data.transfert\n ? ('pages.generate_ticket_modal.modal_title_redistribute' | translate)\n : ('pages.generate_ticket_modal.modal_title' | translate)\n }}\n
    \n\n \n \n
    \n \n {{ 'pages.generate_ticket_modal.product_choice' | translate }}\n \n\n
    \n \n\n
    \n \n {{ 'pages.generate_ticket_modal.select_network' | translate }}\n \n \n {{ customerNetwork.network.name }}\n \n \n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n \n {{ data.transfert?.network_name }}\n \n
    \n
    \n\n \n
    \n \n\n
    \n \n \n \n {{ category.name }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.select_network_first_message' | translate }}\n \n \n\n \n {{ category?.name || '' }}\n \n
    \n
    \n\n
    \n \n\n
    \n \n \n \n {{ product.name }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.select_category_first_message' | translate }}\n \n \n
    \n
    \n\n
    \n \n\n
    \n \n \n \n {{ productType.name }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.select_product_first_message' | translate }}\n \n \n
    \n
    \n
    \n\n \n
    \n \n\n
    \n \n \n \n {{ blueprint?.name }} --\n {{ blueprint?.id }}\n \n \n\n \n {{ 'pages.customer_details.select_product_type' | translate }}\n \n \n
    \n
    \n
    \n\n
    \n \n\n
    \n
    \n J\n \n
    \n\n
    \n H\n \n
    \n
    \n
    \n\n
    \n \n\n
    \n
    \n J\n \n
    \n\n
    \n H\n \n
    \n
    \n
    \n\n
    \n \n\n
    \n \n \n \n {{ productPurpose[1] | translate }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n
    \n
    \n\n
    \n \n\n
    \n \n \n\n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n
    \n
    \n\n
    \n \n\n
    \n \n \n \n
    \n
    \n\n
    \n \n\n \n {{ 'pages.generate_ticket_modal.confirm_product' | translate }}\n \n
    \n
    \n
    \n\n \n \n
    \n \n {{ 'pages.generate_ticket_modal.product_infos' | translate }}\n \n\n \n \n\n
    \n \n \n \n {{ 'pages.generate_ticket_modal.dematerialized' | translate }}\n \n \n {{ 'pages.generate_ticket_modal.physical' | translate }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n
    \n \n\n
    \n \n\n
    \n \n \n\n \n \n {{ stop.stop_name }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n
    \n\n \n
    \n\n
    \n \n\n
    \n \n \n\n \n \n {{ stop.stop_name }}\n \n \n\n \n {{ 'pages.generate_ticket_modal.field_required' | translate }}\n \n \n
    \n\n \n
    \n\n
    \n
    \n

    {{ 'pages.generate_ticket_modal.trip_found' | translate }}

    \n
    \n\n
    \n

    \n {{ tripErrorMessage }}\n

    \n
    \n\n
    \n \n
    \n
    \n\n
    \n \n\n
    \n

    \n {{ avatarErrorMessage }}\n

    \n\n \n
    \n
    \n\n
    \n \n\n \n
    \n \n
    \n\n \n \n {{ 'pages.generate_ticket_modal.ticket_generation' | translate }}\n \n
    \n

    {{ 'pages.generate_ticket_modal.ticket_summary' | translate }}

    \n\n

    \n {{ 'pages.generate_ticket_modal.network' | translate }} :\n {{ network?.name }}\n

    \n\n

    \n {{ 'pages.generate_ticket_modal.product' | translate }} :\n {{ product?.name }}\n

    \n\n

    \n {{ 'pages.generate_ticket_modal.product_type' | translate }} :\n \n {{ blueprint?.name }}\n \n {{ blueprint?.id }}\n

    \n\n

    \n {{ 'pages.customer_details.generationreason' | translate }} :\n \n {{ generationReasonLabels[productFormGroup.value.generationReason] | translate }}\n \n

    \n\n

    \n {{ 'pages.generate_ticket_modal.quantity' | translate }} :\n {{ productFormGroup.get('quantity').value }}\n

    \n\n

    \n {{ 'pages.customer_details.ticket_begin_date' | translate }} :\n \n {{ productFormGroup.value.startValidityDate | date }}\n {{ productFormGroup.value.startValidityTime }}\n \n

    \n\n

    \n {{ 'pages.customer_details.ticket_end_date' | translate }} : :\n \n {{ productFormGroup.value.endValidityDate | date }}\n {{ productFormGroup.value.endValidityTime }}\n \n

    \n\n

    \n {{ 'pages.generate_ticket_modal.support' | translate }} :\n \n {{\n productInfoFormGroup.value.dematerialized === '1'\n ? ('pages.generate_ticket_modal.dematerialized' | translate)\n : ('pages.generate_ticket_modal.physical' | translate)\n }}\n \n

    \n\n

    \n {{ 'pages.generate_ticket_modal.origin_destination' | translate }} :\n \n {{ productInfoFormGroup.value.origin?.stop_name }}-{{\n productInfoFormGroup.value.destination?.stop_name\n }}\n ({{ productInfoFormGroup.value.trip?.fareId }})\n \n

    \n\n

    \n {{ 'pages.generate_ticket_modal.photo' | translate }} :\n {{ productInfoFormGroup.value.avatar }}\n

    \n
    \n\n \n

    \n \n {{ 'pages.customer_details.unactive_original_ticket' | translate }}\n \n

    \n

    \n \n {{ 'pages.customer_details.deactivate_ticket_for_security' | translate }}\n
    \n {{ 'pages.customer_details.access_only_new_ticket' | translate }}\n
    \n \n {{ 'pages.customer_details.error_unactive_ticket' | translate }}\n \n \n {{ 'pages.customer_details.old_ticket_inactive' | translate }}\n \n \n {{ 'pages.customer_details.processing' | translate }}\n \n

    \n \n \n\n
    \n \n {{ 'pages.generate_ticket_modal.cancel' | translate }}\n \n\n \n {{ 'pages.generate_ticket_modal.generate_ticket' | translate }}\n \n
    \n
    \n
    \n\n \n
    \n\n","import { Component, EventEmitter, Inject, OnInit, Output } from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl, UntypedFormGroup, Validators } from '@angular/forms';\nimport { MatDialog, MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport { MatStepper } from '@angular/material/stepper';\nimport { Customer } from '@app/modules/customers/models/customer';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { ProductsService } from '@app/modules/customers/services/products.service';\nimport { TicketService } from '@app/modules/customers/services/ticket.service';\nimport { NetworkInfo } from '@app/modules/network/models/network-info';\nimport { NetworkService } from '@app/modules/network/services/network.service';\nimport { Network } from '@app/modules/shared/models/network';\nimport { TicketPurposeLabel } from '@app/modules/shared/models/ticket';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { isMatch as dateIsMatch } from 'date-fns';\nimport { debounceTime, distinctUntilChanged, firstValueFrom } from 'rxjs';\nimport { $$Ticket } from '../../../services/customers.service';\nimport { Blueprint } from '@app/modules/shop/models/blueprint/blueprint';\nimport { Stop } from '@app/modules/customers/models/od';\n\ninterface CustomerNetwork {\n network: Network;\n blueprints: Blueprint[];\n}\n\nexport interface DialogData {\n customer: Customer;\n customerNetworks: CustomerNetwork[];\n version: 'v1' | 'v2';\n transfert: {\n ticket: $$Ticket;\n network_name: string;\n };\n}\n@Component({\n selector: 'tu-ticket-generation-modal',\n templateUrl: './ticket-generation-modal.component.html',\n styleUrls: ['./ticket-generation-modal.component.scss'],\n providers: [CustomersService],\n})\nexport class TicketGenerationModalComponent implements OnInit {\n constructor(\n public modalRef: MatDialogRef,\n public modal: MatDialog,\n public translate: TranslateService,\n private ticketService: TicketService,\n private customersService: CustomersService,\n private productsService: ProductsService,\n private notification: NotificationsService,\n private fb: UntypedFormBuilder,\n private networkService: NetworkService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n //Component Outpouts\n @Output() updateField = new EventEmitter();\n @Output() desactivateTicketEvent = new EventEmitter<$$Ticket>();\n\n public isLoading = false;\n\n //General list\n private networksProducts = [];\n public customerNetworks: CustomerNetwork[] = [];\n public blueprints: Blueprint[] = [];\n\n //Step 1\n public productFormGroup: UntypedFormGroup;\n public currentNetwork: NetworkInfo;\n\n //Step 2\n public productInfoFormGroup: UntypedFormGroup;\n public originStopControl = new UntypedFormControl();\n public destinationStopControl = new UntypedFormControl();\n public loadingOriginStops = false;\n public loadingDestinationStops = false;\n public originStops: Stop[] = [];\n public destinationStops: Stop[] = [];\n public filteredOriginStops = [];\n public filteredDestinationStops = [];\n public isTripFound = false;\n public tripErrorMessage: string | null = null;\n public avatarErrorMessage: string | null = null;\n public desactivateTicketOption: 'WARN' | 'WAIT' | 'KO' | 'OK' = null;\n\n //Transfert\n public productPurposes = Object.entries(TicketPurposeLabel).filter(([index]) => {\n // COMMERCIAL_OFFER is deprecated, but we keep it becasue we need the label for old tickets\n return index !== 'COMMERCIAL_OFFER';\n });\n public generationReasonLabels = TicketPurposeLabel;\n\n //Ticket Generation\n public ticketGenerationInProcess = false;\n public notificationOptions = {\n timeOut: 3000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n get startDate(): Date | null {\n const startValidityDate = this.productFormGroup?.value?.startValidityDate;\n const startValidityTime = this.productFormGroup?.value?.startValidityTime ?? '00:00';\n\n if (startValidityDate) {\n const startStringDate = `${startValidityDate} ${startValidityTime}`;\n\n if (dateIsMatch(startStringDate, 'yyyy-MM-dd HH:mm')) {\n return new Date(startStringDate);\n }\n }\n\n return null;\n }\n\n get endDate(): Date | null {\n const endValidityDate = this.productFormGroup?.value?.endValidityDate;\n const endValidityTime = this.productFormGroup?.value?.endValidityTime ?? '23:59';\n\n if (endValidityDate) {\n const endStringDate = `${endValidityDate} ${endValidityTime}`;\n\n if (dateIsMatch(endStringDate, 'yyyy-MM-dd HH:mm')) {\n return new Date(endStringDate);\n }\n }\n\n return null;\n }\n\n //Component initialisation\n ngOnInit() {\n try {\n this.customerNetworks = this.data.customerNetworks;\n\n //Step 1 form\n this.productFormGroup = this.fb.group({\n networkId: [null, Validators.required],\n generationReason: [null, Validators.required],\n quantity: [null, Validators.required],\n categoryId: [null, this.isTicketV1 ? Validators.required : null],\n productId: [null, this.isTicketV1 ? Validators.required : null],\n productTypeId: [null, this.isTicketV1 ? Validators.required : null],\n blueprint: [null, this.isTicketV2 ? Validators.required : null],\n startValidityDate: [null, this.data.transfert ? Validators.required : null],\n startValidityTime: [null, this.data.transfert ? Validators.required : null],\n endValidityDate: [null, this.data.transfert ? Validators.required : null],\n endValidityTime: [null, this.data.transfert ? Validators.required : null],\n generationComment: [null],\n });\n\n //Step 2 form\n this.productInfoFormGroup = this.fb.group({\n dematerialized: [null],\n origin: [null],\n destination: [null],\n trip: [null],\n avatar: [null],\n });\n\n //Network input changes bind\n this.productFormGroup\n .get('networkId')\n .valueChanges.pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((networkId) => {\n console.log({ networkId });\n\n if (this.isTicketV1 && !this.data.transfert) {\n const defaultCategory = this.categories.length === 1 ? this.categories[0].id : null;\n\n this.productFormGroup.get('categoryId').setValue(defaultCategory);\n }\n\n if (this.data.transfert) {\n this.currentNetwork = null;\n\n firstValueFrom(this.networkService.getNetworksDetails(networkId)).then((network) => {\n this.currentNetwork = network;\n });\n }\n\n if (this.isTicketV2) {\n const network = this.customerNetworks.find(\n (customerNetwork) => customerNetwork.network.id === networkId\n );\n\n console.log(network?.blueprints);\n\n this.blueprints = network?.blueprints || [];\n }\n });\n\n if (this.isTicketV1) {\n //Load Product catalog\n firstValueFrom(this.productsService.getProducts()).then((data) => {\n this.networksProducts = data.sort((a, b) => a.name.localeCompare(b.name));\n this.setDefaultNetwork();\n });\n\n //Category input changes bind\n this.productFormGroup\n .get('categoryId')\n .valueChanges.pipe(debounceTime(200), distinctUntilChanged())\n .subscribe(() => {\n const defaultProduct = this.products.length === 1 ? this.products[0].id : null;\n console.log(\n this.network,\n this.categories,\n this.category,\n this.products,\n defaultProduct\n );\n this.productFormGroup.get('productId').setValue(defaultProduct);\n });\n\n //Product input changes bind\n this.productFormGroup\n .get('productId')\n .valueChanges.pipe(debounceTime(200), distinctUntilChanged())\n .subscribe(() => {\n const defaultProductType =\n this.productTypes.length === 1 ? this.productTypes[0].id : null;\n this.productFormGroup.get('productTypeId').setValue(defaultProductType);\n //Step 2 inputs\n this.tripErrorMessage = null;\n this.avatarErrorMessage = null;\n this.isTripFound = false;\n this.productInfoFormGroup.get('dematerialized').removeValidators(Validators.required);\n this.originStopControl.removeValidators(Validators.required);\n this.destinationStopControl.removeValidators(Validators.required);\n this.productInfoFormGroup.get('origin').removeValidators(Validators.required);\n this.productInfoFormGroup.get('destination').removeValidators(Validators.required);\n this.productInfoFormGroup.get('trip').removeValidators(Validators.required);\n this.productInfoFormGroup.get('avatar').removeValidators(Validators.required);\n this.productInfoFormGroup.reset();\n this.originStopControl.reset();\n this.destinationStopControl.reset();\n\n if (this.product) {\n this.productInfoFormGroup.controls.dematerialized.setValue(\n this.product.supports.includes('PHYSICAL') ? '0' : '1'\n );\n }\n });\n\n //OriginStop input (step2) changes bind\n this.originStopControl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((name) => {\n this.productInfoFormGroup.controls.origin.setValue(null);\n this.destinationStopControl.setValue(null);\n this.productInfoFormGroup.controls.destination.setValue(null);\n\n if (name) {\n this.filteredOriginStops = this.stopsFilter(this.originStops, name);\n\n const stop = this.originStops.find((stop) => {\n return stop.stop_name === name;\n });\n\n if (stop) {\n this.productInfoFormGroup.controls.origin.setValue(stop);\n\n this.loadingDestinationStops = true;\n\n // Retrieve origin stop\n this.ticketService\n .getODStops(this.productFormGroup.value.productTypeId, stop.stop_id)\n .subscribe((stops) => {\n this.destinationStops = stops;\n this.loadingDestinationStops = false;\n });\n }\n } else {\n this.filteredOriginStops = this.originStops;\n }\n });\n\n //DestinationStop input (step2) changes bind\n this.destinationStopControl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((name) => {\n if (name) {\n this.filteredDestinationStops = this.stopsFilter(this.destinationStops, name);\n\n const stop = this.destinationStops.find((stop) => {\n return stop.stop_name === name;\n });\n\n if (stop) {\n this.productInfoFormGroup.controls.destination.setValue(stop);\n } else {\n this.productInfoFormGroup.controls.destination.setValue(null);\n }\n } else {\n this.productInfoFormGroup.controls.destination.setValue(null);\n this.filteredDestinationStops = this.destinationStops;\n }\n });\n\n if (this.data?.transfert) {\n this.productFormGroup\n .get('networkId')\n .setValue(parseInt(this.data.transfert.ticket.network_id, 10));\n\n this.productFormGroup.get('categoryId').setValue(this.data.transfert.ticket.category_id);\n }\n }\n\n if (this.isTicketV2) {\n this.setDefaultNetwork();\n }\n\n if (this.data.transfert) {\n this.productFormGroup.patchValue({\n networkId: this.data.transfert.ticket.network_id,\n categoryId: this.data.transfert.ticket.category_id,\n productId: this.data.transfert.ticket.product_id,\n });\n\n this.desactivateTicketOption = 'WARN';\n } else {\n //DELETE Device Change for ticket creation\n this.productPurposes = this.productPurposes.filter(\n (productPurpose) => productPurpose[0] !== 'DEVICE_CHANGE'\n );\n }\n } catch (error) {\n console.log(error);\n } finally {\n this.isLoading = false;\n }\n }\n\n //Ticket generation\n public generateTicket() {\n const { productId, productTypeId, generationReason, quantity, blueprint, generationComment } =\n this.productFormGroup.value;\n\n this.ticketGenerationInProcess = true;\n //Ticket V1 GENERATION\n if (this.isTicketV1) {\n const {\n origin,\n destination,\n trip,\n avatar,\n dematerialized = 1,\n } = this.productInfoFormGroup.value;\n\n const odData = this.isProductOD\n ? {\n origin,\n destination,\n trip,\n }\n : null;\n\n const isTicketDematerialized =\n this.productType.dematerialized === null ? dematerialized : this.productType.dematerialized;\n\n this.ticketService\n .generateTicketsV1(\n this.data.customer.customer_id,\n productId,\n productTypeId,\n quantity,\n avatar,\n // HACK: This is a hack because in some obscure cases, the `isTicketDematerialized` is\n // valued to `null` which doesn't work for the feed.\n // We need to investigate why this variable isn't valued to '1' or '0'...\n isTicketDematerialized === null ? '1' : isTicketDematerialized,\n odData,\n this.startDate ? this.startDate.toISOString() : null,\n this.endDate ? this.endDate.toISOString() : null,\n generationReason,\n generationComment\n )\n .subscribe((response) => {\n this.ticketGenerationInProcess = false;\n if (response.length > 0) {\n this.updateFieldTrigger();\n this.notification.success(this.translate.instant(`otherslabels.notif_create_ok`));\n this.closeModal();\n return;\n }\n //ERROR ON GENERATION\n this.notification.error(this.translate.instant(`otherslabels.notif_create_ko`));\n });\n return;\n }\n\n //Ticket V2 GENERATION\n let contracts = [];\n for (let i = 0; i < quantity; i++) {\n this.ticketService\n .generateTicketsV2(\n generationReason,\n blueprint,\n this.data.customer.customer_id,\n this.network.id,\n this.startDate ? this.startDate.toISOString() : null,\n this.endDate ? this.endDate.toISOString() : null,\n generationComment\n )\n .subscribe((response) => {\n contracts.push(response);\n if (contracts.length === quantity) {\n this.ticketGenerationInProcess = false;\n this.updateFieldTrigger();\n const nbError = contracts.length - contracts.filter(Boolean).length;\n if (nbError === 0) {\n this.closeModal();\n this.notification.success(this.translate.instant(`otherslabels.notif_create_ok`));\n return;\n }\n //PARTIAL ERROR ON GENERATION\n if (nbError < contracts.length) {\n this.closeModal();\n this.notification.warn(\n this.translate.instant('pages.customer_details.error_add_contracts', {\n nb: nbError,\n total: quantity,\n })\n );\n return;\n }\n //FULL ERROR ON GENERATION\n this.notification.error(this.translate.instant(`otherslabels.notif_create_ko`));\n }\n });\n }\n }\n\n public get isTicketV1() {\n return this.data.version === 'v1';\n }\n\n public get isTicketV2() {\n return this.data.version === 'v2';\n }\n\n get network() {\n const networkId = this.productFormGroup.get('networkId').value;\n if (!networkId) return null;\n\n const network = this.customerNetworks.find(\n (customerNetwork) => customerNetwork.network.id === networkId\n );\n\n return network?.network || null;\n }\n\n get categories() {\n if (!this.network) return [];\n\n const network = this.networksProducts.find((network) => network.id === String(this.network.id));\n\n return network?.categories ?? [];\n }\n\n get category() {\n const categoryId = this.productFormGroup.get('categoryId').value;\n\n if (this.categories.length === 0 || !categoryId) return null;\n\n return this.categories.find((category) => `${category.id}` === `${categoryId}`);\n }\n\n get products() {\n if (!this.category) return [];\n\n return this.category.items;\n }\n\n get product() {\n const productId = this.productFormGroup.get('productId').value;\n if (!productId) return null;\n\n return this.products.find((product) => `${product.id}` === `${productId}`);\n }\n\n get blueprint() {\n const blueprintId = this.productFormGroup.get('blueprint').value;\n if (!blueprintId) return null;\n\n return this.blueprints.find((blueprint) => `${blueprint.id}` === `${blueprintId}`);\n }\n\n get productTypes() {\n if (!this.product) return [];\n\n return this.product.products;\n }\n\n get productType() {\n const productTypeId = this.productFormGroup.get('productTypeId').value;\n if (!productTypeId) return null;\n\n return this.productTypes.find((productType) => `${productType.id}` === `${productTypeId}`);\n }\n\n get hasSupportChoice() {\n if (!this.product) return false;\n\n this.productInfoFormGroup.get('dematerialized').setValidators(Validators.required);\n return this.product.supports.length > 1;\n }\n\n get isProductOD(): boolean {\n if (!this.product) return false;\n\n return this.product.is_od;\n }\n\n get isPhotoMandatory(): boolean {\n if (!this.productFormGroup.get('productId').value) return false;\n\n return this.product?.isPhotoMandatory === true;\n }\n\n get needMoreInfos() {\n if (this.product) {\n const hasMultiSupport = ['PHYSICAL', 'DEMATERIALIZED'].every((support) => {\n return this.product.supports.includes(support);\n });\n\n return this.isProductOD || this.isPhotoMandatory || hasMultiSupport;\n }\n return this.isProductOD || this.isPhotoMandatory;\n }\n\n public getProductOptions(stepper: MatStepper) {\n if (this.startDate !== null && this.endDate !== null && this.startDate > this.endDate) {\n return this.notification.error(\n this.translate.instant('pages.customer_details.start_date_invalid')\n );\n }\n\n if (!this.needMoreInfos) {\n return stepper.next();\n }\n\n if (this.isProductOD) {\n this.originStopControl.setValidators(Validators.required);\n this.destinationStopControl.setValidators(Validators.required);\n this.productInfoFormGroup.controls.origin.setValidators(Validators.required);\n this.productInfoFormGroup.controls.destination.setValidators(Validators.required);\n this.productInfoFormGroup.controls.trip.setValidators(Validators.required);\n\n this.loadingOriginStops = true;\n\n // Retrieve origin stop\n this.ticketService\n .getODStops(this.productFormGroup.value.productTypeId)\n .subscribe((stops) => {\n this.originStops = stops;\n this.loadingOriginStops = false;\n });\n }\n\n if (this.isPhotoMandatory) {\n this.productInfoFormGroup.controls.avatar.setValidators(Validators.required);\n this.productInfoFormGroup.controls.avatar.setValue(this.data.customer.picture);\n }\n\n stepper.next();\n }\n\n public onUpdateAvatar(customer) {\n this.data.customer = customer;\n this.productInfoFormGroup.controls.avatar.setValue(customer.picture);\n this.avatarErrorMessage = null;\n }\n\n public searchTrip() {\n this.isTripFound = false;\n this.tripErrorMessage = null;\n this.avatarErrorMessage = null;\n\n const { origin, destination } = this.productInfoFormGroup.value;\n\n if (!origin || !destination) {\n this.tripErrorMessage = this.translate.instant(\n 'pages.generate_ticket_modal.select_origin_destination'\n );\n\n return;\n }\n\n this.ticketService\n .getODTrips(this.product.id, origin.stop_id, destination.stop_id)\n .subscribe((trips) => {\n if (!trips.length) {\n this.tripErrorMessage = this.translate.instant(\n 'pages.generate_ticket_modal.trip_not_found'\n );\n return;\n }\n\n const trip = trips[0];\n\n this.productInfoFormGroup.controls.trip.setValue({\n agency: trip.agency,\n fareId: trip.fareId,\n price: trip.price,\n });\n\n this.isTripFound = true;\n });\n }\n\n public closeModal(): void {\n this.modalRef.close();\n }\n\n public goForward(stepper: MatStepper): void {\n this.avatarErrorMessage = null;\n this.tripErrorMessage = null;\n\n if (this.isProductOD && !this.productInfoFormGroup.value.trip) {\n this.tripErrorMessage = this.translate.instant(\n 'pages.generate_ticket_modal.search_trip_message'\n );\n return;\n }\n\n if (this.isPhotoMandatory && !this.productInfoFormGroup.value.avatar) {\n this.avatarErrorMessage = this.translate.instant(\n 'pages.generate_ticket_modal.download_photo_message'\n );\n return;\n }\n\n stepper.next();\n }\n\n private stopsFilter(stops: Stop[], value: string) {\n const filterValue = value.toLowerCase();\n\n return stops.filter((stop) => stop.stop_name.toLowerCase().includes(filterValue));\n }\n\n public desactivateTicket = () => {\n this.desactivateTicketOption = 'WAIT';\n this.desactivateTicketEvent.emit(this.data.transfert.ticket);\n };\n\n private setDefaultNetwork() {\n if (this.customerNetworks.length === 1) {\n this.productFormGroup.controls.networkId.setValue(this.customerNetworks[0].network.id);\n }\n }\n\n private async updateFieldTrigger() {\n const customer = await this.customersService.getCustomerDetails(this.data.customer.customer_id);\n const profiles = await this.customersService.$$getProfiles(this.data.customer.customer_id);\n\n this.updateField.emit({\n customer,\n profiles,\n });\n }\n}\n","
    {{ 'pages.customer_details.fusion_modal_title' | translate }}
    \n
    \n
    \n
    \n \n \n \n {{ 'pages.customer_details.fusion_modal_search' | translate }}\n \n

    \n {{ 'pages.customer_details.fusion_modal_base_user' | translate }}\n \n {{ customer.customer_id }} -- {{ customer.firstname }} {{ customer.lastname }}\n \n

    \n
    \n \n
    \n \n \n \n {{ customer.id }} -- {{ customer.firstName }} {{ customer.lastName }}\n \n \n
    \n
    \n \n \n {{ customer.id }} -- {{ customer.firstName }} {{ customer.lastName }}\n \n \n \n\n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n {{ 'pages.customer_details.fusion_modal_next_step' | translate }}\n \n
    \n
    \n \n \n {{ 'pages.customer_details.fusion_modal_data_choice' | translate }}\n \n

    \n {{ recapLabel }}\n
    \n \n {{\n 'pages.customer_details.fusion_modal_users_delete' | translate : { customersLabel }\n }}\n \n
    \n {{ 'pages.customer_details.fusion_modal_confirm_fusion' | translate }}\n

    \n

    {{ 'pages.customer_details.fusion_modal_choices' | translate }}

    \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 !loading\n ? ('otherslabels.btn_save' | translate)\n : ('otherslabels.loading' | translate)\n }}\n \n \n
    \n
    \n
    \n
    \n
    \n \n
    \n","import { Component, Inject, OnInit } from '@angular/core';\nimport { UntypedFormControl } from '@angular/forms';\nimport { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';\nimport { MatDialogRef, MAT_DIALOG_DATA } from '@angular/material/dialog';\nimport {\n CustomerFusion,\n Customer,\n CustomerFusionConfiguration,\n Attribute,\n} from '@app/modules/customers/models/customer';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService, Notification } from 'angular2-notifications';\nimport { debounceTime, distinctUntilChanged, firstValueFrom } from 'rxjs';\nimport { CustomersService, $$GlobalAttribute } from '../../../services/customers.service';\nimport { Router } from '@angular/router';\n\nexport interface DialogData {\n customer: Customer;\n attributes: {\n global: $$GlobalAttribute[];\n customer: Attribute[];\n };\n}\n@Component({\n selector: 'tu-fusion-modal',\n templateUrl: './fusion-modal.component.html',\n})\nexport class FusionModalComponent implements OnInit {\n public customer: Customer = null;\n public attributes: {\n global: $$GlobalAttribute[];\n customer: Attribute[];\n } = null;\n\n public search = new UntypedFormControl();\n public fusionEmail = new UntypedFormControl();\n public fusionAddress = new UntypedFormControl();\n public fusionPhone = new UntypedFormControl();\n public fusionBirthdate = new UntypedFormControl();\n public fusionAttributes: { [key: string]: UntypedFormControl }[] = [];\n\n public filteredCustomers: CustomerFusion[] = [];\n public selectedCustomers: CustomerFusion[] = [];\n\n public customerEmails: { label: string; value: any }[] = [];\n public customerAddresses: { label: string; value: any }[] = [];\n public customerPhones: { label: string; value: any }[] = [];\n public customerBirthdates: { label: string; value: any }[] = [];\n public customerAttributes: {\n key: string;\n id: string;\n label: string;\n values: { value: number; label: any }[];\n }[] = [];\n\n public loading = false;\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n public get canGoNextStep() {\n return this.selectedCustomers.length > 0;\n }\n\n public get customersLabel() {\n let label = [];\n for (const customer of this.selectedCustomers) {\n label.push(`${customer.id} (${customer.firstName || ''} ${customer.lastName || ''})`);\n }\n return label.join(', ');\n }\n\n public get recapLabel() {\n return this.translate.instant('pages.customer_details.fusion_modal_users_recap', {\n customersLabel: this.customersLabel,\n id: this.customer.customer_id,\n customerLabel: `${this.customer.firstname} ${this.customer.lastname}`,\n });\n }\n\n constructor(\n public modalRef: MatDialogRef,\n private customersService: CustomersService,\n private translateService: TranslateService,\n private notification: NotificationsService,\n private translate: TranslateService,\n private router: Router,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {\n this.customer = data.customer;\n this.attributes = data.attributes;\n }\n\n ngOnInit() {\n this.search.valueChanges.pipe(debounceTime(400), distinctUntilChanged()).subscribe((search) => {\n this.filteredCustomers = [];\n if (!search) return;\n\n this.customersService\n .getFusionList(this.data.customer.customer_id, search)\n .then(({ response }) => {\n if (response.errorMessage) {\n throw new Error(response.errorMessage);\n }\n\n this.filteredCustomers = response.customers.filter((customer) => {\n return (\n this.selectedCustomers.filter(\n (selectedCustomer) => selectedCustomer.id === customer.id\n ).length === 0\n );\n });\n\n if (this.filteredCustomers.length === 0) {\n this.notification.warn(\n this.translate.instant('pages.customer_details.fusion_modal_no_customer_search'),\n search\n );\n }\n })\n .catch((reason) => {\n this.filteredCustomers = [];\n const error = reason?.error?.message || reason;\n this.notification.error(\n this.translate.instant('pages.customer_details.fusion_modal_error_on_search'),\n error\n );\n console.error(reason);\n });\n });\n }\n\n public closeModal() {\n this.modalRef.close();\n }\n\n public selectedCustomer(event: MatAutocompleteSelectedEvent): void {\n this.selectedCustomers.push(event.option.value);\n this.search.setValue(null);\n }\n\n public addCustomer(customer: CustomerFusion) {\n const customerMatching = this.selectedCustomers.find((customerA) => {\n customerA.id === customer.id;\n });\n if (!customerMatching) {\n return;\n }\n this.selectedCustomers.push(customer);\n this.search.setValue(null);\n }\n\n public removeCustomer(customer: CustomerFusion) {\n this.selectedCustomers = this.selectedCustomers.filter(\n (customerA) => customerA.id !== customer.id\n );\n }\n\n public generateFusionLists() {\n //Order selectedCustomer by last updatedDate DESC\n this.selectedCustomers.sort((customerA, customerB) => {\n return customerA.updatedAt < customerB.updatedAt ? 1 : -1;\n });\n //Generate lists\n this.customerEmails = this._generateList('email');\n this.customerAddresses = this._generateList('address');\n this.customerPhones = this._generateList('phone');\n this.customerBirthdates = this._generateList('birthdate');\n this.customerAttributes = this._generateAttributesList();\n this._updateAttributesFormControl();\n\n //Default value on select\n this.fusionEmail.setValue(this.customerEmails[0]?.value || null);\n this.fusionAddress.setValue(this.customerAddresses[0]?.value || null);\n this.fusionPhone.setValue(this.customerPhones[0]?.value || null);\n this.fusionBirthdate.setValue(this.customerBirthdates[0]?.value || null);\n }\n\n private _generateList(listSlug: 'email' | 'address' | 'phone' | 'birthdate') {\n type CustomerInfoList = { id: number; value: any };\n const customerInfos: CustomerInfoList[] = [];\n\n //Current customer value\n const currentCustomerData: CustomerInfoList = {\n id: parseInt(this.customer.customer_id, 10),\n value: null,\n };\n switch (listSlug) {\n case 'birthdate':\n currentCustomerData.value = this.customer.birthday;\n break;\n default:\n currentCustomerData.value = this.customer[listSlug];\n }\n\n if (currentCustomerData.value) {\n customerInfos.push(currentCustomerData);\n }\n //Fusion customers values\n for (const customer of this.selectedCustomers) {\n if (!customer[listSlug]) continue;\n const customerData: CustomerInfoList = { id: customer.id, value: customer[listSlug] };\n const valueMatching = customerInfos.find((customer) => customer.value === customerData.value);\n if (customerData.value && !valueMatching) {\n customerInfos.push(customerData);\n }\n }\n\n //Transform into list object (label, value)\n const options = customerInfos\n .map((customerData) => {\n let label = customerData.value;\n if (!label) return;\n switch (listSlug) {\n case 'address':\n //Current customer address\n if (label?.country !== undefined) {\n label = [label.streetNumber, label.route, label.zipCode, label.city, label.country]\n .map((string) => string || '')\n .join(' ')\n .trim();\n }\n //Fusion customer address\n else {\n label = [label.lines[0], label.zip, label.city, label.state]\n .map((string) => string || '')\n .join(' ')\n .trim();\n }\n if (label.length === 0) {\n return;\n }\n break;\n case 'birthdate':\n const birthday = new Date(label);\n label = birthday.toLocaleDateString();\n break;\n default:\n break;\n }\n\n return {\n value: customerData.id,\n label,\n };\n })\n .filter(Boolean);\n if (options.length === 0)\n return [\n {\n value: this.customer.customer_id,\n label: this.translate.instant('pages.customer_details.fusion_modal_no_value'),\n },\n ];\n return options;\n }\n\n public async fusionCustomers() {\n this.loading = true;\n const customerId = parseInt(this.customer.customer_id, 10);\n\n let notification: Notification = null;\n let redirect = true;\n for (const fusionCustomer of this.selectedCustomers) {\n let fusionConfiguration: CustomerFusionConfiguration = {\n email: parseInt(this.fusionEmail.value, 10) === fusionCustomer.id ? 'FORMER' : 'RECIPIENT',\n address:\n parseInt(this.fusionAddress.value, 10) === fusionCustomer.id ? 'FORMER' : 'RECIPIENT',\n phone: parseInt(this.fusionPhone.value, 10) === fusionCustomer.id ? 'FORMER' : 'RECIPIENT',\n birthdate:\n parseInt(this.fusionBirthdate.value, 10) === fusionCustomer.id ? 'FORMER' : 'RECIPIENT',\n };\n if (this.fusionAttributes.length) {\n let attributeValues: Record = {};\n for (const attributeId in this.fusionAttributes) {\n const value = this.fusionAttributes[attributeId].value as unknown as string; //FormControl error on def Oo\n attributeValues = {\n ...attributeValues,\n [parseInt(attributeId, 10)]:\n parseInt(value, 10) === fusionCustomer.id ? 'FORMER' : 'RECIPIENT',\n };\n }\n fusionConfiguration = {\n ...fusionConfiguration,\n attributeValues,\n };\n }\n\n await this.customersService\n .fusionCustomers(customerId, fusionCustomer.id, fusionConfiguration)\n .then((success) => {\n if (success) {\n notification = this.notification.success(\n this.translate.instant('pages.customer_details.fusion_modal_user_ok', {\n id: fusionCustomer.id,\n })\n );\n } else {\n notification = this.notification.error(\n this.translate.instant('pages.customer_details.fusion_modal_user_ko', {\n id: fusionCustomer.id,\n })\n );\n redirect = false;\n }\n })\n .catch((error) => {\n notification = this.notification.error(\n this.translate.instant('pages.customer_details.fusion_modal_user_ko', {\n id: fusionCustomer.id,\n })\n );\n console.error('Fusion error', error);\n redirect = false;\n });\n }\n\n await firstValueFrom(notification.timeoutEnd);\n this.loading = false;\n\n if (redirect) {\n this.modalRef.close();\n this.router.navigate(['/customers/', this.customer.customer_id]);\n }\n }\n\n private _generateAttributesList() {\n const attributesValues: {\n key: string;\n id: string;\n label: string;\n values: { value: number; label: any }[];\n }[] = [];\n\n //Global / Customer attributes\n for (const attribute of this.attributes.global) {\n const id = attribute.id;\n const key = attribute.key;\n const customerValue = this.attributes.customer.find(\n (customerAttribute) => customerAttribute.key === key\n );\n attributesValues.push({\n id,\n key,\n label: attribute.label,\n values: [\n {\n value: parseInt(this.customer.customer_id, 10),\n label: customerValue?.value || null,\n },\n ],\n });\n }\n\n //Fusionned customer attributes\n for (const customer of this.selectedCustomers) {\n for (const attributes of customer.attributeValues) {\n const attributeValue = attributesValues.find(\n (attributesValue) => attributesValue.key === attributes.attributeKey\n );\n if (attributeValue) {\n attributeValue.values.push({\n value: customer.id,\n label: attributes.value,\n });\n attributeValue.values = attributeValue.values.filter(Boolean);\n }\n }\n }\n\n const attributesOptions = attributesValues.map((attribute) => {\n if (attribute.values.length !== 1 || attribute.values[0].label) return attribute;\n return {\n ...attribute,\n values: [\n {\n value: attribute.values[0].value,\n label: this.translate.instant('pages.customer_details.fusion_modal_no_value'),\n },\n ],\n };\n });\n\n return attributesOptions;\n }\n\n private _updateAttributesFormControl() {\n this.fusionAttributes = [];\n for (const attribute of this.customerAttributes) {\n const attributeFormControl = new UntypedFormControl(attribute.values[0].value);\n this.fusionAttributes[attribute.id] = attributeFormControl;\n }\n }\n}\n","\n
    {{ 'pages.customer_details.delete_ticket_title' | translate: { name: data.label } }}
    \n\n
    \n
    \n
    \n \n \n
    \n
    \n
    \n\n
    \n \n \n
    ","import { Component, Inject } from '@angular/core';\nimport { UntypedFormBuilder } from '@angular/forms';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\n\nexport interface DeleteTicketModalResponse {\n delete: boolean;\n deleteLinkedSubscription?: boolean;\n}\n\n@Component({\n selector: 'tu-delete-ticket-modal',\n templateUrl: './delete-ticket-modal.component.html',\n styleUrls: ['./delete-ticket-modal.component.scss'],\n})\nexport class DeleteTicketModalComponent {\n constructor(\n private fb: UntypedFormBuilder,\n private dialogRef: MatDialogRef,\n @Inject(MAT_DIALOG_DATA) public data: { label: string; showSubscriptionCheckbox: boolean }\n ) {}\n\n public form = this.fb.group({\n deleteLinkedSubscription: [false],\n });\n\n public close(): void {\n this.dialogRef.close({\n delete: false,\n });\n }\n\n public deleteTicket(): void {\n const deleteLinkedSubscription = this.form.value.deleteLinkedSubscription;\n\n this.dialogRef.close({\n delete: true,\n deleteLinkedSubscription,\n });\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { Observable, firstValueFrom, map } from 'rxjs';\nimport { environment as env } from '@env/environment';\n\n@Injectable({\n providedIn: 'root',\n})\nexport class PaperworkService {\n constructor(private http: HttpClient) {}\n\n public updateDocument(\n documentId: number | string,\n params?: {\n status?: string;\n startingAt?: string;\n expiresAt?: string;\n comment?: string;\n examined?: string;\n }\n ): Observable {\n return this.http\n .patch>(\n env.config.feedRoot + `Paperwork/updateDocument`,\n {\n documentId,\n ...params,\n }\n )\n .pipe(map(({ response }) => response.document));\n }\n\n public createDocument(customerId: number, typeId: string, file: File, status?: string) {\n const formData = new FormData();\n formData.append('customerId', `${customerId}`);\n formData.append('typeId', typeId);\n formData.append('file', file);\n formData.append('status', status);\n\n return firstValueFrom(\n this.http\n .post>(\n env.config.feedRoot + `Paperwork/createDocument`,\n formData\n )\n .pipe(map(({ response }) => response.document))\n );\n }\n}\n","
    \n
    {{ 'pages.document_upload_modal.title' | translate }}
    \n \n
    \n \n
    \n \n\n \n \n\n \n \n {{ net.name }}\n \n \n
    \n
    \n \n \n
    \n {{ 'pages.document_upload_modal.no_types' | translate }}\n
    \n \n
    \n \n\n \n \n\n \n \n {{ type.name }}\n \n \n
    \n

    {{ 'pages.document_upload_modal.type_info' | translate }}

    \n \n \n \n \n
    \n \n\n \n
    \n \n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n \n {{ 'otherslabels.btn_add' | translate }}\n \n
    \n \n\n\n","import { Component, Inject, OnInit } from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl, Validators } from '@angular/forms';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\nimport { PaperworkService } from '@app/modules/customers/services/paperwork.service';\nimport { NetworkService } from '@app/modules/network/services/network.service';\nimport { IDocument } from '@app/modules/orders/models/document';\nimport { Network } from '@app/modules/shared/models/network';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { debounceTime, distinctUntilChanged, firstValueFrom } from 'rxjs';\n\ninterface DialogData {\n customerId: number;\n}\n\n@Component({\n selector: 'tu-document-upload-modal',\n templateUrl: './document-upload-modal.component.html',\n styleUrls: ['./document-upload-modal.component.scss'],\n})\nexport class DocumentUploadModalComponent implements OnInit {\n public networkNameCtrl = new UntypedFormControl();\n public filteredNetworkList: Network[] = [];\n\n public documentTypeNameCtrl = new UntypedFormControl();\n public documentTypeLoader: boolean = false;\n private allDocTypeList: IDocument[] = [];\n public filteredDocTypeList: IDocument[] = [];\n\n public loader: boolean = false;\n\n public notificationOptions = {\n timeOut: 3000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n public form = this.fb.group({\n networkId: [null, Validators.required],\n typeId: [null, Validators.required],\n document: [null, Validators.required],\n });\n\n constructor(\n private modalRef: MatDialogRef,\n private fb: UntypedFormBuilder,\n private authService: AuthService,\n private networkService: NetworkService,\n private paperworkService: PaperworkService,\n private notification: NotificationsService,\n private translate: TranslateService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n get hasSelectNetwork(): boolean {\n return Boolean(this.form.controls.networkId.value);\n }\n\n get networkHasNoDocumentType(): boolean {\n return !this.documentTypeLoader && !this.documentTypeNameCtrl.disabled && this.filteredDocTypeList.length === 0\n }\n \n public ngOnInit(): void {\n this.documentTypeNameCtrl.disable();\n\n // If user has only one network, we select it\n if (this.authService.networks.length === 1) {\n this.networkNameCtrl.setValue(this.authService.networks[0].name);\n this.form.controls.networkId.setValue(this.authService.networks[0].id);\n this.networkNameCtrl.disable();\n this.retrieveDocumentTypeList(+this.authService.networks[0].id);\n }\n\n this.networkNameCtrl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((name) => {\n if (name) {\n this.filteredNetworkList = this.networksFilter(name);\n\n const network = this.authService.networks.find(\n ({ name: networkName }) => networkName === name\n );\n\n if (network) {\n this.form.controls.networkId.setValue(network.id);\n\n this.retrieveDocumentTypeList(+network.id);\n }\n } else {\n this.filteredNetworkList = this.authService.networks;\n }\n });\n\n this.documentTypeNameCtrl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((doc: IDocument | string | null) => {\n if (typeof doc === 'string') {\n this.filteredDocTypeList = this.docTypesFilter(doc);\n } else if (doc) {\n this.form.controls.typeId.setValue(doc.id);\n }\n });\n }\n\n private networksFilter(value: string): Network[] {\n const filterValue = value.toLowerCase();\n\n return this.authService.networks.filter((network) =>\n network.name.toLowerCase().includes(filterValue)\n );\n }\n\n private docTypesFilter(value: string): IDocument[] {\n const filterValue = value.toLowerCase();\n\n return this.allDocTypeList.filter((docType) =>\n docType.name.toLowerCase().includes(filterValue)\n );\n }\n\n public async submit(): Promise {\n const { typeId, document } = this.form.value;\n\n try {\n this.loader = true;\n\n await this.paperworkService.createDocument(this.data.customerId, typeId, document, 'VALID');\n\n this.notification.success(this.translate.instant('pages.document_upload_modal.success'));\n this.modalRef.close({ success: true });\n } catch {\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n } finally {\n this.loader = false;\n }\n }\n\n public displayDocumentType(documentType: IDocument | string | null): string {\n if (typeof documentType === 'string') {\n return documentType;\n } else if (documentType) {\n return documentType.name;\n }\n }\n\n public onFileSelected(event: Event): void {\n const target = event.target as HTMLInputElement;\n\n if (target.files.length > 0) {\n const file = target.files[0];\n this.form.controls.document.setValue(file);\n } else {\n this.form.controls.document.setValue(null);\n }\n }\n\n public async retrieveDocumentTypeList(networkId: number): Promise {\n try {\n this.documentTypeLoader = true;\n\n const docTypeList = await firstValueFrom(this.networkService.getDocuments(networkId));\n\n this.allDocTypeList = docTypeList;\n this.filteredDocTypeList = docTypeList;\n this.documentTypeNameCtrl.enable();\n } catch {\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n } finally {\n this.documentTypeLoader = false;\n }\n }\n\n public closeModal() {\n this.modalRef.close({ success: false });\n }\n}\n","\n
    \n {{ 'pages.create_firebase_account_modal.info' | translate }}\n
    \n \n
    \n \n \n
    \n \n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n \n \n {{ 'otherslabels.btn_save' | translate }}\n \n \n
    \n\n\n","import { Component, Inject } from '@angular/core';\nimport { FormBuilder, Validators } from '@angular/forms';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\n\ninterface DialogData {\n customerId: string;\n}\n\n@Component({\n selector: 'tu-create-firebase-account-modal',\n templateUrl: './create-firebase-account-modal.component.html',\n styleUrls: ['./create-firebase-account-modal.component.scss'],\n})\nexport class CreateFirebaseAccountModalComponent {\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n public loading: boolean = false;\n\n public form = this.fb.group({\n email: [null, [Validators.required, Validators.email]],\n });\n\n constructor(\n private fb: FormBuilder,\n private modalRef: MatDialogRef,\n private notification: NotificationsService,\n private translate: TranslateService,\n private customerService: CustomersService,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n public close(needRefresh: boolean = false): void {\n this.modalRef.close({ needRefresh });\n }\n\n public async submit(): Promise {\n if (this.form.invalid) {\n this.notification.error(this.translate.instant('otherslabels.error_required_fields'));\n return;\n }\n\n try {\n this.loading = true;\n\n const { email } = this.form.value;\n\n await this.customerService.sendResetPasswordEmail(this.data.customerId, email);\n\n this.notification.success(\n this.translate.instant('pages.create_firebase_account_modal.success_message'),\n this.translate.instant('pages.create_firebase_account_modal.success_message_description')\n );\n this.close(true);\n } catch (error) {\n if (error.error?.code === 'CUSTOMER_RESET_PASSWORD_FIREBASE_ACCOUNT_EXISTS') {\n this.notification.error(\n this.translate.instant('pages.customer_details.email_already_exist')\n );\n } else {\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n }\n } finally {\n this.loading = false;\n }\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { firstValueFrom, map } from 'rxjs';\nimport { environment as env } from '@env/environment';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { Category, ProductDetails, ProductItem, ProductTrip, TripStop } from '../models/shop';\n\n@Injectable()\nexport class ShopService {\n constructor(private http: HttpClient) {}\n\n public countNetworksProductTypes(networks: number[]): Promise {\n return firstValueFrom(\n this.http\n .get(env.config.feedRoot + 'Shop/getProductsType', {\n params: {\n pagination: JSON.stringify({ offset: 0, limit: 1 }),\n network_id: networks.join(','),\n },\n observe: 'response',\n })\n .pipe(\n map((response) => {\n return Number(response.headers.get('x-airweb-pagination-count'));\n })\n )\n );\n }\n\n public async getNetworkCategories(networkId: string): Promise {\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + 'Shop/getCategories',\n {\n params: {\n network_id: networkId,\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.categories.filter((category) => {\n return category.active;\n });\n })\n )\n );\n }\n\n public async getNetworkProducts(networkId: string): Promise {\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + 'Shop/getProductsItem',\n {\n params: {\n network_id: networkId,\n },\n }\n )\n .pipe(\n map(({ response }) => {\n return response.productsItem.filter((product) => {\n return product.active;\n });\n })\n )\n );\n }\n\n public async getProduct(productId: number): Promise {\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + 'Shop/getProductsItem',\n {\n params: {\n id: productId,\n },\n }\n )\n .pipe(\n map(({ response }) => {\n if (response.productsItem.length === 1) {\n return response.productsItem[0];\n } else {\n return null;\n }\n })\n )\n );\n }\n\n public async productTripStops(productId: number, originStop?: string): Promise {\n const params: {\n productId: number;\n originStop?: string;\n } = {\n productId,\n };\n\n if (originStop) {\n params.originStop = originStop;\n }\n\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + 'Shop/productTripStops',\n {\n params,\n }\n )\n .pipe(\n map(({ response }) => {\n return response.stops;\n })\n )\n );\n }\n\n public async productTrips(\n productId: number,\n originStop: string,\n destinationStop: string,\n viaKey?: string,\n timestamp?: string\n ): Promise {\n const params: {\n productId: number;\n originStop: string;\n destinationStop: string;\n viaKey?: string;\n timestamp?: string;\n } = {\n productId,\n originStop,\n destinationStop,\n };\n\n if (viaKey) {\n params.viaKey = viaKey;\n }\n\n if (timestamp) {\n params.timestamp = timestamp;\n }\n\n return firstValueFrom(\n this.http\n .get>(\n env.config.feedRoot + 'Shop/productTrips',\n {\n params,\n }\n )\n .pipe(\n map(({ response }) => {\n return response.trips[0];\n })\n )\n );\n }\n}\n","import { HttpClient } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { GenericResponse } from '@app/modules/shared/models/generic-response';\nimport { environment as env } from '@env/environment';\nimport { firstValueFrom, map } from 'rxjs';\n\n@Injectable({\n providedIn: 'root',\n})\nexport class FieldService {\n constructor(private http: HttpClient) {}\n\n public getFieldsItem(productId: number): Promise {\n return firstValueFrom(\n this.http\n .get>(env.config.feedRoot + 'Field/getFieldsItem', {\n params: {\n item_id: productId,\n },\n })\n .pipe(\n map(({ response }) => {\n return response.fields;\n })\n )\n );\n }\n}\n","\n

    {{ 'pages.credit_product_modal.title' | translate }}

    \n\n
    \n \n \n
    \n \n\n \n \n {{ network.name }}\n \n \n
    \n
    \n \n
    \n \n \n
    \n \n\n \n \n {{ category.cname }}\n \n \n
    \n
    \n \n
    \n \n \n
    \n \n\n \n \n {{ product.name }}\n \n \n
    \n
    \n \n \n {{ 'pages.credit_product_modal.complementary_fields_alert' | translate }}\n \n \n \n {{ 'pages.credit_product_modal.documents_alert' | translate }}\n \n \n \n
    \n \n \n
    \n \n \n \n \n {{ trip.stop_name }}\n \n \n
    \n
    \n \n
    \n \n \n
    \n \n \n \n \n {{ trip.stop_name }}\n \n \n
    \n
    \n \n
    1 && tripsHaveVia\" class=\"tw-flex tw-flex-col tw-gap-2\">\n \n \n
    \n \n \n \n \n {{ trip.viaLabel }}\n \n \n
    \n
    \n \n \n
    \n \n \n \n \n {{ support.label | translate }}\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 {{ generationReason.label | translate }}\n \n \n
    \n \n
    \n \n \n \n
    \n\n
    \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n \n \n {{ 'pages.credit_product_modal.validate' | translate }}\n \n
    \n\n\n","import { Component, Inject, OnInit } from '@angular/core';\nimport { FormBuilder, FormControl, Validators } from '@angular/forms';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\nimport { Document } from '@app/modules/customers/models/customer';\nimport {\n Category,\n ProductDetails,\n ProductItem,\n ProductTrip,\n TripStop,\n} from '@app/modules/customers/models/shop';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { FieldService } from '@app/modules/customers/services/field.service';\nimport { ShopService } from '@app/modules/customers/services/shop.service';\nimport { Network } from '@app/modules/shared/models/network';\nimport { TicketPurposeLabel } from '@app/modules/shared/models/ticket';\nimport { ProductSupportsLabels } from '@app/modules/shop/models/product-item';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { debounceTime, distinctUntilChanged, merge } from 'rxjs';\n\ninterface DialogData {\n customerId: string;\n networks: Network[];\n documents: Document[];\n}\n\n@Component({\n selector: 'tu-credit-product-modal',\n templateUrl: './credit-product-modal.component.html',\n styleUrls: ['./credit-product-modal.component.scss'],\n})\nexport class CreditProductModalComponent implements OnInit {\n public submitLoader: boolean = false;\n public categoriesLoader: boolean = false;\n public productsLoader: boolean = false;\n public originTripLoader: boolean = false;\n public destinationTripLoader: boolean = false;\n public tripLoader: boolean = false;\n\n public form = this.fb.group({\n networkId: [null, Validators.required],\n categoryId: [null, Validators.required],\n productId: [null, Validators.required],\n support: [null, Validators.required],\n originTripStop: [null],\n destinationTripStop: [null],\n tripToken: [null],\n viaKey: [null],\n validityStartDate: [null],\n validityEndDate: [null],\n quantity: [1, [Validators.required, Validators.min(1)]],\n generationReason: [null, Validators.required],\n generationComment: [null],\n });\n\n public networkNameCtrl = new FormControl();\n public categoryNameCtrl = new FormControl();\n public productNameCtrl = new FormControl();\n public originTripNameCtrl = new FormControl();\n public destinationTripNameCtrl = new FormControl();\n public tripCtrl = new FormControl();\n\n private categories: Category[] = [];\n private products: ProductItem[] = [];\n\n private originStops: TripStop[] = [];\n private destinationStops: TripStop[] = [];\n private trips: ProductTrip[] = [];\n\n public selectedProduct: ProductDetails | null = null;\n private selectedProductFields: number[] = [];\n\n public generationReasonOptions = Object.entries(TicketPurposeLabel)\n .filter(([index]) => {\n // COMMERCIAL_OFFER is deprecated, but we keep it becasue we need the label for old tickets\n return index !== 'COMMERCIAL_OFFER';\n })\n .map(([value, label]) => {\n return {\n label,\n value,\n };\n });\n\n constructor(\n private fb: FormBuilder,\n private shopService: ShopService,\n private customerService: CustomersService,\n private fieldService: FieldService,\n private notification: NotificationsService,\n private translate: TranslateService,\n private modalRef: MatDialogRef,\n @Inject(MAT_DIALOG_DATA) public data: DialogData\n ) {}\n\n get filteredNetworks(): Network[] {\n const networkSearch: string = this.networkNameCtrl.value;\n\n if (networkSearch) {\n return this.data.networks.filter((network) => {\n return network.name.toLowerCase().includes(networkSearch.toLowerCase());\n });\n }\n\n return this.data.networks;\n }\n\n get filteredCategories(): Category[] {\n const categorySearch: Category | string = this.categoryNameCtrl.value;\n\n if (categorySearch) {\n return this.categories.filter((category) => {\n if (typeof categorySearch === 'string') {\n return category.cname.toLowerCase().includes(categorySearch.toLowerCase());\n } else {\n return category.cid === categorySearch.cid;\n }\n });\n }\n\n return this.categories;\n }\n\n get filteredProducts(): ProductItem[] {\n const categoryId = this.form.controls.categoryId.value;\n\n if (categoryId) {\n const productSearch: ProductItem | string = this.productNameCtrl.value;\n\n if (productSearch) {\n return this.products.filter((product) => {\n if (typeof productSearch === 'string') {\n return (\n +product.categorie_id === +categoryId &&\n product.name.toLowerCase().includes(productSearch.toLowerCase())\n );\n } else {\n return +product.categorie_id === +categoryId && product.id === productSearch.id;\n }\n });\n }\n\n return this.products.filter((product) => {\n return +product.categorie_id === +categoryId;\n });\n }\n\n return [];\n }\n\n get filteredOriginTripStops(): TripStop[] {\n const originTripSearch: TripStop | string = this.originTripNameCtrl.value;\n\n if (originTripSearch) {\n return this.originStops.filter((originStop) => {\n if (typeof originTripSearch === 'string') {\n return originStop.stop_name.toLowerCase().includes(originTripSearch.toLowerCase());\n } else {\n return originStop.stop_id === originTripSearch.stop_id;\n }\n });\n }\n\n return this.originStops;\n }\n\n get filteredDestinationTripStops(): TripStop[] {\n const destinationTripSearch: TripStop | string = this.destinationTripNameCtrl.value;\n\n if (destinationTripSearch) {\n return this.destinationStops.filter((destinationStop) => {\n if (typeof destinationTripSearch === 'string') {\n return destinationStop.stop_name\n .toLowerCase()\n .includes(destinationTripSearch.toLowerCase());\n } else {\n return destinationStop.stop_id === destinationTripSearch.stop_id;\n }\n });\n }\n\n return this.destinationStops;\n }\n\n get filteredTripStops(): ProductTrip[] {\n const tripSearch: ProductTrip | string = this.tripCtrl.value;\n\n if (tripSearch) {\n return this.trips.filter((trip) => {\n if (typeof tripSearch === 'string') {\n return trip.viaLabel.toLowerCase().includes(tripSearch.toLowerCase());\n } else {\n return tripSearch.viaKey === tripSearch.viaKey;\n }\n });\n }\n\n return this.trips;\n }\n\n get selectedProductHasOD(): boolean {\n const product = this.selectedProduct;\n\n if (product) {\n return Boolean(product.journey_matrix_id && product.journey_matrix_layer_key);\n }\n\n return false;\n }\n\n get selectedProductHasComplementaryFields(): boolean {\n return this.selectedProductFields.length > 0;\n }\n\n get selectedProductSupportOptions() {\n if (this.selectedProduct) {\n return ProductSupportsLabels.filter((support) => {\n return this.selectedProduct.supports.includes(support.value);\n });\n }\n\n return [];\n }\n\n get customerHasAllProductDocuments(): boolean {\n if (!this.selectedProduct) {\n return true;\n }\n\n const documentIds = this.selectedProduct.documents;\n\n return documentIds.every((documentId) => {\n return this.data.documents.some((document) => {\n return +document.id === +documentId;\n });\n });\n }\n\n get tripsHaveVia(): boolean {\n return this.trips.every((trip) => {\n return trip.viaKey && trip.viaLabel;\n });\n }\n\n ngOnInit(): void {\n /*****************/\n /* Search inputs */\n /*****************/\n\n this.networkNameCtrl.valueChanges.subscribe((name) => {\n if (name) {\n const network = this.data.networks.find(({ name: networkName }) => networkName === name);\n\n if (network) {\n // @ts-ignore\n this.form.controls.networkId.setValue(network.id);\n }\n }\n });\n\n this.categoryNameCtrl.valueChanges.subscribe((category: Category | string) => {\n if (typeof category !== 'string') {\n // @ts-ignore\n this.form.controls.categoryId.setValue(category.cid);\n }\n });\n\n this.productNameCtrl.valueChanges.subscribe((product: ProductItem | string) => {\n if (typeof product !== 'string') {\n // @ts-ignore\n this.form.controls.productId.setValue(product.id);\n }\n });\n\n this.originTripNameCtrl.valueChanges.subscribe((originTrip: TripStop | string) => {\n if (typeof originTrip !== 'string') {\n this.form.controls.originTripStop.setValue(originTrip.stop_id);\n }\n });\n\n this.destinationTripNameCtrl.valueChanges.subscribe((destinationTrip: TripStop | string) => {\n if (typeof destinationTrip !== 'string') {\n this.form.controls.destinationTripStop.setValue(destinationTrip.stop_id);\n }\n });\n\n this.tripCtrl.valueChanges.subscribe((trip: ProductTrip | string) => {\n if (typeof trip !== 'string') {\n this.form.controls.tripToken.setValue(trip.tripToken);\n }\n });\n\n this.form.controls.networkId.valueChanges\n .pipe(debounceTime(500), distinctUntilChanged())\n .subscribe((networkId) => {\n this.retrieveCategories(networkId);\n this.retrieveProducts(networkId);\n });\n\n /*************************/\n /* Verify product has OD */\n /*************************/\n\n this.form.controls.productId.valueChanges.subscribe(async (productId) => {\n if (productId) {\n // HACK: the `Shop/getProductsItem` feed returns more information if a product id is entered\n this.productsLoader = true;\n this.selectedProduct = await this.shopService.getProduct(+productId);\n this.productsLoader = false;\n\n this.fieldService.getFieldsItem(+this.selectedProduct.id).then((fields) => {\n this.selectedProductFields = fields;\n });\n\n // If selected product has OD, we required\n if (this.selectedProductHasOD) {\n this.form.controls.originTripStop.addValidators(Validators.required);\n this.form.controls.destinationTripStop.addValidators(Validators.required);\n this.form.controls.tripToken.addValidators(Validators.required);\n\n this.originTripLoader = true;\n this.shopService.productTripStops(+this.selectedProduct.id).then((originStops) => {\n this.originStops = originStops;\n this.originTripLoader = false;\n });\n } else {\n this.form.controls.originTripStop.clearValidators();\n this.form.controls.destinationTripStop.clearValidators();\n this.form.controls.tripToken.clearValidators();\n }\n\n this.form.updateValueAndValidity();\n }\n });\n\n /*************/\n /* Handle OD */\n /*************/\n\n this.form.controls.originTripStop.valueChanges.subscribe((originTripStop) => {\n this.destinationTripLoader = true;\n this.shopService\n .productTripStops(+this.selectedProduct.id, originTripStop)\n .then((destinationStops) => {\n this.destinationStops = destinationStops;\n this.destinationTripLoader = false;\n });\n });\n\n merge(\n this.form.controls.originTripStop.valueChanges,\n this.form.controls.destinationTripStop.valueChanges\n ).subscribe(() => {\n const originTripStop = this.form.controls.originTripStop.value;\n const destinationTripStop = this.form.controls.destinationTripStop.value;\n\n if (originTripStop && destinationTripStop) {\n this.tripLoader = true;\n this.shopService\n .productTrips(this.form.controls.productId.value, originTripStop, destinationTripStop)\n .then((trips) => {\n this.trips = trips;\n\n if (this.trips.length === 1) {\n this.form.controls.tripToken.setValue(this.trips[0].tripToken);\n } else if (this.tripsHaveVia) {\n this.form.controls.viaKey.setValidators(Validators.required);\n } else {\n this.form.controls.viaKey.clearValidators();\n }\n\n this.form.updateValueAndValidity();\n\n this.tripLoader = false;\n });\n }\n });\n }\n\n public close(needRefresh: boolean = false): void {\n this.modalRef.close({ needRefresh });\n }\n\n public async submit(): Promise {\n if (this.form.invalid) {\n this.notification.error(this.translate.instant('otherslabels.error_required_fields'));\n return;\n }\n\n try {\n this.submitLoader = true;\n\n const {\n productId,\n support,\n quantity,\n generationReason,\n generationComment,\n validityStartDate,\n validityEndDate,\n tripToken,\n } = this.form.value;\n\n const data: {\n tripToken?: string;\n validityStartDate?: string;\n validityEndDate?: string;\n isSubscription?: boolean;\n } = {};\n\n if (this.selectedProduct.is_renewable) {\n data.isSubscription = true;\n }\n\n if (validityStartDate) {\n data.validityStartDate = validityStartDate;\n }\n\n if (validityEndDate) {\n data.validityEndDate = validityEndDate;\n }\n\n if (validityStartDate && validityEndDate) {\n const startDate = new Date(validityStartDate);\n const endDate = new Date(validityEndDate);\n\n if (startDate > endDate) {\n this.notification.error(this.translate.instant('pages.credit_product_modal.date_error'));\n return;\n }\n }\n\n if (this.selectedProductHasOD) {\n if (tripToken) {\n data.tripToken = tripToken;\n } else {\n this.notification.error(this.translate.instant('pages.credit_product_modal.no_trip'));\n return;\n }\n }\n\n await this.customerService.deliverProduct(\n +this.data.customerId,\n productId,\n support,\n quantity,\n generationReason,\n generationComment,\n data\n );\n\n this.notification.success(\n this.translate.instant('pages.credit_product_modal.success_message')\n );\n\n this.close(true);\n } catch {\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n } finally {\n this.submitLoader = true;\n }\n }\n\n public displayAutocompleteCategory(category: Category | null): string {\n return category?.cname;\n }\n\n public displayAutocompleteProduct(product: ProductItem | null): string {\n return product?.name;\n }\n\n public displayTrip(trip: TripStop | null): string {\n return trip?.stop_name;\n }\n\n public displayProductTrip(trip: ProductTrip | null): string {\n return trip?.viaLabel;\n }\n\n private async retrieveCategories(networkId: string): Promise {\n this.categoriesLoader = true;\n this.categories = await this.shopService.getNetworkCategories(networkId);\n this.categoriesLoader = false;\n }\n\n private async retrieveProducts(networkId: string): Promise {\n this.productsLoader = true;\n this.products = await this.shopService.getNetworkProducts(networkId);\n this.productsLoader = false;\n }\n}\n","import { Component, Input } from '@angular/core';\nimport { heroicons } from '@assets/icons/heroicons';\n\ntype Status = 'default' | 'info' | 'warning' | 'success' | 'error';\n\n@Component({\n selector: 'tu-closable-card',\n styleUrls: ['./card.component.scss'],\n template: `\n \n \n \n \n \n \n\n \n \n\n
    \n \n
    \n\n
    \n \n
    \n \n `,\n})\nexport class ClosableCardComponent {\n public heroicons = heroicons;\n\n @Input('defaultOpen')\n public defaultOpen = false;\n\n @Input('icon')\n public icon?: string;\n\n @Input('withFooter')\n public withFooter = false;\n\n @Input('klass')\n public klass = '';\n\n @Input('status')\n public status: Status = 'default';\n}\n","\n
    \n
    \n \n En cours d'édition\n
    \n \n \n {{ 'pages.customer_details.edit_comment' | translate }}\n \n \n \n \n {{ 'pages.customer_details.delete_comment' | translate }}\n \n \n \n
    \n\n \n
    \n \n \n {{ 'pages.customer_details.edit_comment' | translate }}\n \n \n \n \n \n {{ 'pages.customer_details.delete_comment' | translate }}\n \n \n \n
    \n
    \n \n\n
    \n","import { Component, Input } from '@angular/core';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { TranslateService } from '@ngx-translate/core';\n\n@Component({\n selector: 'tu-customer-comment',\n templateUrl: './customer-comment.component.html',\n styleUrls: ['./customer-comment.component.scss'],\n})\nexport class CustomerCommentComponent {\n public heroicons = heroicons;\n\n @Input()\n public comment: any;\n\n @Input()\n public commentToEdit: (id: number, message: string) => void;\n\n @Input()\n public editedComment: string;\n\n @Input()\n public commentToDelete: (id: number) => void;\n\n @Input()\n public markdownToHTML: (markdown: string) => string;\n\n constructor(public auth: AuthService, public translate: TranslateService) {}\n}\n","\n \n \n \n \n \n \n {{ $$customer.customer_id }}\n \n\n \n\n \n {{ 'pages.customer_details.anonymize_date' | translate }}\n {{ $$customer.anonymized_at | date : 'shortDate' : undefined : translate.currentLang }}\n \n\n \n {{ $$customer.firstname }} {{ $$customer.lastname }}\n \n {{ 'pages.customer.guest_mode' | translate }}\n \n \n

    \n\n \n
    \n \n {{ 'pages.customer_details.action' | translate }}\n\n \n \n\n \n
    \n\n \n \n
  • \n \n \n \n \n {{ 'pages.customer_details.contact_support' | translate }}\n \n {{\n (isCustomerActive\n ? 'pages.customer_details.turn_off_account'\n : 'pages.customer_details.turn_on_account'\n ) | translate\n }}\n \n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.anonymize' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.send_verification_mail' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.account_check_validation' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.reset_password' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.update_personal_information' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.phone_lost' | translate }}\n \n
  • \n\n \n
  • \n \n \n {{ 'pages.customer_details.user_fusion' | translate }}\n \n
  • \n \n \n
  • \n \n \n {{ 'pages.customer_details.create_firebase_account' | translate }}\n \n
  • \n \n \n \n\n
    \n
    \n
    \n \n \n \n\n \n \n \n {{ 'pages.customer_details.update_user_picture' | translate }}\n \n\n \n \n
    \n
    \n\n \n
    \n \n\n \n {{ $$customer.email }}\n \n\n \n {{ ( emailIsValid ? 'pages.customer_details.valid' : 'pages.customer_details.invalid') | translate }}\n
    \n
    \n\n
    \n \n\n

    \n {{ $$customer.address.streetNumber }} {{ $$customer.address.route }}\n
    \n\n {{ $$customer.address.zipCode }} {{ $$customer.address.city }}\n
    \n\n {{ $$customer.address.country }}\n

    \n
    \n\n

    \n \n\n {{ $$customer.phone }}\n

    \n\n

    \n \n\n \n {{ $$customer.birthday | date : 'shortDate' : undefined : translate.currentLang }}\n \n

    \n\n
    \n

    \n \n {{ 'pages.customer_details.signin' | translate }} :\n \n\n \n {{ $$customer.date | date : 'short' : undefined : translate.currentLang }}\n \n

    \n\n

    \n \n {{ 'pages.customer_details.last_activity_label' | translate }} :\n \n\n \n {{\n $$customer.lastActivityAt\n ? ($$customer.lastActivityAt\n | date : 'shortDate' : undefined : translate.currentLang)\n : ('pages.customer_details.unknown' | translate)\n }}\n \n

    \n
    \n\n

    \n \n {{ 'pages.customer_details.network_language' | translate }}\n \n\n \n {{ 'otherslabels.' + $$customer.locale | lowercase | translate }}\n \n

    \n\n
    \n

    {{ 'pages.customer_details.account_deletion' | translate }} :

    \n\n

    \n \n {{ 'pages.customer_details.account_deletion_request_date' | translate }} :\n \n\n \n {{\n $$customer.anonymization_requested_at\n | date : 'shortDate' : undefined : translate.currentLang\n }}\n \n

    \n\n

    \n \n {{ 'pages.customer_details.account_deletion_notification_date' | translate }} :\n \n\n \n {{\n $$customer.anonymization_notified_at\n | date : 'shortDate' : undefined : translate.currentLang\n }}\n \n\n \n \n {{ 'pages.customer_details.notification_not_sent' | translate }}\n \n \n

    \n
    \n\n

    \n \n {{\n (!$$customer.verified_at\n ? 'pages.customer_details.verified_not'\n : 'pages.customer_details.verified'\n ) | translate\n }}\n \n

    \n \n
    0\">\n

    {{ 'pages.customer_details.auth_mode' | translate }}

    \n
    \n \n {{ identity.providerSlug }}\n \n
    \n
    \n \n \n
    \n\n \n \n
    \n
    {{ 'pages.customer_details.profiles' | translate }}
    \n\n \n \n \n {{ 'pages.customer_details.profiles_list' | translate }}\n \n \n \n \n {{ 'pages.customer_details.firstname' | translate }}\n {{ 'pages.customer_details.lastname' | translate }}\n {{ 'pages.customer_details.birthday' | translate }}\n \n \n\n \n \n \n \n
    \n \n \n
    \n \n \n \n \n {{ profile.firstname }}\n \n \n \n \n {{ profile.lastname }}\n \n \n \n \n {{ profile.birthdate | date: 'shortDate' }}\n \n \n \n \n \n\n \n

    {{ 'pages.customer_details.no_children_profil' | translate }}

    \n
    \n
    \n\n \n

    {{ 'pages.customer_details.profiles_load' | translate }}

    \n
    \n
    \n
    \n\n \n
    \n
    \n
    \n
    \n
    \n \n
    {{ statusType['REMAINING'].count }}
    \n
    \n {{ statusType['REMAINING'].label | translate }}\n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n \n
    {{ statusType['ACTIVE'].count }}
    \n
    \n {{ statusType['ACTIVE'].label | translate }}\n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n \n
    {{ statusType['PENDING'].count }}
    \n
    \n {{ statusType['PENDING'].label | translate }}\n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n \n
    {{ statusType['EXPIRED'].count }}
    \n
    \n {{ statusType['EXPIRED'].label | translate }}\n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n \n
    {{ statusType['TRANSFER_OUT'].count }}
    \n
    \n {{ statusType['TRANSFER_OUT'].label | translate }}\n
    \n
    \n
    \n
    \n\n
    \n
    \n
    \n \n
    {{ statusType['TRANSFER_IN'].count }}
    \n
    \n {{ statusType['TRANSFER_IN'].label | translate }}\n
    \n
    \n
    \n
    \n
    \n
    \n\n \n \n
    \n {{ 'pages.order.filters' | translate }}\n
    \n\n \n
    \n \n {{ 'pages.customer_details.wallet_filters' | translate }}\n \n\n
    \n \n \n \n\n \n
    \n\n 1\"\n class=\"input-group tw-mt-4 tw-flex md:tw-mt-0 md:tw-px-2\"\n >\n \n \n \n\n \n\n \n \n
    {{ name }}
    \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 {{ 'pages.customer_details.user_wallet' | translate }}\n
    \n\n \n {{ 'pages.customer_details.add_title' | translate }}\n \n\n \n {{ 'pages.customer_details.delete_tickets' | translate }}\n \n \n\n \n \n \n \n \n

    \n {{ 'pages.customer_details.sav_ticketv2' | translate }}\n

    \n
    \n
    \n {{ 'otherslabels.generic_error' | translate }}\n
    \n\n \n
    \n
    \n \n\n \n \n
    \n
    \n {{ 'otherslabels.generic_error' | translate }}\n
    \n \n
    \n
    \n \n\n \n \n \n \n \n
    \n
    \n
    \n\n \n \n
    \n {{ 'pages.validation.validations' | translate }}\n \n {{ 'pages.customer_details.validations_last_48h' | translate }}\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
    {{ 'pages.customer_details.order_history' | translate }}
    \n \n
    \n

    {{ 'pages.order.filters' | translate }}

    \n\n \n \n
    \n
    \n \n \n \n
    \n\n \n
    \n \n\n \n
    \n \n {{ 'pages.order.reset_form' | translate }}\n \n \n
    \n \n\n
    \n \n
    \n \n
    \n\n \n \n
    {{ 'pages.customer_details.medias' | translate }}
    \n\n
    \n \n
    \n
    \n\n \n \n
    {{ 'pages.subscriptions.subscriptions' | translate }}
    \n\n
    \n \n
    \n
    \n\n \n \n
    {{ 'pages.customer_details.transfered_titles' | translate }}
    \n
    \n \n
    \n
    \n\n \n 1\" klass=\"tw-mt-6\" [defaultOpen]=\"true\">\n
    \n {{ 'pages.customer_details.customer_networks_title' | translate }}\n
    \n ]\n
    \n \n \n\n \n \n \n\n \n \n {{ 'pages.customer_details.actions' | translate }}\n \n \n \n \n\n \n \n \n \n\n \n \n \n\n \n \n Supprimer\n \n \n \n \n\n \n \n\n \n \n {{ 'otherslabels.btn_save' | translate }}\n \n \n \n \n\n \n \n \n \n \n
    \n {{ 'pages.customer_details.customer_networks' | translate }}\n
    \n {{ 'pages.customer_details.network_name' | translate }}\n
    \n {{ 'pages.customer_details.no_network_message' | translate }}\n
    \n \n {{ network.name }}\n \n
    \n \n \n\n \n {{ network.name }}\n \n \n
    \n \n \n\n \n {{ 'pages.customer_details.add_network' | translate }}\n \n \n\n \n \n {{ 'otherslabels.btn_cancel' | translate }}\n \n \n
    \n
    \n
    \n\n \n \n
    \n
    {{ 'pages.customer_details.documents' | translate }}
    \n \n {{ 'pages.document_upload_modal.title' | translate }}\n \n
    \n\n
    \n \n \n
    \n \n
    \n
    \n\n \n
    \n \n
    \n
    \n
    \n\n \n

    {{ 'otherslabels.loading' | translate }}

    \n
    \n
    \n
    \n\n \n \n
    {{ 'pages.customer_details.submissions' | translate }}
    \n\n
    \n \n\n \n

    {{ 'otherslabels.loading' | translate }}

    \n
    \n
    \n
    \n\n \n \n
    {{ 'pages.customer_details.vouchers' | translate }}
    \n\n
    \n \n\n \n

    {{ 'otherslabels.loading' | translate }}

    \n
    \n
    \n
    \n\n \n \n
    {{ 'pages.customer_details.metadata' | translate }}
    \n\n
    \n \n\n \n

    \n {{ 'pages.customer_details.loading_attributes' | translate }}\n

    \n
    \n\n

    \n {{ 'pages.customer_details.hasauthasguestorders' | translate }} :\n\n \n {{\n $$customer.hasAuthAsGuestOrders\n ? ('otherslabels.yes' | translate)\n : ('otherslabels.no' | translate)\n }}\n \n

    \n
    \n
    \n\n \n \n
    {{ 'pages.customer_details.lastconnections' | translate }}
    \n\n
    \n \n
    \n
    \n\n \n \n \n\n \n \n \n\n
    \n
    \n
    \n\n \n
    \n\n\n
    \n

    {{ 'otherslabels.loading' | translate }}

    \n
    \n
    \n","import { ChangeDetectorRef, Component, OnInit, ViewChild } from '@angular/core';\nimport { UntypedFormBuilder } from '@angular/forms';\nimport { MatDialog } from '@angular/material/dialog';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { TableElementType, TagType } from '@app/modules/shared/components/table/table.component';\nimport { PermissionGuard } from '@app/modules/shared/guards/permission.guard';\nimport { CommentFormValues } from '@app/modules/shared/models/comment';\nimport { Network } from '@app/modules/shared/models/network';\nimport { Ticket, TicketPurpose, TicketPurposeLabel } from '@app/modules/shared/models/ticket';\nimport { Voucher } from '@app/modules/shared/models/voucher';\nimport { KeyPipe } from '@app/modules/shared/pipes/key.pipe';\nimport { SearchPipe } from '@app/modules/shared/pipes/search.pipe';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { DeleteDocumentModalComponent } from '@app/modules/submissions/components/submission-detail/delete-document-modal/delete-document-modal.component';\nimport { SubmissionService } from '@app/modules/submissions/services/submission.service';\nimport { SubscriptionDetails } from '@app/modules/subscriptions/models/subscription.interface';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationsService } from 'angular2-notifications';\nimport { DateFormatPipe } from 'ngx-moment';\nimport { firstValueFrom, Observable, startWith } from 'rxjs';\nimport { distinctUntilChanged, map } from 'rxjs/operators';\nimport { UpdateDocumentModalComponent } from '../../../shared/components/update-document-modal/update-document-modal.component';\nimport {\n Attribute,\n Contract,\n CustomerOrderStatus,\n CustomerValidation,\n CustomerWalletEvent,\n CustomerWalletEventType,\n Document,\n Session,\n Transfer,\n} from '../../models/customer';\nimport {\n $$Comment,\n $$Customer,\n $$GlobalAttribute,\n $$Profile,\n $$Ticket,\n CustomersService,\n} from '../../services/customers.service';\nimport { $$Item, ProductsService } from '../../services/products.service';\nimport { TicketService } from '../../services/ticket.service';\nimport { AnonymizationModalComponent } from './anonymization-modal/anonymization-modal.component';\nimport {\n AttributesModalComponent,\n DialogData,\n} from './attributes-modal/attributes-modal.component';\nimport { ImageUpdateModalComponent } from './avatar-update-modal/avatar-update-modal.component';\nimport { InformationModalComponent } from './information-modal/information-modal.component';\nimport { CustomerLostPhoneModalComponent } from './lost-phone-modal/customer-lost-phone-modal.component';\nimport { NetworkDissociationModalComponent } from './network-dissociation-modal/network-dissociation-modal.component';\nimport { TicketGenerationModalComponent } from './ticket-generation-modal/ticket-generation-modal.component';\nimport { compareAsc, differenceInMinutes } from 'date-fns';\nimport { Blueprint } from '@app/modules/shop/models/blueprint/blueprint';\nimport { FusionModalComponent } from './fusion-modal/fusion-modal.component';\nimport { SubscriptionsService } from '@app/modules/subscriptions/services/subscriptions.service';\nimport { PaperworkService } from '../../services/paperwork.service';\nimport { UsersService } from '@app/modules/users/services/users.service';\nimport { User } from '@app/modules/users/models/user';\nimport { match } from 'ts-pattern';\nimport { ColumnConfig } from '@app/modules/shared/models/server-pagination';\nimport { ServerPaginationService } from '@app/modules/shared/services/server-pagination.service';\nimport { Belonging, BookingFilters } from '../../models/booking';\nimport { DatePipe } from '@angular/common';\nimport { ArticleBooking } from '@app/modules/orders/models/order';\nimport {\n DeleteTicketModalComponent,\n DeleteTicketModalResponse,\n} from './delete-ticket-modal/delete-ticket-modal.component';\nimport { ERROR_FIREBASE_RESET_PASSWORD } from '../../constants/resetPassword';\nimport { ShopService } from '../../services/shop.service';\nimport { DocumentUploadModalComponent } from './document-upload-modal/document-upload-modal.component';\nimport { Identity } from '../../models/identity';\nimport { CreateFirebaseAccountModalComponent } from './create-firebase-account-modal/create-firebase-account-modal.component';\nimport { CreditProductModalComponent } from './credit-product-modal/credit-product-modal.component';\n\nconst CONTRACT_STATUS = {\n CANCELED: 'pages.customer_details.status_canceled',\n EXPIRED: 'pages.customer_details.status_expired',\n REMAINING: 'pages.customer_details.status_remaining',\n PENDING: 'pages.customer_details.status_pending',\n EMPTY: 'pages.customer_details.status_empty',\n ACTIVE: 'pages.customer_details.status_active',\n} as const;\n\n@Component({\n selector: 'tu-customer-detail',\n templateUrl: './customer-detail.component.html',\n styleUrls: ['./customer-detail.component.scss'],\n providers: [CustomersService, TicketService, ProductsService, SearchPipe, KeyPipe, DatePipe],\n})\nexport class CustomerDetailComponent implements OnInit {\n constructor(\n public auth: AuthService,\n public translate: TranslateService,\n public modal: MatDialog,\n private customersService: CustomersService,\n private ticketService: TicketService,\n private productsService: ProductsService,\n private route: ActivatedRoute,\n private fb: UntypedFormBuilder,\n private permissionGuard: PermissionGuard,\n private notification: NotificationsService,\n private router: Router,\n private dateFormat: DateFormatPipe,\n private submissionService: SubmissionService,\n private searchPipe: SearchPipe,\n private keyPipe: KeyPipe,\n private readonly changeDetectorRef: ChangeDetectorRef,\n public lostPhoneModal: MatDialog,\n private paperworkService: PaperworkService,\n private subscriptionService: SubscriptionsService,\n private userService: UsersService,\n public serverPaginationService: ServerPaginationService,\n private datePipe: DatePipe,\n private shopService: ShopService\n ) {}\n\n private facebookRegex = /https:\\/\\/scontent\\.xx\\.fbcdn\\.net\\/v\\/t1\\.0-1\\/[^\\/]*\\/\\d*_(\\d*)_.*/g;\n\n public customerId: string | null = null;\n\n public $$customer: $$Customer;\n public $$loadingCustomer = true;\n\n public loadingIdentities: boolean = true;\n public identities: Identity[] = [];\n\n public $$profiles: $$Profile[] = [];\n public $$loadingProfiles = true;\n\n public isTicketV1Enabled: boolean = false;\n\n public $$wallet: $$Ticket[] = [];\n public $$errorWallet: boolean = false;\n public $$filteredWallet: $$Ticket[] = [];\n public $$loadingWallet = true;\n public $$selectedTicketRows = [];\n\n public $$contracts: Contract[] = [];\n public $$filteredContracts: Contract[] = [];\n public $$loadingContracts: boolean = true;\n public $$errorContracts: boolean = false;\n\n public bookingFilters: BookingFilters = {};\n\n public $$validations: CustomerValidation[] = [];\n public $$loadingValidations = false;\n public $$walletEvents: CustomerWalletEvent[] = [];\n public $$loadingwalletEvents = false;\n public get customerValidationsView() {\n if (this.$$validations.length && this.$$walletEvents.length) return 'TABS';\n if (this.$$validations.length) return 'V1';\n if (this.$$walletEvents.length) return 'V2';\n return 'NONE';\n }\n\n private _cmsUsers: User[] = [];\n public get hasV1Tickets() {\n return this.$$filteredWallet?.length > 0;\n }\n public get hasV2Tickets() {\n return Boolean(this.$$filteredContracts?.length);\n }\n public get isTicketV2Enabled() {\n return (\n this.$$customerNetworks?.some((customerNetwork) => customerNetwork.blueprints.length > 0) ||\n false\n );\n }\n\n public get isBookingEnabled() {\n return this.$$networks.some((network) => network.hasReservation) || false;\n }\n\n private get activeTicketsTab() {\n if (this.ticketV1Tab?.active) {\n return 'v1';\n }\n\n if (this.ticketV2Tab?.active) {\n return 'v2';\n }\n\n if (this.bookingTab?.active) {\n return 'booking';\n }\n }\n\n public get isFusionDisabled() {\n const isDisabled =\n this.$$customer.active === '0' ||\n this.$$customer.anonymized_at ||\n !this.isCustomerParent ||\n !this.$$customer.isFusionable;\n return isDisabled ? true : null;\n }\n\n public get emailIsValid(): boolean {\n if (this.$$customer.emailBouncedAt) {\n const now = new Date();\n const emailBouncedAt = new Date(this.$$customer.emailBouncedAt);\n\n if (this.$$customer.emailBouncedUntil) {\n const emailBouncedUntil = new Date(this.$$customer.emailBouncedUntil);\n\n return emailBouncedUntil < now;\n }\n\n return emailBouncedAt > now;\n }\n\n return true;\n }\n\n public get hasFranceConnectProvider(): boolean {\n return this.identities.some((identity) => {\n return identity.providerSlug === 'FranceConnect';\n });\n }\n\n public get hasFirebase(): boolean {\n return Boolean(this.$$customer.firebaseUid);\n }\n\n public get hasAirwebProvider(): boolean {\n return this.$$customer.provider === 'airweb';\n }\n\n public $$medias: any[] = [];\n public $$loadingMedias = true;\n\n public $$subscriptions: SubscriptionDetails[] = [];\n public $$loadingSubscriptions = true;\n\n public $$networks: Network[] = [];\n public $$isAddingNetwork = false;\n public $$newNetworkId = '';\n public $$newNetWorkLoading = false;\n public $$customerNetworks: { network: Network; blueprints: Blueprint[] }[];\n\n public $$documents: Document[] = [];\n public $$deletedDocuments: Document[] = [];\n public $$loadingDocuments = true;\n\n public $$submissions: any[] = [];\n public $$loadingSubmissions = true;\n\n public $$vouchers: Voucher[] = [];\n public $$loadingVouchers = true;\n\n public $$allAttributes: $$GlobalAttribute[] = [];\n public $$loadingAllAttributes = true;\n\n public $$attributes: Attribute[] = [];\n public $$loadingAttributes = true;\n\n public $$comments: $$Comment[] = [];\n public $$loadingComments = true;\n public $$selectedCommentNetwork = '';\n\n public $$products: $$Item[] = [];\n public $$loadingProducts = true;\n\n public sessions: Session[] = [];\n\n public isSuperAdmin = false;\n\n public heroicons = heroicons;\n\n public filterForm = this.fb.group({\n status: [null],\n customerQuery: [null],\n customerNetworkName: [null],\n customerNetwork: [null],\n origin: [null],\n });\n\n public filterOrderHistory = this.fb.group({\n status: [null],\n });\n\n // Used for autocomplete\n public filteredNetworksName: Observable =\n this.filterForm.controls.customerNetworkName.valueChanges.pipe(\n startWith(null),\n map((networkName: string | null) => {\n const networkNames = this.auth.networks.map((network) => network.name);\n if (!networkName) return networkNames;\n\n const network = this.auth.networks.find((network) => network.name === networkName);\n\n if (network) {\n this.filterForm.controls.customerNetwork.setValue(+network.id);\n }\n\n return networkNames.filter((network) => {\n return network.toLowerCase().includes(networkName.toLowerCase());\n });\n })\n );\n\n public isActionOpen = false;\n\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n //TRANSFER HISTORY\n public customer_history: Transfer[] = [];\n\n @ViewChild('ticketV1Tab') ticketV1Tab;\n @ViewChild('ticketV2Tab') ticketV2Tab;\n @ViewChild('bookingTab') bookingTab;\n\n private styles = {\n canceled: {\n text: 'color:gray; text-decoration:line-through; background-color:rgb(232, 232, 232);',\n icon: 'color:transparent; pointer-events: none;background-color:rgb(232, 232, 232);',\n tag: 'background-color: rgb(232, 232, 232);',\n },\n };\n\n public statusType = {\n PENDING: {\n count: 0,\n label: this.translate.instant('pages.customer_details.validated_titles'),\n },\n ACTIVE: {\n count: 0,\n label: this.translate.instant('pages.customer_details.to_be_validated_titles'),\n },\n REMAINING: {\n count: 0,\n label: this.translate.instant('pages.customer_details.remaining_titles'),\n },\n EXPIRED: {\n count: 0,\n label: this.translate.instant('pages.customer_details.expired_titles'),\n },\n TRANSFER_OUT: {\n count: 0,\n label: this.translate.instant('pages.customer_details.transfered_out_titles'),\n },\n TRANSFER_IN: {\n count: 0,\n label: this.translate.instant('pages.customer_details.transfered_in_titles'),\n },\n };\n\n public walletTableConfig = [\n {\n key: 'ticket_id',\n label: this.translate.instant('pages.customer_details.title_id'),\n type: TableElementType.Link,\n modifier: (tid: string, ticket: Ticket) => {\n return {\n link: ['/ticket', ticket.id],\n label: tid,\n };\n },\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'created_at',\n label: this.translate.instant('pages.customer_details.created_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY') : null;\n },\n },\n {\n key: 'label',\n label: this.translate.instant('pages.customer_details.title'),\n type: TableElementType.Text,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'order_identifier',\n label: this.translate.instant('pages.customer_details.sale'),\n type: TableElementType.Link,\n exportable: true,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n modifier: (oid: string, ticket: Ticket) => {\n return {\n link: this._walletOriginLink({ ticket }),\n label: this._walletOriginLabel({ ticket }),\n };\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.status'),\n type: TableElementType.Tag,\n config: {\n tags: {\n EXPIRED: {\n type: TagType.Danger,\n label: this.translate.instant('pages.customer_details.expired'),\n },\n PENDING: {\n type: TagType.Success,\n label: this.translate.instant('pages.customer_details.validated'),\n },\n ACTIVE: {\n type: TagType.Warning,\n label: this.translate.instant('pages.customer_details.to_be_validated'),\n },\n REMAINING: {\n type: TagType.Info,\n label: this.translate.instant('pages.customer_details.remaining'),\n },\n CANCELED: {\n type: TagType.Default,\n label: this.translate.instant('pages.customer_details.canceled'),\n },\n CANCELLED: {\n type: TagType.Default,\n label: this.translate.instant('pages.customer_details.canceled'),\n },\n },\n },\n style: this.canceledStyle.bind(this, this.styles.canceled.tag),\n },\n\n {\n key: 'dematerialized',\n label: this.translate.instant('pages.order.delivery'),\n type: TableElementType.Icon,\n config: {\n source: 'heroicons',\n labels: {\n '0': this.translate.instant('pages.customer_details.shipping_physical'),\n '1': this.translate.instant('pages.customer_details.shipping_demat'),\n },\n icons: {\n '0': heroicons.outline.creditCard,\n '1': heroicons.outline.deviceMobile,\n },\n },\n },\n {\n key: 'network_id',\n label: this.translate.instant('pages.submissions.network'),\n type: TableElementType.Text,\n modifier: (networkId: string) => {\n const network = this.auth.getNetwork(networkId);\n\n return network?.name ?? '-';\n },\n isSortable: true,\n },\n {\n key: 'generationReason',\n label: this.translate.instant('pages.customer_details.generationreason'),\n type: TableElementType.Tag,\n modifier: (generationReason: string | null) => {\n return generationReason || 'EMPTY';\n },\n config: {\n tags: {\n DEVICE_CHANGE: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['DEVICE_CHANGE']),\n },\n COMMERCIAL_GESTURE: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['COMMERCIAL_GESTURE']),\n },\n COMMERCIAL_OFFER: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['COMMERCIAL_OFFER']),\n },\n CUSTOMER_SUPPORT: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['CUSTOMER_SUPPORT']),\n },\n TEST: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['TEST']),\n },\n EMPTY: {\n type: TagType.Default,\n label: this.translate.instant('otherslabels.unkown'),\n },\n },\n },\n isSortable: false,\n },\n {\n key: 'generationComment',\n label: this.translate.instant('pages.customer_details.comment'),\n type: TableElementType.Text,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details.action'),\n type: TableElementType.Submenu,\n displayed: this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_WALLET']),\n config: {\n icon: heroicons.outline.dotsVertical,\n actions: [\n {\n label: this.translate.instant('pages.customer_details.transfert'),\n fn: (ticket: $$Ticket) => {\n console.log(ticket);\n this.openTicketGenerationModal(ticket);\n },\n icon: heroicons.outline.documentDuplicate,\n active: this._isTicketNotCanceled,\n },\n {\n label: this.translate.instant('otherslabels.btn_delete'),\n fn: async (ticket: $$Ticket) => {\n this._desactivateTicket(ticket);\n },\n icon: heroicons.outline.trash,\n active: this._isTicketNotCanceled,\n },\n {\n label: this.translate.instant('pages.customer_details.reactivate'),\n fn: async (ticket: Ticket) => {\n if (\n !this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_WALLET']) ||\n !['CANCELED', 'CANCELLED'].includes(ticket.status)\n )\n return;\n const ok = await this.ticketService.reactivateTicket(+ticket.id);\n if (ok) {\n this.updateWallet(this.$$customer.customer_id);\n this.notification.success(\n this.translate.instant(`otherslabels.notif_reactivate_ok`)\n );\n } else {\n this.notification.error(this.translate.instant(`otherslabels.notif_reactivate_ko`));\n }\n },\n icon: heroicons.outline.refresh,\n active: this._isTicketCanceled,\n },\n ],\n },\n },\n ];\n\n public contractTableConfig = [\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details_tickets_v2.contract'),\n type: TableElementType.Link,\n modifier: (id: string) => {\n return {\n link: ['/ticket', 'v2', id],\n label: id,\n };\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-xs',\n },\n },\n {\n key: 'blueprintName',\n label: this.translate.instant('pages.customer_details_tickets_v2.ticket'),\n type: TableElementType.Text,\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: '',\n label: this.translate.instant('pages.customer_details.sale'),\n type: TableElementType.Link,\n modifier: (empty: null, contract: Contract) => {\n return {\n link: this._walletOriginLink({ contract }),\n label: this._walletOriginLabel({ contract }),\n };\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'generationReason',\n label: this.translate.instant('pages.customer_details.generationreason'),\n type: TableElementType.Tag,\n config: {\n tags: {\n DEVICE_CHANGE: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['DEVICE_CHANGE']),\n },\n COMMERCIAL_GESTURE: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['COMMERCIAL_GESTURE']),\n },\n COMMERCIAL_OFFER: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['COMMERCIAL_OFFER']),\n },\n CUSTOMER_SUPPORT: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['CUSTOMER_SUPPORT']),\n },\n TEST: {\n type: TagType.Warning,\n label: this.translate.instant(TicketPurposeLabel['TEST']),\n },\n EMPTY: {\n label: this.translate.instant('otherslabels.unkown'),\n type: TagType.Default,\n },\n },\n },\n modifier(purpose: TicketPurpose) {\n return purpose || 'EMPTY';\n },\n isSortable: false,\n },\n {\n key: 'generationComment',\n label: this.translate.instant('pages.customer_details.comment'),\n type: TableElementType.Text,\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'createdAt',\n label: this.translate.instant('pages.customer_details_tickets_v2.date'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'state.contractStartDate',\n label: this.translate.instant('pages.customer_details_tickets_v2.starting_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'state.contractEndDate',\n label: this.translate.instant('pages.customer_details_tickets_v2.ending_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'firstEvent.occurredAt',\n label: this.translate.instant('pages.ticket_details_v2.first_validation'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'lastEvent.occurredAt',\n label: this.translate.instant('pages.ticket_details_v2.last_validation'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: 'blockedAt',\n label: this.translate.instant('pages.customer_details_tickets_v2.blocked_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY HH:mm') : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-sm',\n },\n },\n {\n key: '',\n label: this.translate.instant('pages.customer_details.status'),\n type: TableElementType.Text,\n modifier: (empty: null, contract: Contract) => {\n const status_key = this._contractStatus(contract);\n return this.translate.instant(status_key);\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'networkId',\n label: this.translate.instant('pages.submissions.network'),\n type: TableElementType.Text,\n modifier: (networkId: string) => {\n const network = this.auth.getNetwork(networkId);\n return network?.name ?? '-';\n },\n isSortable: true,\n },\n {\n key: 'state.validationRemainingPunches',\n label: this.translate.instant('pages.customer_details.tickets_number'),\n type: TableElementType.Text,\n modifier: (remainingPunch: number) => {\n return remainingPunch > 0 ? remainingPunch : null;\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'transferredAt',\n label: this.translate.instant('pages.customer_details.contract_transferredat'),\n type: TableElementType.Tag,\n modifier: (transferredAt: string) => {\n return transferredAt ? 'YES' : 'NO';\n },\n config: {\n tags: {\n NO: { type: TagType.Default, label: this.translate.instant('otherslabels.no') },\n YES: { type: TagType.Primary, label: this.translate.instant('otherslabels.yes') },\n },\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n class: {\n row: 'tw-text-center',\n },\n },\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details.action'),\n type: TableElementType.Submenu,\n displayed: this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_WALLET']),\n config: {\n icon: heroicons.outline.dotsVertical,\n actions: [\n {\n label: this.translate.instant('otherslabels.btn_delete'),\n fn: async (contract: Contract) => {\n this._desactivateContract(contract);\n },\n icon: heroicons.outline.trash,\n active: this._isContractActive,\n },\n {\n label: this.translate.instant('pages.customer_details.reactivate'),\n fn: async (contract: Contract) => {\n this._activateContract(contract);\n },\n icon: heroicons.outline.refresh,\n active: this._isContractCanceled,\n },\n ],\n },\n style: this.canceledContractStyle.bind(this, this.styles.canceled.text),\n },\n ];\n\n public get bookingTableConfig(): ColumnConfig[] {\n return [\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details.booking_number'),\n type: 'text',\n isSortable: false,\n },\n {\n key: 'orderId',\n label: this.translate.instant('pages.customer_details.order'),\n type: 'link',\n config: {\n href: (_, belonging: Belonging) => {\n return `/orders/${belonging.orderId}`;\n },\n },\n isSortable: false,\n },\n {\n key: '',\n label: this.translate.instant('pages.customer_details.booking_details'),\n type: 'html',\n modifier: (_, belonging: Belonging) => {\n return this.getHtmlBelongingCell(belonging.reservation);\n },\n isSortable: false,\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.status'),\n type: 'badge',\n config: {\n REMAINING: {\n type: 'information',\n label: this.translate.instant('pages.customer_details.remaining'),\n },\n IN_PROGRESS: {\n type: 'success',\n label: this.translate.instant('pages.customer_details.in_progress'),\n },\n EXPIRED: {\n type: 'default',\n label: this.translate.instant('pages.customer_details.expired'),\n },\n },\n modifier(_, belonging: Belonging) {\n const now = new Date();\n if (belonging.reservation) {\n const tripDepartureDate = new Date(belonging.reservation.tripDepartureDate);\n const tripArrivalDate = new Date(belonging.reservation.tripArrivalDate);\n\n if (tripDepartureDate > now) {\n return 'REMAINING';\n }\n\n if (now > tripDepartureDate && tripArrivalDate > now) {\n return 'IN_PROGRESS';\n }\n }\n return 'EXPIRED';\n },\n isSortable: false,\n },\n {\n key: 'networkId',\n label: this.translate.instant('pages.submissions.network'),\n type: 'text',\n modifier: (networkId: string) => {\n const network = this.auth.getNetwork(networkId);\n return network?.name ?? '-';\n },\n isSortable: false,\n },\n ];\n }\n\n public transfersHistoryTableConfig = [\n {\n key: 'direction',\n label: this.translate.instant('pages.customer_details.history_transfer_direction'),\n type: TableElementType.Tag,\n config: {\n tags: {\n IN: {\n type: TagType.Success,\n label: this.translate.instant('pages.customer_details.history_transfer_receive'),\n },\n OUT: {\n type: TagType.Warning,\n label: this.translate.instant('pages.customer_details.history_transfer_emit'),\n },\n },\n },\n class: {\n row: 'tw-text-base',\n },\n },\n {\n key: 'issuerCustomerLabel',\n label: this.translate.instant('pages.customer_details.history_emitter'),\n type: TableElementType.Link,\n modifier: (label: string, transfert: Transfer) => {\n return {\n link: transfert.direction === 'IN' ? ['/customers', transfert.issuerCustomerId] : '',\n label: label || transfert.issuerCustomerId,\n };\n },\n },\n {\n key: 'recipientCustomerLabel',\n label: this.translate.instant('pages.customer_details.history_recipient'),\n type: TableElementType.Link,\n modifier: (label: string, transfert: Transfer) => {\n return {\n link: transfert.direction === 'OUT' ? ['/customers', transfert.recipientCustomerId] : '',\n label: label || transfert.recipientCustomerId,\n };\n },\n },\n {\n key: 'createdAt',\n label: this.translate.instant('pages.customer_details.created_at'),\n type: TableElementType.Date,\n },\n {\n key: 'contractLabel',\n label: this.translate.instant('pages.customer_details.history_contractlabel'),\n type: TableElementType.Link,\n modifier: (contractLabel, row) => {\n return {\n link: ['/ticket/v2/', row.contractCode],\n label: contractLabel,\n };\n },\n },\n ];\n\n //Validations v1\n public validationsTableConfig = [\n {\n key: 'ticketId',\n label: this.translate.instant('pages.validation.title_id'),\n type: TableElementType.Link,\n modifier: (ticketId: number) => {\n return {\n link: ['/ticket', ticketId],\n label: ticketId,\n };\n },\n },\n {\n key: 'itemLabel',\n label: this.translate.instant('pages.validation.title'),\n type: TableElementType.Text,\n },\n {\n key: 'occurredAt',\n label: this.translate.instant('pages.validation.date_time'),\n type: TableElementType.Date,\n format: 'medium',\n },\n {\n key: 'isTransfer',\n label: this.translate.instant('pages.ticket_details.connection'),\n type: TableElementType.Tag,\n modifier: (isTransfer: boolean) => (isTransfer ? 'TRUE' : 'FALSE'),\n config: {\n tags: {\n TRUE: { type: TagType.Success, label: this.translate.instant('otherslabels.yes') },\n FALSE: { type: TagType.Danger, label: this.translate.instant('otherslabels.no') },\n },\n },\n class: {\n row: 'tw-text-center',\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.validation.status'),\n type: TableElementType.Tag,\n modifier: (status) =>\n match(status)\n .when(\n (status) => ['EXPIRED', 'PENDING'].includes(status),\n () => status\n )\n .otherwise(() => null),\n config: {\n tags: {\n EXPIRED: {\n type: TagType.Danger,\n label: this.translate.instant('pages.validation.expired'),\n },\n PENDING: {\n type: TagType.Success,\n label: this.translate.instant('pages.validation.pending'),\n },\n null: {\n type: TagType.Info,\n label: null,\n },\n },\n },\n },\n {\n key: 'vehicleCode',\n label: this.translate.instant('pages.validation.vehicle'),\n type: TableElementType.Text,\n },\n {\n key: 'networkId',\n label: this.translate.instant('pages.validation.network'),\n type: TableElementType.Text,\n modifier: (networkId) => {\n if (!networkId) return;\n const networks = this.auth.networks.concat(this.$$networks);\n let network = networks.find((network) => +network.id === +networkId);\n return network?.name || `${networkId}`;\n },\n },\n {\n key: 'validationNetworkId',\n label: this.translate.instant('pages.validation.network_validation'),\n type: TableElementType.Text,\n modifier: (validationNetworkId) => {\n if (!validationNetworkId) return;\n const networks = this.auth.networks.concat(this.$$networks);\n const network = networks.find((network) => +network.id === +validationNetworkId);\n return network?.name || `${validationNetworkId}`;\n },\n },\n {\n key: 'provider',\n label: this.translate.instant('otherslabels.col_platform'),\n type: TableElementType.Image,\n modifier: (provider) =>\n match(provider)\n .with('tixipass', () => '/assets/img/logo_tixipass.png')\n .with('instantsystem', () => '/assets/img/providers/instant_system.png')\n .when(\n (provider) => ['sncf', 'smt_nfc', 'modalis'].includes(provider),\n (provider) => `/assets/img/providers/${provider}.png`\n )\n .otherwise(() => ''),\n config: { width: '66%' },\n },\n {\n key: 'tripOriginLabel',\n label: this.translate.instant('otherslabels.col_origin'),\n type: TableElementType.Text,\n },\n {\n key: 'tripDestinationLabel',\n label: this.translate.instant('otherslabels.col_destination'),\n type: TableElementType.Text,\n },\n {\n key: 'stopCode',\n label: this.translate.instant('pages.validation.stop_code'),\n type: TableElementType.Text,\n },\n {\n key: 'lineCode',\n label: this.translate.instant('pages.validation.line'),\n type: TableElementType.Text,\n },\n {\n key: 'mediaType',\n label: this.translate.instant('pages.validation.media_type'),\n type: TableElementType.Icon,\n modifier: (mediaType) =>\n match(mediaType)\n .when(\n (mediaType) => ['CARD', 'PASS', 'QR', 'APP'].includes(mediaType),\n () => mediaType\n )\n .otherwise(() => 'UNKNOWN'),\n config: {\n source: 'heroicons',\n labels: {\n CARD: this.translate.instant('pages.validators_validations.media_card'),\n PASS: this.translate.instant('pages.validators_validations.media_pass'),\n QR: this.translate.instant('pages.validators_validations.media_qr'),\n APP: this.translate.instant('pages.validators_validations.media_app'),\n UNKNOWN: this.translate.instant('pages.validators_validations.media_unknow'),\n },\n icons: {\n CARD: heroicons.outline.creditCard,\n PASS: heroicons.outline.ticket,\n QR: heroicons.outline.qrcode,\n APP: heroicons.outline.deviceMobile,\n UNKNOWN: heroicons.outline.questionMarkCircle,\n },\n },\n },\n {\n key: 'mediaId',\n label: this.translate.instant('pages.validation.media'),\n type: TableElementType.Text,\n },\n ];\n //Validations v2\n public walletEventsTableConfig = [\n {\n key: 'contractId',\n label: this.translate.instant('pages.validation.title_id'),\n type: TableElementType.Link,\n modifier: (contractId: number) => {\n return {\n link: ['/ticket', 'v2', contractId],\n label: contractId,\n };\n },\n class: { row: 'tw-text-xs' },\n },\n {\n key: 'contractId',\n label: this.translate.instant('pages.validation.title'),\n type: TableElementType.Text,\n modifier: (contractId: string) => {\n if (this.$$contracts.length > 0) {\n const contract = this.$$contracts.find((contract) => contract.id === contractId);\n return contract?.blueprintName || '-';\n }\n },\n },\n {\n key: 'occurredAt',\n label: this.translate.instant('pages.validation.date_time'),\n type: TableElementType.Date,\n format: 'medium',\n },\n {\n key: 'type',\n label: this.translate.instant('pages.validation.type'),\n type: TableElementType.Tag,\n modifier: (type: string) =>\n match(type)\n .when(\n (type) => ['TRANSFER', 'PUNCH'].includes(type),\n () => type\n )\n .otherwise(() => 'UNKNOWN'),\n config: {\n tags: {\n TRANSFER: {\n type: TagType.Success,\n label: this.translate.instant('pages.validation.transfer'),\n },\n PUNCH: { type: TagType.Danger, label: this.translate.instant('pages.validation.punch') },\n UNKNOWN: { type: TagType.Info, label: '' },\n },\n },\n },\n {\n key: 'occurredAt',\n label: this.translate.instant('pages.validation.offline'),\n type: TableElementType.Tag,\n modifier(occuredAt, walletEvent: CustomerWalletEvent) {\n const createdAtDate = new Date(walletEvent.createdAt);\n const occuredAtDate = new Date(occuredAt);\n\n return differenceInMinutes(createdAtDate, occuredAtDate) === 0 ? 'ONLINE' : 'OFFLINE';\n },\n config: {\n tags: {\n OFFLINE: {\n type: TagType.Success,\n label: this.translate.instant('otherslabels.yes'),\n },\n ONLINE: { type: TagType.Danger, label: this.translate.instant('otherslabels.no') },\n },\n },\n class: { row: 'tw-text-center' },\n },\n {\n key: 'contractNetworkId',\n label: this.translate.instant('pages.validation.network'),\n type: TableElementType.Text,\n modifier: (contractNetworkId) => {\n if (!contractNetworkId) return;\n const networks = this.auth.networks.concat(this.$$networks);\n const network = networks.find((network) => +network.id === +contractNetworkId);\n return network?.name || `${contractNetworkId}`;\n },\n },\n {\n key: 'contractId',\n label: this.translate.instant('otherslabels.col_origin'),\n type: TableElementType.Text,\n modifier: (contractId) => {\n const contract = this.$$contracts.find(({ id }) => id === contractId);\n\n if (contract && contract?.trip?.origin?.label) {\n return contract.trip.origin.label;\n }\n },\n },\n {\n key: 'contractId',\n label: this.translate.instant('otherslabels.col_destination'),\n type: TableElementType.Text,\n modifier: (contractId) => {\n const contract = this.$$contracts.find(({ id }) => id === contractId);\n\n if (contract && contract?.trip?.destination?.label) {\n return contract.trip.destination.label;\n }\n },\n },\n {\n key: 'location',\n label: this.translate.instant('pages.validation.stop'),\n type: TableElementType.Text,\n modifier: (location) => location?.stationName || location?.stationCode,\n },\n {\n key: 'location',\n label: this.translate.instant('pages.validation.line'),\n type: TableElementType.Text,\n modifier: (location) => location?.lineName || location?.lineCode,\n },\n {\n key: 'location',\n label: this.translate.instant('pages.validation.course'),\n type: TableElementType.Text,\n modifier: (location) => location?.course,\n },\n {\n key: 'location',\n label: this.translate.instant('pages.validation.direction'),\n type: TableElementType.Text,\n modifier: (location) => location?.tripDirection,\n },\n ];\n\n public showCustomerOrders: boolean = false;\n\n public customerOrders: {\n productId: string;\n productName: string;\n purchaseDate: string;\n price: number;\n orderId: string;\n orderCode: string;\n recipient: {\n id: string;\n firstName: string;\n lastName: string;\n };\n network: Network | string;\n status: CustomerOrderStatus;\n reservation: ArticleBooking | null;\n quantity: number;\n }[] = [];\n\n public customerOrdersTableConfig = [\n {\n key: 'productId',\n label: this.translate.instant('pages.customer_details.order_productid'),\n type: TableElementType.Link,\n modifier: (productId) => {\n return {\n link: ['/shop', 'products', 'edit', productId],\n label: productId,\n };\n },\n },\n {\n key: 'productName',\n label: this.translate.instant('pages.customer_details.order_productname'),\n type: TableElementType.Text,\n },\n {\n key: 'purchaseDate',\n label: this.translate.instant('pages.customer_details.order_purchasedate'),\n type: TableElementType.Date,\n format: 'short',\n },\n {\n key: 'quantity',\n label: this.translate.instant('pages.order.article_quantity'),\n type: TableElementType.Text,\n modifier: (quantity) => {\n return quantity ?? 1;\n },\n },\n {\n key: 'price',\n label: this.translate.instant('pages.order.article_unit_price'),\n type: TableElementType.Price,\n modifier: (price) => {\n const unitPrice = +(price || 0);\n\n return unitPrice;\n },\n currency: (article) => article.network?.currency || 'EUR',\n locale: () => this.translate.currentLang,\n },\n {\n key: 'price',\n label: this.translate.instant('pages.order.article_total_price'),\n type: TableElementType.Price,\n modifier: (price, article) => {\n const quantity = article?.quantity ?? 1;\n const unitPrice = +(price || 0);\n\n return quantity * unitPrice;\n },\n currency: (article) => article.network?.currency || 'EUR',\n locale: () => this.translate.currentLang,\n },\n {\n key: 'orderId',\n label: this.translate.instant('pages.customer_details.order_orderid'),\n type: TableElementType.Link,\n modifier: (orderId, article) => {\n return {\n link: ['/orders', orderId],\n label: article.orderCode,\n };\n },\n },\n {\n key: 'recipient',\n label: this.translate.instant('pages.customer_details.order_recipient'),\n type: TableElementType.Link,\n modifier: (recipient) => {\n return {\n link: recipient.id ? ['/customers', recipient.id] : null,\n label: `${recipient.firstName} ${recipient.lastName}`,\n };\n },\n },\n {\n key: 'network',\n label: this.translate.instant('pages.customer_details.order_network'),\n type: TableElementType.Text,\n modifier: (network) => {\n return network?.name || network;\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.status'),\n type: TableElementType.Tag,\n config: {\n tags: {\n COMPLETED: { type: TagType.Success, label: this.translate.instant('pages.order.ok') },\n ERROR: { type: TagType.Danger, label: this.translate.instant('pages.order.refused') },\n PENDING: { type: TagType.Info, label: this.translate.instant('pages.order.unfinished') },\n CANCELED: {\n type: TagType.Default,\n label: this.translate.instant('pages.order.cancelled'),\n },\n REGULARIZATION: {\n type: TagType.Default,\n label: this.translate.instant('pages.order.regularization'),\n },\n TO_REFUND: {\n type: TagType.Warning,\n label: this.translate.instant('pages.order.to_refund'),\n },\n REFUNDED: {\n type: TagType.Default,\n label: this.translate.instant('pages.order.refunded'),\n },\n PAYMENT_ISSUE: {\n type: TagType.Danger,\n label: this.translate.instant('pages.order.default_payment'),\n },\n PREAUTHORIZED: {\n type: TagType.Success,\n label: this.translate.instant('pages.order.preauthorized'),\n },\n WAITING_FOR_PAYMENT: {\n type: TagType.Warning,\n label: this.translate.instant('pages.order.waiting_for_payment'),\n },\n PROCESSING: {\n type: TagType.Warning,\n label: this.translate.instant('pages.order.processing'),\n },\n },\n },\n },\n {\n key: '',\n label: this.translate.instant('pages.order_details.product_type'),\n type: TableElementType.Text,\n modifier: (_, article) => {\n if (article.reservation) {\n return this.translate.instant('pages.order_details.booking');\n }\n\n return this.translate.instant('pages.order_details.default_product');\n },\n },\n {\n key: '',\n label: this.translate.instant('pages.customer_details.order_booking'),\n type: TableElementType.Text,\n modifier: (_, article) => {\n if (article.reservation) {\n return this.getHtmlBelongingCell(article.reservation);\n }\n\n return;\n },\n },\n ];\n\n public mediaTableConfig = [\n {\n key: 'code',\n label: this.translate.instant('pages.customer_details.card_number'),\n type: TableElementType.Text,\n },\n {\n key: 'token',\n label: this.translate.instant('pages.customer_details.media_id'),\n type: TableElementType.Text,\n },\n {\n key: 'card_type',\n label: this.translate.instant('pages.customer_details.card_type'),\n type: TableElementType.Text,\n modifier: (type: string) => {\n switch (type) {\n case 'CIPURSE':\n return 'CiPurse';\n case 'CALYPSO':\n return 'Calypso';\n case 'DESFIRE':\n return 'DesFire';\n case 'ACTOLL_MIFARE_CLASSIC':\n return 'Actoll - Mifare Classic';\n case 'ACTOLL_CALYPSO':\n return 'Actoll - Calypso';\n case 'DIRECT_PAYMENT_INGENICO_FRANCE':\n return 'Ingenico France';\n case 'OPEN_PAYMENT_INGENICO_OP2GO':\n return 'Ingenico OP2GO';\n case 'WORDLINE_TAP2USE':\n return 'WorldLine Tap2Use';\n\n default:\n return type || null;\n }\n },\n },\n {\n key: 'created_at',\n label: this.translate.instant('pages.customer_details.created_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY') : null;\n },\n },\n {\n key: 'expires_at',\n label: this.translate.instant('pages.network_details.expire_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date ? this.dateFormat.transform(date, 'DD-MM-YYYY') : null;\n },\n },\n ];\n\n public docTableConfig = [\n {\n key: 'name',\n label: this.translate.instant('pages.customer_details.document'),\n type: TableElementType.Text,\n style: (document) => {\n return document.status === 'EXPIRED' ? 'text-decoration: line-through' : '';\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.status'),\n type: TableElementType.Select,\n style: 'min-width: 160px;',\n config: {\n disabled: (value: string) => {\n return value === 'EXPIRED';\n },\n options: [\n {\n type: TagType.Success,\n label: this.translate.instant('pages.network_details.valid'),\n value: 'VALID',\n },\n {\n type: TagType.Danger,\n label: this.translate.instant('pages.network_details.invalid'),\n value: 'INVALID',\n },\n {\n type: TagType.Info,\n label: this.translate.instant('otherslabels.select_waiting'),\n value: 'PENDING',\n },\n {\n type: TagType.Danger,\n label: this.translate.instant('otherslabels.select_expired'),\n value: 'EXPIRED',\n disabled: true,\n },\n {\n type: TagType.Danger,\n label: this.translate.instant('otherslabels.select_processing'),\n value: 'VALIDATING',\n },\n ],\n },\n change: (status, doc: Document) => {\n try {\n this.paperworkService\n .updateDocument(doc.id, {\n status,\n })\n .subscribe(() => {\n this.notification.success(\n this.translate.instant('pages.customer_details.document_edit_notif')\n );\n });\n } catch {\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n }\n },\n },\n {\n key: 'expires_at',\n label: this.translate.instant('pages.network_details.expire_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details.download_label'),\n type: TableElementType.Function,\n modifier: (id: string, customerDocument: Document) => {\n return {\n icon:\n customerDocument.status !== 'EXPIRED' &&\n !customerDocument.deleted_at &&\n customerDocument.file_id\n ? 'icon-cloud-download'\n : '',\n fn: async () => {\n try {\n const url = await firstValueFrom(this.submissionService.getDocumentDetail(+id));\n\n if (!url) return;\n\n const link = document.createElement('a');\n link.href = url;\n link.download = customerDocument.name;\n link.target = '_blank';\n link.dispatchEvent(new MouseEvent('click'));\n } catch (error) {\n return;\n }\n },\n };\n },\n },\n {\n key: 'deleted_at',\n label: this.translate.instant('pages.customer_details.delete'),\n type: TableElementType.Function,\n modifier: (deleteDate: string, document: Document) => {\n return {\n icon: deleteDate || !document.file_id ? '' : 'trash-o',\n fn: this.openDeleteDocumentModal.bind(this),\n };\n },\n displayed: this.permissionGuard.isOnePermissionAllowed(['DELETE_SUBMISSION_DOCUMENT']),\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.update'),\n type: TableElementType.Function,\n modifier: (status: string, doc: any) => {\n return {\n icon: status === 'INVALID' && doc.file_id ? 'cloud-upload' : '',\n fn: () => {\n this.openUpdateDocumentModal(doc);\n },\n };\n },\n displayed: this.permissionGuard.isOnePermissionAllowed(['DELETE_SUBMISSION_DOCUMENT']),\n },\n ];\n\n public deletedDocTableConfig = [\n {\n key: 'name',\n label: this.translate.instant('pages.customer_details.document'),\n type: TableElementType.Text,\n style: (document) => {\n return document.status === 'EXPIRED' ? 'text-decoration: line-through' : '';\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.customer_details.status'),\n type: TableElementType.Tag,\n config: {\n tags: {\n VALID: {\n type: TagType.Success,\n label: this.translate.instant('pages.network_details.valid'),\n },\n INVALID: {\n type: TagType.Danger,\n label: this.translate.instant('pages.network_details.invalid'),\n },\n EXPIRED: {\n type: TagType.Danger,\n label: this.translate.instant('pages.customer_details.expired'),\n },\n PENDING: {\n type: TagType.Info,\n label: this.translate.instant('pages.customer_details.status_pending'),\n },\n },\n },\n },\n {\n key: 'file_deleted_at',\n label: this.translate.instant('pages.customer_details.deleted_at'),\n type: TableElementType.Text,\n modifier: (date: string) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n ];\n\n public submissionsTableConfig = [\n {\n key: 'id',\n label: 'Id',\n type: TableElementType.Link,\n modifier: (id) => {\n return {\n link: ['/submissions', id],\n label: id,\n };\n },\n },\n {\n key: 'doc_types',\n label: this.translate.instant('pages.customer_details.doc_types'),\n type: TableElementType.Text,\n },\n {\n key: 'product',\n label: this.translate.instant('pages.submissions.product'),\n type: TableElementType.Text,\n isSortable: true,\n },\n {\n key: 'status',\n label: this.translate.instant('pages.submissions.status'),\n type: TableElementType.Tag,\n config: {\n tags: {\n CANCELED: {\n type: TagType.Default,\n label: this.translate.instant('pages.submissions.cancelled'),\n },\n REJECTED: {\n type: TagType.Danger,\n label: this.translate.instant('pages.submissions.rejected'),\n },\n VALIDATED: {\n type: TagType.Success,\n label: this.translate.instant('pages.submissions.validated'),\n },\n PENDING: {\n type: TagType.Info,\n label: this.translate.instant('pages.submissions.pending'),\n },\n PROCESSING: {\n type: TagType.Warning,\n label: this.translate.instant('pages.submissions.processing'),\n },\n ON_HOLD: {\n type: TagType.Default,\n label: this.translate.instant('pages.submissions.on_hold'),\n },\n },\n },\n isSortable: true,\n },\n {\n key: 'date',\n label: this.translate.instant('pages.customer_details.created_at'),\n type: TableElementType.Date,\n exportable: true,\n format: 'mediumDate',\n isSortable: true,\n },\n {\n key: 'network_id',\n label: this.translate.instant('pages.submissions.network'),\n type: TableElementType.Text,\n modifier: (networkId: string) => {\n const network = this.auth.getNetwork(networkId);\n\n return network?.name ?? '-';\n },\n isSortable: true,\n },\n ];\n\n public voucherTableConfig = [\n {\n key: 'code',\n label: this.translate.instant('pages.customer_details.code'),\n type: TableElementType.Text,\n },\n {\n key: 'expire_at',\n label: this.translate.instant('pages.network_details.expire_at'),\n type: TableElementType.Text,\n modifier: (date) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n ];\n\n public subscriptionTableConfig = [\n {\n key: 'id',\n label: 'Id',\n type: TableElementType.Link,\n modifier: (subscriptionId: string) => {\n return {\n link: ['/subscriptions', subscriptionId],\n label: subscriptionId,\n };\n },\n },\n {\n key: 'order_id',\n label: this.translate.instant('pages.order.sale_id'),\n type: TableElementType.Link,\n modifier: (orderId: string) => {\n return {\n link: ['/orders', orderId],\n label: orderId,\n };\n },\n },\n {\n key: 'item_id',\n label: this.translate.instant('pages.subscriptions.product'),\n type: TableElementType.Link,\n modifier: (itemId: number) => {\n const product = this.$$products.find((product) => product.id === `${itemId}`);\n\n return {\n link: ['/shop/products/edit/', itemId],\n label: product?.name ?? itemId,\n };\n },\n },\n {\n key: 'created_at',\n label: this.translate.instant('pages.subscriptions.purchase_date'),\n type: TableElementType.Text,\n modifier: (date) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n {\n key: 'starting_at',\n label: this.translate.instant('pages.subscriptions.start_date'),\n type: TableElementType.Text,\n modifier: (date) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n {\n key: 'updated_at',\n label: this.translate.instant('pages.subscriptions.update'),\n type: TableElementType.Text,\n modifier: (date) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n {\n key: 'disabled_at',\n label: this.translate.instant('pages.subscriptions.termination_date'),\n type: TableElementType.Text,\n modifier: (date) => {\n return date\n ? this.dateFormat.transform(date, 'DD-MM-YYYY')\n : this.translate.instant(`pages.customer_details.undefined`);\n },\n },\n {\n key: 'termination_origin',\n label: this.translate.instant('pages.customer_details.termination_origin'),\n type: TableElementType.Tag,\n modifier: (terminationOrigin) => {\n if (!terminationOrigin) {\n return 'UNDEFINED';\n }\n\n return terminationOrigin;\n },\n config: {\n tags: {\n OPERATOR: {\n type: TagType.Info,\n label: this.translate.instant('pages.customer_details.operator'),\n },\n CUSTOMER: {\n type: TagType.Info,\n label: this.translate.instant('pages.customer_details.customer'),\n },\n UNDEFINED: {\n type: TagType.Default,\n label: this.translate.instant('pages.customer_details.undefined'),\n },\n SCRIPT: { type: TagType.Info, label: this.translate.instant('otherslabels.other') },\n },\n },\n },\n {\n key: 'network_id',\n label: this.translate.instant('pages.order.network'),\n type: TableElementType.Text,\n modifier: (id: string) => this.auth.networks.find((n) => +n.id === +id).name,\n },\n {\n key: 'id',\n label: this.translate.instant('pages.customer_details.action'),\n type: TableElementType.Submenu,\n config: {\n icon: heroicons.outline.dotsVertical,\n actions: [\n {\n label: this.translate.instant('pages.customer_details.unsubscribe'),\n fn: async (subscription) => {\n await this.disableSubscription(subscription.id);\n\n // Send notif\n const product = this.$$products.find(\n (product) => product.id === `${subscription.item_id}`\n );\n\n this.notification.success(\n this.translate.instant('pages.customer_details.unsubscribe_notif', {\n name: product.name,\n })\n );\n },\n active: (subscription) => {\n return !subscription.disabled_at;\n },\n },\n {\n label: this.translate.instant('pages.customer_details.cancel_unsubscribe'),\n fn: async (subscription) => {\n await this.subscriptionService.enableSubscription(subscription.id);\n\n // Update data\n // HACK: The table component doesn't update when only one element is updated, so I have to retrieve the list to update the table.\n // It's not pretty, but I couldn't find any other solution.\n this.retrieveSubscriptions(this.$$customer.customer_id);\n\n // Send notif\n const product = this.$$products.find(\n (product) => product.id === `${subscription.item_id}`\n );\n\n this.notification.success(\n this.translate.instant('pages.customer_details.cancel_unsubscribe_notif', {\n name: product.name,\n })\n );\n },\n active: (subscription) => {\n return subscription.disabled_at;\n },\n },\n ],\n },\n },\n ];\n\n public attributesTableConfig = [\n {\n key: 'label',\n label: this.translate.instant('pages.customer_details.description'),\n type: TableElementType.Text,\n modifier: (label: string) => this.translate.instant(label),\n },\n {\n key: 'network_id',\n label: this.translate.instant('pages.customer_details.network'),\n type: TableElementType.Text,\n displayed: () => this.attributeNetworks.length > 1,\n style: (attribute: Attribute) => (attribute.network_id === null ? 'opacity: 0.2;' : null),\n modifier: (networkId: string) => {\n if (!networkId) return this.translate.instant('pages.customer_details.global');\n const network = this.auth.getNetwork(networkId);\n if (!network) return 'Unknown'; // not supposed to happen\n\n return network.name;\n },\n },\n {\n key: 'key',\n label: this.translate.instant('pages.customer_details.value'),\n type: TableElementType.Text,\n style: (customerAttribute: Attribute) => {\n const attribute = this.$$attributes.find(\n (attribute) => attribute.key === customerAttribute.key\n );\n if (!attribute) return 'opacity: 0.2;';\n if (attribute.value === null) return 'opacity: 0.2;';\n if (attribute.value.length < 1) return 'opacity: 0.2;';\n\n return null;\n },\n modifier: (attributeKey: string) => {\n const attribute = this.$$attributes.find((attribute) => attribute.key === attributeKey);\n if (!attribute) return this.translate.instant('pages.customer_details.empty');\n if (attribute.value === null) return this.translate.instant('pages.customer_details.empty');\n if (attribute.value === false) return this.translate.instant('otherslabels.no');\n if (attribute.value === true) return this.translate.instant('otherslabels.yes');\n if (attribute.value.length < 1) {\n return this.translate.instant('pages.customer_details.empty');\n }\n\n return attribute.value;\n },\n },\n {\n key: 'key',\n label: this.translate.instant('pages.customer_details.edition'),\n type: TableElementType.Function,\n modifier: (attributeKey: string) => {\n const globalAttribute = this.$$allAttributes.find(\n (attribute) => attribute.key === attributeKey\n );\n\n return {\n icon: 'icon-pencil',\n fn: () => {\n const customerAttribute = this.$$attributes.find(\n (attribute) => globalAttribute.key === attribute.key\n );\n\n const attributeData: DialogData = {\n customerId: this.$$customer.customer_id,\n label: this.translate.instant(globalAttribute.label),\n value: customerAttribute ? customerAttribute.value : undefined,\n attributeId: globalAttribute.id,\n attributeType: globalAttribute.type,\n attributeValues:\n globalAttribute.type === 'BOOLEAN'\n ? [true, false]\n : globalAttribute.values?.sort((a, b) => a.localeCompare(b)).filter(Boolean),\n };\n\n this.openAttributesModal(attributeData);\n },\n };\n },\n displayed: this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_ATTRIBUTES']),\n },\n ];\n\n public sessionTableConfig = [\n {\n key: 'date',\n label: this.translate.instant('pages.customer_details.lastconnections_date'),\n type: TableElementType.Date,\n format: 'short',\n },\n {\n key: 'originType',\n label: this.translate.instant('pages.customer_details.lastconnections_origintype'),\n type: TableElementType.Text,\n },\n {\n key: 'originAgent',\n label: this.translate.instant('pages.customer_details.lastconnections_originagent'),\n type: TableElementType.Text,\n modifier: (originAgent: string) => {\n if (!originAgent) return null;\n\n // Get the OS info from the first part between parenthesis\n const regex = /\\((.*?)\\)/;\n const results = regex.exec(originAgent);\n if (!results?.length) return null;\n\n const osInfo = results[0].replace(/\\(|\\)/g, '');\n\n return osInfo;\n },\n },\n {\n key: 'originAgent',\n label: this.translate.instant('pages.customer_details.lastconnections_originname'),\n type: TableElementType.Text,\n modifier: (originAgent: string) => {\n if (!originAgent) return null;\n\n // Get the app info from the origin agent last part\n const regex = /\\([^)]*\\)/g;\n const userAgentWithoutParenthesisParts = originAgent.replace(regex, '');\n if (!userAgentWithoutParenthesisParts) return null;\n\n const userAgentParts = userAgentWithoutParenthesisParts\n .split(' ')\n .filter((part) => part !== '');\n if (!userAgentParts?.length) return null;\n\n const appInfo = userAgentParts[userAgentParts.length - 1];\n\n return appInfo;\n },\n },\n {\n key: 'providerId',\n label: this.translate.instant('pages.customer_details.provider'),\n type: TableElementType.Text,\n modifier: (providerId: number) => {\n const identity = this.identities.find((identity) => {\n return +identity.providerId === +providerId;\n });\n\n if (identity) {\n return identity.providerSlug;\n }\n\n return providerId;\n },\n },\n ];\n\n /**\n * Get a list of unique network Ids in visible attributes\n */\n get attributeNetworks(): any[] {\n return this.$$attributes.reduce((networks, attribute) => {\n const isGlobal = attribute.network_id === null;\n const isAlreadyCounted = networks.includes(attribute.network_id);\n\n if (isGlobal || isAlreadyCounted) {\n return networks;\n }\n\n return [...networks, attribute.network_id];\n }, []);\n }\n\n public get canBeAnonymized() {\n return this.$$customer?.active !== '1' && !this.$$customer?.anonymized_at;\n }\n\n public get isCustomerActiveOrAnonymized() {\n return this.$$customer.active === '1' || this.$$customer?.anonymized_at;\n }\n\n public get isCustomerActive() {\n return (\n ['true', '1', 1, true].includes(this.$$customer.active) && !this.$$customer?.anonymized_at\n );\n }\n\n public get isCustomerParent() {\n const profile = this.$$profiles.find(\n (profile) => profile.customer_id === this.$$customer.customer_id\n );\n return profile?.isParent;\n }\n\n public get areTicketsSelected() {\n if (!this.$$selectedTicketRows.length) {\n return false;\n }\n if (this.activeTicketsTab === 'v1' && this.$$selectedTicketRows[0].elem?.ticket_id) {\n return true;\n }\n if (this.activeTicketsTab === 'v2' && this.$$selectedTicketRows[0].elem?.blueprintId) {\n return true;\n }\n this.$$selectedTicketRows = [];\n return false;\n }\n\n public get operatorHasSameNetworks() {\n if (this.isSuperAdmin) return true;\n\n const networkIds = this.auth.networks.map(({ id }) => Number.parseInt(id, 10));\n\n return this.$$customer.network_ids.every((id) => {\n return networkIds.includes(id);\n });\n }\n\n public filterWalletContent(ticketsContracts: $$Ticket[] | Contract[]): $$Ticket[] | Contract[] {\n if (!ticketsContracts || ticketsContracts.length === 0) return null;\n\n const filterForm = this.filterForm.value;\n const isTicket = ticketsContracts[0]?.hasOwnProperty('network_id');\n const filterKey = isTicket ? 'network_id' : 'networkId';\n\n //Network filter\n const networkFilteredContent = this.keyPipe.transform(\n ticketsContracts as any[],\n filterKey,\n filterForm.customerNetwork\n );\n\n //Text search filter\n const searchFilteredContent = this.searchPipe.transform(\n networkFilteredContent,\n filterForm.customerQuery\n );\n\n //Status filter\n const statusFilteredContent = searchFilteredContent.filter((ticketContract) => {\n if (filterForm.status === null) return true;\n //Ticket\n if (isTicket) return ticketContract.status === filterForm.status;\n //Contract\n const status = this._contractStatus(ticketContract);\n return match(filterForm.status)\n .with('ACTIVE', () => status === CONTRACT_STATUS.ACTIVE)\n .with('PENDING', () => status === CONTRACT_STATUS.PENDING)\n .with('REMAINING', () => status === CONTRACT_STATUS.REMAINING)\n .with('EXPIRED', () => status === CONTRACT_STATUS.EXPIRED)\n .otherwise(() => false);\n });\n\n //Origin filter\n const originFilteredContent = statusFilteredContent.filter((ticketOrContract) => {\n if (filterForm.origin === null) return true;\n return match(filterForm.origin)\n .with('SHOP', () => !!!ticketOrContract.generationReason)\n .with('OPERATOR', () => !!ticketOrContract.generationReason)\n .otherwise(() => true);\n });\n\n return originFilteredContent;\n }\n\n private canceledStyle(style, ticket) {\n if (['CANCELED', 'CANCELLED'].includes(ticket.status)) return style;\n }\n\n private reactivateStyle(style, ticket) {\n if (!['CANCELED', 'CANCELLED'].includes(ticket.status)) return style;\n }\n\n private canceledContractStyle(style, contract) {\n if (!contract.blockedAt) return;\n\n const blockDate = new Date(contract.blockedAt);\n const now = new Date();\n\n if (blockDate < now) return style;\n }\n\n private reactivateContractStyle(style, contract) {\n if (!contract.blockedAt) return style;\n }\n\n private $$populateNetworks(networkIds: number[]) {\n const networks = networkIds\n .map((networkId) => {\n return this.auth.getNetwork(`${networkId}`);\n })\n .filter(Boolean);\n this.updateCustomerNetworks(networks);\n\n return networks;\n }\n\n async ngOnInit() {\n const userSession = await this.auth.session();\n this.isSuperAdmin = userSession?.user?.roles.some((role) => role.name === 'SUPER_ADMIN');\n\n // Whenever the route params change, we fetch the customer\n this.route.params.pipe(distinctUntilChanged()).subscribe((params) => {\n this.customerId = params.id;\n\n this.$$customer = null;\n this.$$loadingCustomer = true;\n\n this.$$profiles = [];\n this.$$loadingProfiles = true;\n\n this.$$wallet = [];\n this.$$filteredWallet = [];\n this.$$loadingWallet = true;\n this.$$errorWallet = false;\n this.$$errorContracts = false;\n this.$$selectedTicketRows = [];\n\n this.$$contracts = [];\n this.$$filteredContracts = [];\n this.$$loadingContracts = true;\n\n this.$$medias = [];\n this.$$loadingMedias = true;\n\n this.$$subscriptions = [];\n this.$$loadingSubscriptions = true;\n\n this.$$networks = [];\n this.$$isAddingNetwork = false;\n this.$$newNetworkId = '';\n this.$$newNetWorkLoading = false;\n this.$$customerNetworks = null;\n\n this.$$documents = [];\n this.$$deletedDocuments = [];\n this.$$loadingDocuments = true;\n\n this.$$submissions = [];\n this.$$loadingSubmissions = true;\n\n this.$$vouchers = [];\n this.$$loadingVouchers = true;\n\n this.$$allAttributes = [];\n this.$$loadingAllAttributes = true;\n\n this.$$attributes = [];\n this.$$loadingAttributes = true;\n\n this.$$comments = [];\n this.$$loadingComments = true;\n this.$$selectedCommentNetwork = '';\n\n this.$$products = [];\n this.$$loadingProducts = true;\n\n this.customerOrders = [];\n this.customer_history = [];\n\n this.$$validations = [];\n this.$$loadingValidations = true;\n this.$$walletEvents = [];\n this.$$loadingwalletEvents = true;\n\n this.computeBookingFilters();\n\n this.customersService\n .$$getCustomer(params.id)\n .then(async (customer) => {\n const allowedNetworks = customer.network_ids.filter((networkId) => {\n return this.auth.networks.some((network) => {\n return `${network.id}` === `${networkId}`;\n });\n });\n\n this.shopService.countNetworksProductTypes(allowedNetworks).then((countV1ProductType) => {\n this.isTicketV1Enabled = Boolean(countV1ProductType);\n });\n\n // Handle the case where the customer has a facebook profile picture\n const match = this.facebookRegex.exec(customer.picture);\n\n customer['fb_url'] = match ? `https://www.facebook.com/${match[1]}` : null;\n\n this.$$customer = customer;\n this.$$networks = this.$$populateNetworks(customer.network_ids);\n this.$$loadingCustomer = false;\n\n const allAttributes = await this.customersService.$$getGlobalAttributes();\n\n const attributes = allAttributes.filter((attribute) => {\n const isGlobal = attribute.network_id === null;\n if (isGlobal) return true;\n\n const isInCustomerNetworks = customer.network_ids.includes(+attribute.network_id);\n return isInCustomerNetworks;\n });\n\n this.$$allAttributes = attributes;\n this.$$loadingAllAttributes = false;\n\n // Orders history\n this.computeOrderHistoryFilter();\n\n this.loadingIdentities = true;\n this.customersService.identities(params.id).then(async (identities) => {\n this.loadingIdentities = false;\n this.identities = identities;\n\n // Sessions\n if (this.isSuperAdmin) {\n this.sessions = await this.customersService.getCustomerSessions(params.id);\n }\n });\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n });\n\n this.customersService\n .$$getProfiles(params.id)\n .then((profiles) => {\n this.$$profiles = profiles;\n\n this.retrieveSubscriptions(params.id);\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingProfiles = false;\n });\n\n this.customersService\n .$$getWallet(params.id)\n .then((tickets) => {\n this.$$wallet = tickets;\n this.$$filteredWallet = this.filterWalletContent(tickets) as $$Ticket[];\n this.computeContractsStats();\n this._updateCMSUserList().finally(() => {\n this.$$loadingWallet = false;\n });\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n this.$$errorWallet = true;\n }\n });\n\n if (this.permissionGuard.isOnePermissionAllowed(['ACCESS_PLATFORM'])) {\n this.customersService\n .getContracts(params.id)\n .then((contracts) => {\n this.$$contracts = contracts;\n this.$$filteredContracts = this.filterWalletContent(contracts) as Contract[];\n this.computeContractsStats();\n this._updateCMSUserList().finally(() => {\n this.$$loadingContracts = false;\n });\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n this.$$errorContracts = true;\n }\n });\n }\n\n this.customersService\n .$$getMedias(params.id)\n .then((medias) => {\n this.$$medias = medias;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingMedias = false;\n });\n\n this.retrieveDocuments(params.id);\n\n this.submissionService\n .$$getUserSubmissions(params.id)\n .then((submissions) => {\n this.$$submissions = submissions;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingSubmissions = false;\n });\n\n this.customersService\n .$$getVouchers(params.id)\n .then((vouchers) => {\n this.$$vouchers = vouchers;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingVouchers = false;\n });\n\n this.customersService\n .getCustomerAttributes(params.id)\n .then((attributes) => {\n this.$$attributes = attributes;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingAttributes = false;\n });\n\n this.customersService\n .$$getComments(params.id)\n .then((comments) => {\n this.$$comments = comments;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingComments = false;\n });\n\n this.customersService\n .getHistory(params.id)\n .then((transfersHistory) => {\n //Transfers stats\n for (const transfers of transfersHistory?.transfers) {\n if (transfers.recipientCustomerId !== transfers.issuerCustomerId) {\n transfers.direction = +transfers.recipientCustomerId === +params.id ? 'IN' : 'OUT';\n this.statusType[`TRANSFER_${transfers.direction}`].count++;\n let recipient = null;\n let issuer = null;\n for (const customer_id in transfersHistory.customers) {\n const customer = transfersHistory.customers[customer_id];\n if (!issuer && +customer.id === +transfers.issuerCustomerId) {\n issuer = customer;\n }\n if (!recipient && +customer.id === +transfers.recipientCustomerId) {\n recipient = customer;\n }\n if (recipient && issuer) {\n break;\n }\n }\n\n const issuerEmail = issuer.email ? `
    ${issuer.email}` : '';\n const recipientEmail = recipient.email ? `
    ${recipient.email}` : '';\n\n transfers.issuerCustomerLabel = issuer\n ? `${issuer.firstName} ${issuer.lastName} ${issuerEmail}`\n : '';\n transfers.recipientCustomerLabel = recipient\n ? `${recipient.firstName} ${recipient.lastName} ${recipientEmail}`\n : '';\n this.customer_history.push(transfers);\n }\n }\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n });\n\n //Validation v1\n this.customersService\n .getValidations(params.id)\n .then((validations) => {\n this.$$validations = validations.filter((validation) => {\n return this.auth.networks.some((n) => `${n.id}` === `${validation.networkId}`);\n });\n })\n .finally(() => {\n this.$$loadingValidations = false;\n });\n //Validation v2 (wallet events)\n this.customersService\n .getWalletEvents(params.id, [\n CustomerWalletEventType.PUNCH,\n CustomerWalletEventType.TRANSFER,\n ])\n .then((validations) => {\n this.$$walletEvents = validations;\n })\n .finally(() => {\n this.$$loadingwalletEvents = false;\n });\n });\n\n this.productsService\n .$$getProducts()\n .then((products) => {\n this.$$products = products;\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingProducts = false;\n });\n\n // this.translate.onLangChange.subscribe(this.setTranslation);\n }\n\n /**\n * This method is used whenever the wallet (v1) or the contracts (v2) tickets\n * are being retrieved. It's used to calculate the states of the differents tickets\n * and aggregate them into little statistics.\n */\n public computeContractsStats() {\n // Reset all the stats to 0\n for (const key of Object.keys(this.statusType)) {\n this.statusType[key].count = 0;\n }\n\n // Compute the stats\n for (const ticket of this.$$wallet) {\n if (!this.statusType[ticket.status]) continue;\n\n this.statusType[ticket.status].count++;\n }\n\n for (const contract of this.$$contracts) {\n const contractStatus = this._contractStatus(contract);\n\n switch (contractStatus) {\n case CONTRACT_STATUS.ACTIVE:\n this.statusType['ACTIVE'].count++;\n break;\n case CONTRACT_STATUS.PENDING:\n this.statusType['PENDING'].count++;\n break;\n case CONTRACT_STATUS.REMAINING:\n this.statusType['REMAINING'].count++;\n break;\n case CONTRACT_STATUS.CANCELED:\n case CONTRACT_STATUS.EXPIRED:\n this.statusType['EXPIRED'].count++;\n break;\n case CONTRACT_STATUS.EMPTY:\n default:\n break;\n }\n\n if (contract.transferredAt) {\n this.statusType['TRANSFER_OUT'].count++;\n }\n }\n }\n\n public async onChangeCommentNetwork(networkId: string | null) {\n this.$$selectedCommentNetwork = networkId;\n this.$$comments = await this.customersService.$$getComments(this.$$customer.customer_id, {\n networkId,\n });\n }\n\n private async updateWallet(customerId: string) {\n this.$$loadingWallet = true;\n\n const tickets = await this.customersService.$$getWallet(customerId);\n // Reset all the stats to 0\n for (const key of Object.keys(this.statusType)) {\n this.statusType[key].count = 0;\n }\n\n // Compute the stats\n for (const ticket of tickets) {\n if (!this.statusType[ticket.status]) continue;\n\n this.statusType[ticket.status].count++;\n }\n\n this.$$wallet = tickets;\n this.$$filteredWallet = this.filterWalletContent(tickets) as $$Ticket[];\n\n this.$$loadingWallet = false;\n }\n\n public async toggleCustomerStatus() {\n try {\n let isActive = ['true', '1', 1, true].includes(this.$$customer.active);\n\n if (isActive) {\n await this.customersService.deactivateCustomer(this.$$customer.customer_id);\n\n isActive = false;\n } else {\n await this.customersService.reactivateCustomer(this.$$customer.customer_id);\n\n isActive = true;\n }\n\n this.$$customer.active = `${isActive}`;\n\n this.$$profiles = this.$$profiles.map((profile) => {\n if (profile.customer_id === this.$$customer.customer_id) {\n // Boolean(0) === false\n // Boolean(1) === true\n return { ...profile, active: Boolean(isActive) };\n }\n\n return profile;\n });\n\n this.isActionOpen = false;\n } catch (error) {\n if (error?.error?.code === 'CUSTOMER_DEACTIVATION_ACTIVE_SUBSCRIPTION_INVALID') {\n this.notification.warn(\n this.translate.instant('pages.customer_details.error_tacit_active'),\n undefined,\n {\n timeOut: 10000,\n pauseOnHover: true,\n clickToClose: true,\n }\n );\n return;\n }\n this.notification.error(this.translate.instant('otherslabels.generic_error'));\n }\n }\n\n public postComment = async (comment: CommentFormValues) => {\n const requestBody = { text: comment.message, networkId: comment.networkId };\n const customerId = this.$$customer.customer_id;\n\n const request = await firstValueFrom(\n this.customersService.addCustomerComment(customerId, requestBody)\n );\n\n if (request.id) {\n this.$$comments = await this.customersService.$$getComments(customerId, {\n networkId: this.$$selectedCommentNetwork,\n });\n\n return { success: true };\n }\n\n return { success: false, errorMessage: request.errorMessage };\n };\n\n public editComment = async (comment: CommentFormValues) => {\n const requestBody = { text: comment.message, networkId: comment.networkId };\n const customerId = this.$$customer.customer_id;\n\n const request = await this.customersService.editCustomerComment(\n comment.id,\n customerId,\n requestBody\n );\n\n if (request.id) {\n this.$$comments = await this.customersService.$$getComments(customerId, {\n networkId: this.$$selectedCommentNetwork,\n });\n\n return { success: true };\n }\n\n return { success: false, errorMessage: request.errorMessage };\n };\n\n public deleteComment = async (id: number) => {\n const request = await this.customersService.deleteCustomerComment(String(id));\n const customerId = this.$$customer.customer_id;\n\n if (request === true) {\n this.$$comments = await this.customersService.$$getComments(customerId, {\n networkId: this.$$selectedCommentNetwork,\n });\n\n return { success: true };\n }\n\n return { success: false, errorMessage: request.errorMessage };\n };\n\n public async cancelTickets() {\n if (!this.areTicketsSelected) return;\n\n const confirmDelete = confirm(this.translate.instant('pages.customer_details.confirm_delete'));\n\n if (!confirmDelete) return;\n\n this.$$loadingWallet = true;\n let tickets = { success: [], failed: [] };\n if (this.activeTicketsTab === 'v1') {\n tickets = await this.ticketV1Cancel(this.$$selectedTicketRows);\n } else {\n tickets = await this.ticketV2Cancel(this.$$selectedTicketRows);\n }\n\n this.$$loadingWallet = false;\n if (tickets.failed.length) {\n return this.notification.error(\n this.translate.instant('pages.customer_details.error_delete_ticket', {\n failed: tickets.failed.length,\n })\n );\n }\n\n return this.notification.success(this.translate.instant(`otherslabels.notif_delete_ok`));\n }\n\n public openAnonymisationModal(): void {\n if (this.$$customer.active === '1' || this.$$customer.anonymized_at) return;\n\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(AnonymizationModalComponent, {\n width: '550px',\n data: {\n customerId: this.$$customer.customer_id,\n customerEmail: this.$$customer.email,\n customerAnonymized: this.$$customer.anonymized_at,\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n this.isActionOpen = false;\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public async sendResetPasswordEmail() {\n try {\n await this.customersService.sendResetPasswordEmail(this.$$customer.customer_id);\n\n this.notification.success(\n this.translate.instant('pages.customer_details.reset_password_email_sent')\n );\n } catch (error: any) {\n if (error?.error?.code && error.error.code === ERROR_FIREBASE_RESET_PASSWORD) {\n this.notification.error(\n this.translate.instant('pages.customer_details.error_reset_password_no_firebase'),\n undefined,\n {\n timeOut: 10000,\n }\n );\n } else {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n }\n } finally {\n this.isActionOpen = false;\n }\n }\n\n public openUserPhotoModal(isTicketPicture = false) {\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(ImageUpdateModalComponent, {\n width: '600px',\n disableClose: isTicketPicture,\n data: {\n customerId: this.$$customer.customer_id,\n picture: this.$$customer.picture,\n isTicketPicture,\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n });\n\n modalRef.componentInstance.updateField.subscribe((customer) => {\n this.$$customer = customer;\n });\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n\n return modalRef;\n }\n\n public openAttributesModal(attributeData: Record): void {\n const modalRef = this.modal.open(AttributesModalComponent, {\n width: '450px',\n data: attributeData,\n });\n\n modalRef.componentInstance.updateField.subscribe((attributes) => {\n this.$$attributes = attributes;\n\n // Probably force the refresh?\n if (this.$$allAttributes) {\n this.$$allAttributes = [...this.$$allAttributes];\n }\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public openInformationModal(): void {\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(InformationModalComponent, {\n width: '450px',\n maxHeight: '100vh',\n data: {\n isCustomerParent: this.isCustomerParent,\n customerId: this.$$customer.customer_id,\n firstname: this.$$customer.firstname,\n lastname: this.$$customer.lastname,\n birthday: this.$$customer.birthday,\n email: this.$$customer.email,\n phone: this.$$customer.phone,\n address: this.$$customer.address,\n hasFranceConnectProvider: this.hasFranceConnectProvider,\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n this.isActionOpen = false;\n });\n\n modalRef.componentInstance.updateField.subscribe((customer) => {\n this.$$customer = customer;\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n private async openUpdateDocumentModal(documentData) {\n const isDocStatusAcceptable = ['INVALID'].includes(documentData.status);\n if (!isDocStatusAcceptable) return;\n\n this.changeDetectorRef.detach();\n\n const modal = this.modal.open(UpdateDocumentModalComponent, {\n width: '600px',\n data: {\n documentTypeId: documentData.type_id,\n userId: this.$$customer.customer_id,\n },\n });\n\n const success = await firstValueFrom(modal.afterClosed());\n this.changeDetectorRef.reattach();\n\n if (success) {\n this.retrieveDocuments(+this.$$customer.customer_id);\n }\n }\n\n private openDeleteDocumentModal(documentData): void {\n const isDocStatusAcceptable = ['VALID', 'INVALID'].includes(documentData.status);\n if (!isDocStatusAcceptable) return;\n\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(DeleteDocumentModalComponent, {\n width: '600px',\n data: {\n documentId: documentData.id,\n documentTypeName: documentData.name,\n customerId: this.$$customer.customer_id,\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n });\n\n modalRef.componentInstance.updateField.subscribe((documents) => {\n this.handleFilterDocument(documents);\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public $$openNetworkDissociationModal(networkId: string): void {\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(NetworkDissociationModalComponent, {\n width: '450px',\n maxHeight: '100vh',\n data: {\n customerId: this.$$customer.customer_id,\n networkId,\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n });\n\n modalRef.componentInstance.updateField.subscribe((networkIds) => {\n this.$$networks = this.$$populateNetworks(networkIds);\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public async openTicketGenerationModal(ticket: $$Ticket = null) {\n this.changeDetectorRef.detach();\n\n let ticketNetwork = null;\n\n if (ticket?.network_id) {\n ticketNetwork = this.auth.getNetwork(ticket.network_id);\n }\n\n // TODO: Document what this does. It doesn't seem very clear on first read.\n const customerNetworks = this.$$customerNetworks.filter((customerNetwork) => {\n return this.activeTicketsTab === 'v1' || customerNetwork.blueprints.length > 0;\n });\n\n const modalRef = this.modal.open(TicketGenerationModalComponent, {\n width: '80vw',\n minHeight: '200px',\n data: {\n customer: this.$$customer,\n customerNetworks,\n version: this.activeTicketsTab,\n transfert: ticketNetwork ? { ticket, network_name: ticketNetwork?.name } : false,\n },\n });\n\n modalRef.componentInstance.desactivateTicketEvent.subscribe((ticket) => {\n modalRef.componentInstance.ticketGenerationInProcess = true;\n this._desactivateTicket(ticket).then(\n (isDeactivate) => {\n let optionStatus = null;\n switch (isDeactivate) {\n case true:\n optionStatus = 'OK';\n break;\n case false:\n optionStatus = 'KO';\n break;\n default:\n optionStatus = 'WARN';\n }\n modalRef.componentInstance.desactivateTicketOption = optionStatus;\n modalRef.componentInstance.ticketGenerationInProcess = false;\n },\n () => {\n modalRef.componentInstance.desactivateTicketOption = 'KO';\n modalRef.componentInstance.ticketGenerationInProcess = false;\n }\n );\n });\n\n modalRef.componentInstance.updateField.subscribe((customerData) => {\n const { customer, profiles } = customerData;\n\n this.$$customer = customer;\n this.$$profiles = profiles;\n this.updateWallet(customer.customer_id);\n this.updateWalletContracts(customer.customer_id);\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public openFusionModal() {\n this.changeDetectorRef.detach();\n\n const modalRef = this.modal.open(FusionModalComponent, {\n width: '80vw',\n minHeight: '200px',\n data: {\n customer: this.$$customer,\n attributes: {\n global: this.$$allAttributes,\n customer: this.$$attributes,\n },\n },\n });\n\n modalRef.afterClosed().subscribe(() => {\n this.changeDetectorRef.reattach();\n this.isActionOpen = false;\n });\n }\n\n public async $$saveNetwork(networkId: string) {\n this.$$newNetWorkLoading = true;\n\n try {\n const { networkIds } = await this.customersService.linkNetwork(\n Number.parseInt(this.$$customer.customer_id, 10),\n Number.parseInt(networkId, 10)\n );\n\n this.$$networks = this.$$populateNetworks(networkIds);\n\n this.$$newNetworkId = '';\n this.$$isAddingNetwork = false;\n\n this.notification.success(\n this.translate.instant('pages.customer_details.network_association_success_notification')\n );\n } catch (e) {\n switch (e.error?.code) {\n case 'NETWORK_FORBIDDEN':\n this.notification.error(\n this.translate.instant('pages.customer_details.network_forbidden')\n );\n break;\n case 'CUSTOMER_NOT_FOUND':\n this.notification.error(\n this.translate.instant('pages.customer_details.customer_not_found')\n );\n break;\n case 'CUSTOMER_NETWORK_DUPLICATE':\n this.notification.error(\n this.translate.instant('pages.customer_details.network_association_already_exists')\n );\n break;\n default:\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n }\n } finally {\n this.$$newNetWorkLoading = false;\n }\n }\n\n public resetForm() {\n this.filterForm.reset();\n this.$$filteredWallet = this.$$wallet;\n this.$$filteredContracts = this.$$contracts;\n }\n\n public resetNetworkFilter() {\n this.filterForm.controls.customerNetworkName.reset();\n this.filterForm.controls.customerNetwork.reset();\n }\n\n public isVerificationMailsent = false;\n\n public async sendVerificationMail() {\n this.isVerificationMailsent = true;\n\n setTimeout(() => {\n this.isVerificationMailsent = false;\n }, 5000);\n\n try {\n const success = await this.customersService.sendVerificationMail(this.$$customer.customer_id);\n if (!success) throw new Error('UNKNOWN ERROR');\n\n this.notification.success(this.translate.instant('pages.customer_details.verification_send'));\n } catch (error) {\n switch (error?.error?.code) {\n case 'CUSTOMER_ACCOUNT_ALREADY_VERIFIED':\n return this.notification.error(\n this.translate.instant('pages.customer_details.error_already_verified')\n );\n case 'CUSTOMER_NOT_FOUND':\n return this.notification.error(\n this.translate.instant('pages.customer_details.error_user_not_found')\n );\n case 'CUSTOMER_ACCOUNT_VERIFICATION_WAITING_TIME_NOT_REACHED':\n return this.notification.error(\n this.translate.instant('pages.customer_details.wait_for_send')\n );\n default:\n this.notification.error(\n this.translate.instant('pages.customer_details.error_verification_send')\n );\n break;\n }\n }\n }\n\n //Lost phone Modal\n public openLostPhoneModal() {\n const lostPhoneModal = this.lostPhoneModal.open(CustomerLostPhoneModalComponent, {\n width: '50vw',\n data: {\n title: this.translate.instant('pages.customer_details.declare_phone_lost'),\n text: `${this.translate.instant(\n 'pages.customer_details.lost_phone_disconnect'\n )}
    ${this.translate.instant(\n 'pages.customer_details.confirm_phone_lost'\n )}`,\n },\n });\n\n lostPhoneModal.afterClosed().subscribe((result) => {\n if (!result) return;\n this.disconnectUserSessions();\n });\n }\n\n //Lost Phone feature -- Disconnection\n private async disconnectUserSessions() {\n try {\n const response = await this.customersService.releaseSessions(this.$$customer.customer_id);\n if (response.success) {\n this.notification.success(\n this.translate.instant('pages.customer_details.delete_active_connections')\n );\n return;\n }\n } catch (e) {}\n this.notification.error(\n this.translate.instant('pages.customer_details.error_delete_active_connections')\n );\n }\n\n public _isTicketCanceled(ticket: Ticket) {\n return ['CANCELED', 'CANCELLED'].includes(ticket.status);\n }\n\n public _isTicketNotCanceled(ticket: Ticket) {\n return !['CANCELED', 'CANCELLED'].includes(ticket.status);\n }\n\n private async _desactivateTicket(ticket: $$Ticket) {\n if (\n !this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_WALLET']) ||\n ['CANCELED', 'CANCELLED'].includes(ticket.status)\n ) {\n return false;\n }\n\n const showSubscriptionCheckbox = Boolean(ticket.subscription_id);\n\n const dialogRef = this.modal.open(DeleteTicketModalComponent, {\n data: {\n label: ticket.label,\n showSubscriptionCheckbox,\n },\n });\n\n dialogRef.afterClosed().subscribe(async (result: DeleteTicketModalResponse) => {\n if (result.delete) {\n try {\n const promises: Promise[] = [this.ticketService.cancelTicket(+ticket.id)];\n\n if (showSubscriptionCheckbox && result?.deleteLinkedSubscription === true) {\n promises.push(\n this.disableSubscription(ticket.subscription_id, false).catch((error) => {\n // If the subscription is already disabled -> no error\n if (error === 'SUBSCRIPTION_ALREADY_DISABLED') {\n return;\n }\n throw new Error(error);\n })\n );\n }\n\n await Promise.all(promises);\n\n this.updateWallet(this.$$customer.customer_id);\n this.notification.success(this.translate.instant(`otherslabels.notif_delete_ok`));\n } catch {\n this.notification.error(this.translate.instant(`otherslabels.notif_delete_ko`));\n }\n }\n });\n }\n\n private _contractStatus(\n contract: Contract\n ): (typeof CONTRACT_STATUS)[keyof typeof CONTRACT_STATUS] {\n if (contract.blockedAt || contract.deletedAt) {\n return CONTRACT_STATUS.CANCELED;\n }\n\n if (!contract.state) {\n return CONTRACT_STATUS.EXPIRED;\n }\n\n const now = new Date();\n const contractStartDate = new Date(contract.state.contractStartDate);\n const contractEndDate = new Date(contract.state.contractEndDate);\n const validationEndDate = new Date(contract.state.validationEndDate);\n\n if (contract.state.contractEndDate !== null && compareAsc(contractEndDate, now) === -1) {\n return CONTRACT_STATUS.EXPIRED;\n }\n\n if (contract.state.contractStartDate !== null && compareAsc(contractStartDate, now) >= 0) {\n return CONTRACT_STATUS.REMAINING;\n }\n\n if (contract.state.validationStartDate === null) {\n return CONTRACT_STATUS.REMAINING;\n }\n\n if (contract.state.validationEndDate !== null && compareAsc(validationEndDate, now) >= 0) {\n return CONTRACT_STATUS.PENDING;\n }\n\n if (\n contract.state.validationRemainingPunches !== null &&\n contract.state.validationRemainingPunches <= 0\n ) {\n return CONTRACT_STATUS.EMPTY;\n }\n\n return CONTRACT_STATUS.ACTIVE;\n }\n\n private _isContractCanceled(contract: Contract) {\n return contract.blockedAt !== null;\n }\n\n private _isContractActive(contract: Contract) {\n return contract.blockedAt === null;\n }\n\n private async _activateContract(contract: Contract) {\n if (\n !this.permissionGuard.isOnePermissionAllowed(['MANAGE_CUSTOMER_WALLET']) ||\n this._isContractActive(contract)\n ) {\n return;\n }\n const activeContractResponse = await this.customersService.reactivateContract(contract.id);\n if (activeContractResponse) {\n this.updateWalletContracts(this.$$customer.customer_id);\n this.notification.success(this.translate.instant(`otherslabels.notif_reactivate_ok`));\n } else {\n this.notification.error(this.translate.instant(`otherslabels.notif_reactivate_ko`));\n }\n }\n\n private async _desactivateContract(contract: Contract) {\n const showSubscriptionCheckbox = Boolean(contract.subscriptionId);\n\n const dialogRef = this.modal.open(DeleteTicketModalComponent, {\n data: {\n label: contract.blueprintName,\n showSubscriptionCheckbox,\n },\n });\n\n dialogRef.afterClosed().subscribe(async (result: DeleteTicketModalResponse) => {\n if (result.delete) {\n const promises: Promise[] = [this.customersService.disableContract(contract.id)];\n\n if (showSubscriptionCheckbox && result?.deleteLinkedSubscription === true) {\n promises.push(\n this.disableSubscription(contract.subscriptionId, false).catch((error) => {\n // If the subscription is already disabled -> no error\n if (error === 'SUBSCRIPTION_ALREADY_DISABLED') {\n return;\n }\n throw new Error(error);\n })\n );\n }\n\n const responses = await Promise.all(promises);\n\n if (!responses[0]) {\n this.notification.error(this.translate.instant(`otherslabels.notif_delete_ko`));\n return;\n }\n\n this.updateWalletContracts(this.$$customer.customer_id);\n this.notification.success(this.translate.instant(`otherslabels.notif_delete_ok`));\n }\n });\n }\n\n private updateWalletContracts(customerId: string) {\n this.$$loadingContracts = true;\n this.customersService\n .getContracts(parseInt(customerId, 10))\n .then((contracts) => {\n this.$$contracts = contracts;\n this.$$filteredContracts = this.filterWalletContent(contracts) as Contract[];\n this.computeContractsStats();\n })\n .finally(() => {\n this.$$loadingContracts = false;\n });\n }\n\n //Update blueprints list for customer's networks\n private updateCustomerNetworks(networks: Network[]) {\n const networksIds = networks.map((network) => String(network.id));\n this.productsService.getNetworksBlueprints({ network: networksIds }).subscribe({\n next: (response) => {\n this.$$customerNetworks = this.$$networks.map<{\n network: Network;\n blueprints: Blueprint[];\n }>((network) => {\n return {\n network: network,\n blueprints: !response.filter\n ? []\n : response\n .filter((blueprint) => blueprint.networkId === String(network.id))\n .sort((blueprintA, blueprintB) => blueprintA.name.localeCompare(blueprintB.name)),\n };\n });\n },\n error: () => {\n // In the event of an error (e.g. no access to blueprints),\n // no blueprint is placed in customerNetworks.\n this.$$customerNetworks = this.$$networks.map<{\n network: Network;\n blueprints: Blueprint[];\n }>((network) => {\n return {\n network: network,\n blueprints: [],\n };\n });\n },\n });\n }\n\n private async ticketV1Cancel(selectedTickets) {\n const successfulTickets = [];\n const failedTickets = [];\n for (let ticket of selectedTickets) {\n if (['CANCELED', 'CANCELLED'].includes(ticket.elem.status)) continue;\n\n const deleteTicketResponse = await this.ticketService.cancelTicket(ticket.elem.id);\n\n if (deleteTicketResponse) {\n successfulTickets.push(ticket.elem.id);\n } else failedTickets.push(ticket.elem.id);\n }\n\n if (successfulTickets.length) {\n const tickets = await this.customersService.$$getWallet(this.$$customer.customer_id);\n // Reset all the stats to 0\n for (const key of Object.keys(this.statusType)) {\n this.statusType[key].count = 0;\n }\n\n // Compute the stats\n for (const ticket of tickets) {\n if (!this.statusType[ticket.status]) continue;\n\n this.statusType[ticket.status].count++;\n }\n\n this.$$wallet = tickets;\n this.$$filteredWallet = this.filterWalletContent(tickets) as $$Ticket[];\n }\n\n return {\n success: successfulTickets,\n failed: failedTickets,\n };\n }\n\n private async ticketV2Cancel(selectedContracts) {\n const successfulContracts = [];\n const failedContracts = [];\n for (let contract of selectedContracts) {\n if (contract.elem.blockedAt) continue;\n\n const deleteContractResponse = await this.customersService.disableContract(contract.elem.id);\n if (deleteContractResponse) {\n successfulContracts.push(contract.elem.id);\n } else {\n failedContracts.push(contract.elem.id);\n }\n }\n this.updateWalletContracts(this.$$customer.customer_id);\n\n return {\n success: successfulContracts,\n failed: failedContracts,\n };\n }\n\n private retrieveCustomerOrders(customerId: string, status?: CustomerOrderStatus | null) {\n this.customersService.getCustomerOrders(customerId, status).then((orders) => {\n this.customerOrders = [];\n if (!orders) return;\n\n for (const order of orders) {\n for (const article of order.articles) {\n if (!article?.recipient || +article.recipient.id === +this.$$customer.customer_id) {\n article.recipient = {\n id: null,\n lastName: this.$$customer.lastname,\n firstName: this.$$customer.firstname,\n };\n }\n\n const network = this.auth.networks.find((network) => +network.id == +order.networkId);\n\n this.customerOrders.push({\n productId: article.productId,\n productName: article.productName,\n purchaseDate: order.purchaseDate,\n price: article.price,\n orderId: order.id,\n orderCode: order.code,\n recipient: article.recipient,\n network: network || `${order.networkId}`,\n status: order.status,\n reservation: article.reservation,\n quantity: article.quantity,\n });\n }\n }\n\n if (!status) {\n this.showCustomerOrders = this.customerOrders.length > 0;\n }\n\n this.customerOrders.sort((orderA, orderB) =>\n orderA.purchaseDate > orderB.purchaseDate ? -1 : 1\n );\n });\n }\n\n public computeOrderHistoryFilter() {\n const { status } = this.filterOrderHistory.getRawValue();\n this.retrieveCustomerOrders(this.$$customer.customer_id, status);\n }\n\n private async retrieveSubscriptions(customerId: string): Promise {\n const parentProfile = this.$$profiles.find((profile) => profile.isParent);\n if (!parentProfile) return;\n\n this.$$loadingSubscriptions = true;\n\n const subscriptions = await this.customersService.$$getSubscriptions(\n parentProfile.customer_id,\n customerId\n );\n\n this.$$subscriptions = subscriptions;\n this.$$loadingSubscriptions = false;\n }\n\n private handleFilterDocument(documents: Document[]): void {\n const [activeDocuments, deletedDocuments] = documents.reduce(\n ([activeDocuments, deletedDocuments], document) => {\n // A document is considered \"deleted\" if :\n // - it has been soft-deleted\n // - its file has been soft-deleted\n // - it has no file\n if (document.deleted_at || document.file_deleted_at) {\n deletedDocuments.push(document);\n } else {\n activeDocuments.push(document);\n }\n\n return [activeDocuments, deletedDocuments];\n },\n [[], []]\n );\n\n this.$$documents = activeDocuments;\n this.$$deletedDocuments = deletedDocuments;\n }\n\n private async _updateCMSUserList() {\n //Get list of users from contracts and wallets\n const ticketsCmsUserIds: number[] = this.$$wallet.map((ticket) => +ticket.cms_user_id);\n const contractsCmsUserIds: number[] = this.$$contracts.map((contract) => +contract.operatorId);\n let cmsUserIds = [...new Set(ticketsCmsUserIds.concat(contractsCmsUserIds))];\n //Get user via API if needed\n for (const cmsUserId of cmsUserIds) {\n if (!cmsUserId || this._cmsUsers.find((user) => user.id === cmsUserId)) continue;\n const user = await firstValueFrom(this.userService.getUserDetail(cmsUserId));\n this._cmsUsers.push(user);\n }\n }\n\n private _walletOriginLabel(contractTicket: { contract?: Contract; ticket?: Ticket }): string {\n const { contract, ticket } = contractTicket;\n if (contract?.orderCode) return contract.orderCode;\n if (ticket?.order_id && ticket?.order_identifier) return ticket.order_identifier;\n\n let userId: string = null;\n if (contract?.operatorId) userId = contract.operatorId;\n if (ticket?.cms_user_id) userId = ticket.cms_user_id;\n\n if (!userId) return this.translate.instant('pages.customer_details.wallet_unknown_origin');\n\n const user = this._cmsUsers.find((user) => +user.id === +userId);\n const userLabel = !user\n ? `${this.translate.instant('pages.customer_details.user_id')}${userId}`\n : `${user.firstname} ${user.lastname}`;\n\n return this.translate.instant('pages.customer_details.ticket_generated_by', {\n label: userLabel,\n });\n }\n\n private _walletOriginLink(contractTicket: { contract?: Contract; ticket?: Ticket }): string[] {\n const { contract, ticket } = contractTicket;\n if (ticket?.order_id) return ['/orders', ticket.order_id];\n if (ticket?.cms_user_id) return ['/users', ticket.cms_user_id];\n if (contract?.orderId) return ['/orders', contract.orderId];\n if (contract?.operatorId) return ['/users', contract.operatorId];\n return null;\n }\n\n public get isCheckAllowed() {\n return (\n !!!this.$$customer.verified_at &&\n this.$$customer.active === '1' &&\n !!!this.$$customer.anonymized_at &&\n !this.$$loadingCheckAccount &&\n this.isCustomerParent\n );\n }\n\n public $$loadingCheckAccount = false;\n\n public async accountCheck() {\n this.$$loadingCheckAccount = true;\n const response = await this.customersService.checkAccount(this.$$customer.customer_id);\n this.$$loadingCheckAccount = false;\n this.changeDetectorRef.reattach();\n this.isActionOpen = false;\n if (response.success) {\n this.notification.success(\n this.translate.instant('pages.customer_details.account_check_validation_success')\n );\n this.$$customer.verified_at = new Date().toISOString();\n return;\n }\n this.notification.error(\n this.translate.instant('pages.customer_details.account_check_validation_error'),\n this.translate.instant(response?.error?.message)\n );\n }\n\n public computeBookingFilters(): void {\n const { customerNetwork, customerQuery } = this.filterForm.value;\n\n this.bookingFilters = {\n customerId: this.customerId,\n networkId: customerNetwork,\n query: customerQuery,\n };\n }\n\n private getHtmlBelongingCell(booking: ArticleBooking | undefined): string {\n if (booking) {\n return `\n

    ${this.translate.instant('pages.order_details.origin')} : ${\n booking.tripDepartureLocationLabel\n }

    \n

    ${this.translate.instant('pages.order_details.destination')} : ${\n booking.tripArrivalLocationLabel\n }

    \n

    ${this.translate.instant(\n 'pages.order_details.departure_date'\n )} : ${this.datePipe.transform(booking.tripDepartureDate, 'medium')}

    \n

    ${this.translate.instant(\n 'pages.order_details.arrival_date'\n )} : ${this.datePipe.transform(booking.tripArrivalDate, 'medium')}

    \n `;\n }\n }\n\n private async disableSubscription(subscriptionId: number, handleError: boolean = true) {\n await this.subscriptionService.disableSubscription(subscriptionId, handleError);\n\n // Update data\n // HACK: The table component doesn't update when only one element is updated, so I have to retrieve the list to update the table.\n // It's not pretty, but I couldn't find any other solution.\n this.retrieveSubscriptions(this.$$customer.customer_id);\n }\n\n public openDocumentUploadModal(): void {\n const modalRef = this.modal.open(DocumentUploadModalComponent, {\n width: '600px',\n data: {\n customerId: this.customerId,\n },\n });\n\n modalRef.afterClosed().subscribe((response) => {\n if (response?.success) {\n this.retrieveDocuments(+this.customerId);\n }\n });\n\n this.router.events.subscribe(() => {\n modalRef.close();\n });\n }\n\n public retrieveDocuments(customerId: number): void {\n this.customersService\n .$$getDocuments(`${customerId}`)\n .then(async (documents) => {\n this.handleFilterDocument(documents);\n })\n .catch(({ error }) => {\n switch (error.code) {\n case 'CUSTOMER_NOT_FOUND':\n return this.router.navigate(['/404']);\n case 'CUSTOMER_FORBIDDEN':\n return this.router.navigate(['/403']);\n default:\n return this.router.navigate(['/500']);\n }\n })\n .finally(() => {\n this.$$loadingDocuments = false;\n });\n }\n\n public openCreateFirebaseAccountModal(): void {\n if (this.hasFirebase || !this.hasAirwebProvider) {\n return;\n }\n\n const modalRef = this.modal.open(CreateFirebaseAccountModalComponent, {\n data: {\n customerId: this.customerId,\n },\n maxWidth: 600,\n });\n\n modalRef.afterClosed().subscribe(async (response) => {\n if (response?.needRefresh) {\n this.$$customer = await this.customersService.$$getCustomer(this.customerId);\n this.updateWallet(this.customerId);\n this.updateWalletContracts(this.customerId);\n this.retrieveCustomerOrders(this.customerId);\n }\n });\n }\n\n public openCreditProductModal(): void {\n const modalRef = this.modal.open(CreditProductModalComponent, {\n width: '80vw',\n panelClass: 'tw-overflow-y-auto',\n data: {\n customerId: this.customerId,\n networks: this.$$networks,\n documents: this.$$documents,\n },\n });\n\n modalRef.afterClosed().subscribe((response) => {\n if (response?.needRefresh) {\n this.updateWallet(this.customerId);\n this.updateWalletContracts(this.customerId);\n this.retrieveCustomerOrders(this.customerId);\n }\n });\n }\n}\n","export const ERROR_FIREBASE_RESET_PASSWORD = 'CUSTOMER_RESET_PASSWORD_FIREBASE_MISSING';\n","import { Component, OnInit } from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { Filters } from '@app/modules/shared/components/filter/filter.component';\nimport { TableElementType } from '@app/modules/shared/components/table/table.component';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { Observable } from 'rxjs';\nimport {\n debounceTime,\n distinctUntilChanged,\n filter,\n map,\n startWith,\n switchMap,\n tap,\n} from 'rxjs/operators';\nimport { Customer } from '../../models/customer';\nimport { CustomersService } from '../../services/customers.service';\n\n@Component({\n selector: 'tu-customer-search',\n templateUrl: './customer-search.component.html',\n providers: [CustomersService],\n})\nexport class CustomerSearchComponent implements OnInit {\n public feedContent: Observable;\n public customers: Customer[];\n public filtered: any;\n\n private _filter: Observable;\n set filter(filters: Observable) {\n this._filter = filters;\n\n if (!filters) return;\n this.feedContent = filters.pipe(\n startWith({ query: this.term, network: null }),\n tap(({ query }) => (this.term = query)),\n distinctUntilChanged((a, b) => a.query === b.query && a.network === b.network),\n map(({ query }) => query ?? ''),\n filter((query) => query.length >= 3),\n debounceTime(200),\n switchMap((query) => this.customersService.getSearchedCustomers(query)),\n map((r) => {\n r.forEach((el, i) => {\n if (\n el.fullname === this.term ||\n el.firstname === this.term ||\n el.lastname === this.term ||\n el.email === this.term ||\n el.phone === this.term\n ) {\n let tmp = r[i];\n r.splice(i, 1) && r.unshift(tmp);\n }\n });\n return r;\n })\n );\n }\n get filter(): Observable {\n return this._filter.pipe(startWith({ query: this.term, network: null }));\n }\n\n public term: string;\n\n private styles = {\n canceled: {\n text: 'color:gray !important; text-decoration:line-through; background-color:rgba(0, 0, 0,.1);',\n icon: 'color:transparent; pointer-events: none; background-color:rgba(0, 0, 0,.1);',\n tag: 'background-color: rgba(0, 0, 0,.1);',\n avatar:\n 'background: linear-gradient(to right, transparent, rgba(0, 0, 0,.1)); color:transparent; pointer-events: none;',\n origin:\n 'background: linear-gradient(to left, transparent, rgba(0, 0, 0,.1)); color:transparent; pointer-events: none;',\n },\n };\n\n tableConfig = [\n {\n key: 'picture',\n label: '',\n type: TableElementType.Avatar,\n imgDefault: '/assets/img/user.svg',\n style: this.canceledStyle.bind(this, this.styles.canceled.avatar),\n },\n {\n key: 'fullname',\n label: 'otherslabels.col_fullname',\n type: TableElementType.Text,\n isSortable: true,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'email',\n label: 'otherslabels.col_mail',\n type: TableElementType.Email,\n isSortable: true,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'date',\n label: 'pages.customer.signin_date',\n type: TableElementType.Date,\n format: 'mediumDate',\n isSortable: true,\n style: this.canceledStyle.bind(this, this.styles.canceled.text),\n },\n {\n key: 'phone',\n label: 'otherslabels.col_phone',\n type: TableElementType.Text,\n modifier: (phone) => (phone ? phone : 'otherslabels.unspecified'),\n isSortable: true,\n style: (customer) =>\n customer.phone\n ? this.canceledStyle(this.styles.canceled.text, customer)\n : 'font-style: italic; color: darkgray;',\n },\n {\n key: 'network_ids',\n label: 'otherslabels.col_networks',\n type: TableElementType.Text,\n modifier: (ids) => {\n let out = '';\n ids.forEach((id) => {\n out += this.authService.networks.find((n) => +n.id === +id).name + '
    ';\n });\n return out;\n },\n isSortable: true,\n displayed: this.authService.networks.length > 1,\n style: (customer) =>\n this.canceledStyle(this.styles.canceled.tag, customer) + 'font-size: .8em;',\n },\n {\n key: 'provider',\n label: 'otherslabels.col_origin',\n type: TableElementType.Image,\n modifier: (provider) => {\n if (provider?.toLowerCase().includes('instantsystem')) {\n return '/assets/img/providers/instant_system.png';\n }\n\n switch (provider) {\n case 'tixipass':\n return '/assets/img/logo_tixipass.png';\n case 'sncf':\n return '/assets/img/providers/sncf.png';\n default:\n return '';\n }\n },\n config: {\n width: '45px',\n },\n style: (customer) =>\n this.canceledStyle(this.styles.canceled.origin, customer) + 'text-align: center',\n isSortable: true,\n },\n ];\n\n constructor(\n private customersService: CustomersService,\n private authService: AuthService,\n private route: ActivatedRoute,\n private router: Router\n ) {}\n\n ngOnInit() {\n this.term = this.route.snapshot.paramMap.get('term');\n }\n\n private canceledStyle(style, { active }: Customer) {\n return active ? '' : style;\n }\n public navigateToCustomer(c: Customer): void {\n this.router.navigate(['/customers', c.customer_id]);\n }\n}\n","
    \n
    \n \n {{ 'pages.order.filters' | translate }}\n
    \n
    \n \n
    \n
    \n\n
    \n
    \n \n {{ 'pages.customers_search.search_results' | translate }}\n {{ term }}\n
    \n
    \n
    \n
    \n \n
    \n
    \n
    \n
    \n","\n
    \n {{ 'pages.customer.filters' | translate }}\n
    \n
    \n \n
    \n
    \n\n\n
    \n {{ 'pages.customer.customers_filters' | translate }}\n
    \n\n \n
    \n {{ 'pages.customer.customers_filters' | translate }}\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","import { NgModule } from '@angular/core';\nimport { RouterModule, Routes } from '@angular/router';\nimport { AuthGuard } from '@app/modules/shared/guards/auth.guard';\nimport { PermissionGuard } from '@app/modules/shared/guards/permission.guard';\nimport { CustomerDetailComponent } from './components/customer-detail/customer-detail.component';\nimport { CustomerSearchComponent } from './components/customer-search/customer-search.component';\nimport { CustomersComponent } from './components/customers.component';\nimport { ConfigGuard } from '../shared/guards/config.guard';\n\nconst routes: Routes = [\n {\n path: '',\n canActivate: [AuthGuard],\n canActivateChild: [PermissionGuard],\n data: {\n title: 'pages.list_menu.clients',\n permissions: {\n All: ['ACCESS_CUSTOMERS'],\n redirectTo: '/403',\n },\n },\n children: [\n {\n path: '',\n canActivate: [AuthGuard],\n component: CustomersComponent,\n data: {\n title: '',\n },\n },\n {\n path: 'search',\n canActivate: [AuthGuard, ConfigGuard],\n component: CustomerSearchComponent,\n data: {\n title: 'otherslabels.bc_search_client',\n features: ['globalSearch'],\n },\n },\n {\n path: ':id',\n canActivate: [AuthGuard],\n component: CustomerDetailComponent,\n data: {\n title: 'otherslabels.bc_client_detail',\n },\n },\n ],\n },\n];\n\n@NgModule({\n imports: [RouterModule.forChild(routes)],\n exports: [RouterModule],\n})\nexport class CustomersRoutingModule {}\n","import { Component, OnInit, ViewChild } from '@angular/core';\nimport { UntypedFormBuilder } from '@angular/forms';\nimport { TableServerComponent } from '@app/modules/shared/components/table-server/table-server.component';\nimport { ColumnConfig } from '@app/modules/shared/models/server-pagination';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { QueryParamsService } from '@app/modules/shared/services/queryParams.service';\nimport { ServerPaginationService } from '@app/modules/shared/services/server-pagination.service';\nimport { TranslateService } from '@ngx-translate/core';\nimport { Customer, CustomerFilters } from '../models/customer';\nimport { CustomersService } from '../services/customers.service';\n\n@Component({\n selector: 'tu-customers',\n templateUrl: './customers.component.html',\n styleUrls: ['./customers.component.scss'],\n providers: [CustomersService],\n})\nexport class CustomersComponent implements OnInit {\n constructor(\n private customersService: CustomersService,\n private filterService: FilterService,\n private authService: AuthService,\n private readonly translate: TranslateService,\n private readonly forms: UntypedFormBuilder,\n public serverPaginationService: ServerPaginationService,\n private queryParamsService: QueryParamsService\n ) {}\n\n @ViewChild('table', { static: true }) table: TableServerComponent;\n\n get shouldShowCustomerSearch(): boolean {\n return window['__CONFIG__']?.features?.customerSearch ?? true;\n }\n\n public filters: CustomerFilters = {};\n\n public readonly filtersForm = this.forms.group({\n attribute: [null],\n attribute_search: [null],\n virtual_provider: [null],\n });\n\n public computeLink(customer: Customer) {\n return `/customers/${customer.customer_id}`;\n }\n\n public get columnsConfig(): ColumnConfig[] {\n return [\n {\n key: 'customer_id',\n label: this.translate.instant('pages.customer.id'),\n type: 'text',\n },\n {\n key: 'picture',\n label: this.translate.instant('pages.customer.avatar'),\n type: 'image',\n modifier(_, row) {\n if (row.authLastAsGuest) {\n return '/assets/img/user-anonymous.svg';\n }\n },\n config: {\n alt: (_, row) => row.fullname,\n src(value: string) {\n if (!value) return '/assets/img/user.svg';\n return value;\n },\n },\n },\n {\n key: 'firstname',\n label: this.translate.instant('pages.customer.firstname'),\n type: 'text',\n },\n {\n key: 'lastname',\n label: this.translate.instant('pages.customer.lastname'),\n type: 'text',\n },\n {\n key: 'email',\n label: this.translate.instant('otherslabels.col_mail'),\n type: 'link',\n config: {\n href: (_, row) => `mailto:${row.email}`,\n },\n },\n {\n key: 'date',\n label: this.translate.instant('pages.customer.signin_date'),\n type: 'date',\n config: {\n format: 'dd/MM/yyyy',\n },\n },\n {\n key: 'authLastAsGuest',\n label: this.translate.instant('pages.customer.guest_mode'),\n type: 'badge',\n modifier(authLastAsGuest: boolean) {\n return authLastAsGuest ? 'YES' : 'NO';\n },\n config: {\n YES: { type: 'information', label: this.translate.instant('otherslabels.yes') },\n NO: { type: 'information', label: this.translate.instant('otherslabels.no') },\n },\n },\n {\n key: 'order_count',\n label: this.translate.instant('pages.customer.sales_number'),\n type: 'text',\n },\n {\n key: 'parent',\n label: this.translate.instant('pages.customer.main_account'),\n type: 'link',\n modifier: (parent: Customer) => {\n if (!parent) return this.translate.instant('otherslabels.main_account');\n return `${parent.firstname} ${parent.lastname}`;\n },\n config: {\n href: (_, row) => {\n if (row.parent) return `/customers/${row.parent.customer_id}`;\n },\n },\n },\n {\n key: 'network_ids',\n label: this.translate.instant('otherslabels.col_networks'),\n type: 'text',\n modifier: (ids: number[]) => {\n const networks = ids\n .map((id) => {\n const network = this.authService.networks.find((network) => +network.id === +id);\n\n return network?.name;\n })\n .filter(Boolean);\n\n return networks.join(', ');\n },\n isDisplayed: () => this.authService.networks.length > 1,\n },\n {\n key: 'virtual_provider',\n label: this.translate.instant('pages.customer.registration_platform'),\n type: 'image',\n modifier: (_, customer) => {\n if (customer.universal || !customer.network_ids.length) return 'tixipass';\n if (customer.provider?.toLowerCase().includes('instantsystem')) return 'instantsystem';\n return customer.provider;\n },\n config: {\n alt: (value: string) => value,\n src(value: string) {\n switch (value) {\n case 'tixipass':\n return '/assets/img/logo_tixipass.png';\n case 'sncf':\n return '/assets/img/providers/sncf.png';\n case 'smt_nfc':\n return '/assets/img/providers/smt_nfc.png';\n case 'modalis':\n return '/assets/img/providers/modalis.png';\n case 'instantsystem':\n return '/assets/img/providers/instant_system.png';\n default:\n return '';\n }\n },\n },\n isDisplayed: () => this.authService.networks.some((network) => network.universal),\n },\n ];\n }\n\n public computeFilters() {\n const globalFilters = this.filterService.filters;\n const { virtual_provider, attribute, attribute_search } = this.filtersForm.value;\n\n this.filters = {\n network_id: globalFilters.network,\n from: globalFilters.from,\n to: globalFilters.to,\n query: globalFilters.query,\n provider: virtual_provider,\n attribute,\n attribute_search,\n };\n\n this.queryParamsService.localFilterParams = this.filtersForm.value;\n }\n\n public attributes;\n\n get selectedAttribute() {\n const hasAttributes = Boolean(this.attributes);\n const hasSelectedAttribute = Boolean(this.filtersForm.value.attribute);\n\n if (!hasAttributes || !hasSelectedAttribute) return false;\n\n const attribute = this.attributes.find(\n (attribute) => +attribute.id === +this.filtersForm.value.attribute\n );\n\n return {\n ...attribute,\n values: attribute?.values?.sort((a: string, b: string) => a.localeCompare(b)) ?? undefined,\n };\n }\n\n get exportInformation() {\n const hasCheckedRows = this.table.selectedRows.length;\n\n const checkedRowsIds = hasCheckedRows\n ? this.table.selectedRows.map((row) => row.customer_id)\n : [];\n\n const customersFilters = this.filters;\n const hasSelectedAttribute = customersFilters.attribute && customersFilters.attribute_search;\n\n const filters = {};\n\n if (customersFilters.provider) filters['provider'] = customersFilters.provider;\n\n if (hasSelectedAttribute) {\n filters['attributeId'] = customersFilters.attribute.toString();\n filters['attributeValue'] = customersFilters.attribute_search.toString();\n }\n\n return {\n exportId: 'cms-customers-v2',\n checkedRowsIds: checkedRowsIds,\n hasCheckedAll: this.table.allSelected,\n filters: filters,\n };\n }\n\n ngOnInit() {\n if (!this.shouldShowCustomerSearch) {\n this.filterService.reset();\n }\n\n const params = this.queryParamsService.localFilterParams;\n\n const formParams: Record = {};\n\n for (const property of Object.keys(this.filtersForm.value)) {\n formParams[property] = params[property] || null;\n }\n\n this.filtersForm.setValue(formParams);\n\n this.computeFilters();\n\n this.filtersForm.controls.attribute.valueChanges.subscribe((value) => {\n const attribute = this.attributes.find((a) => a.id === value);\n this.filtersForm.controls.attribute_search.setValue(attribute?.values?.[0] || null);\n });\n\n this.customersService.getAttributes().then((attributes) => {\n this.attributes = attributes;\n });\n }\n}\n","import { CommonModule } from '@angular/common';\nimport { NgModule } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatAutocompleteModule } from '@angular/material/autocomplete';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatDialogModule } from '@angular/material/dialog';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatStepperModule } from '@angular/material/stepper';\nimport { MatChipsModule } from '@angular/material/chips';\nimport { RouterModule } from '@angular/router';\nimport { SharedModule } from '@app/modules/shared/shared.module';\nimport { NgbModule } from '@ng-bootstrap/ng-bootstrap';\nimport { SimpleNotificationsModule } from 'angular2-notifications';\nimport { MomentModule } from 'ngx-moment';\nimport { NgxSelectModule } from 'ngx-select-ex';\nimport { NetworkService } from '../network/services/network.service';\nimport { AnonymizationModalComponent } from './components/customer-detail/anonymization-modal/anonymization-modal.component';\nimport { AttributesModalComponent } from './components/customer-detail/attributes-modal/attributes-modal.component';\nimport { ImageUpdateModalComponent } from './components/customer-detail/avatar-update-modal/avatar-update-modal.component';\nimport { CustomerDetailComponent } from './components/customer-detail/customer-detail.component';\nimport { InformationModalComponent } from './components/customer-detail/information-modal/information-modal.component';\nimport { CustomerLostPhoneModalComponent } from './components/customer-detail/lost-phone-modal/customer-lost-phone-modal.component';\nimport { NetworkDissociationModalComponent } from './components/customer-detail/network-dissociation-modal/network-dissociation-modal.component';\nimport { TicketGenerationModalComponent } from './components/customer-detail/ticket-generation-modal/ticket-generation-modal.component';\nimport { CustomerSearchComponent } from './components/customer-search/customer-search.component';\nimport { FusionModalComponent } from './components/customer-detail/fusion-modal/fusion-modal.component';\n// Components\nimport { CustomersComponent } from './components/customers.component';\n// Modules\nimport { CustomersRoutingModule } from './customers.routing';\n// Services\nimport { CustomersService } from './services/customers.service';\nimport { CustomersFiltersService } from './services/filters.service';\nimport { ProductsService } from './services/products.service';\nimport { TicketService } from './services/ticket.service';\nimport { PaperworkService } from './services/paperwork.service';\nimport { SubscriptionsService } from '@app/modules/subscriptions/services/subscriptions.service';\nimport { UsersService } from '../users/services/users.service';\nimport { DeleteTicketModalComponent } from './components/customer-detail/delete-ticket-modal/delete-ticket-modal.component';\nimport { ShopService } from './services/shop.service';\nimport { DocumentUploadModalComponent } from './components/customer-detail/document-upload-modal/document-upload-modal.component';\nimport { CustomerCommentComponent } from './components/customer-detail/customer-comment/customer-comment.component';\nimport { CreateFirebaseAccountModalComponent } from './components/customer-detail/create-firebase-account-modal/create-firebase-account-modal.component';\nimport { CreditProductModalComponent } from './components/customer-detail/credit-product-modal/credit-product-modal.component';\n\n@NgModule({\n declarations: [\n CustomersComponent,\n CustomerDetailComponent,\n CustomerSearchComponent,\n AnonymizationModalComponent,\n AttributesModalComponent,\n InformationModalComponent,\n ImageUpdateModalComponent,\n NetworkDissociationModalComponent,\n TicketGenerationModalComponent,\n CustomerLostPhoneModalComponent,\n FusionModalComponent,\n DeleteTicketModalComponent,\n DocumentUploadModalComponent,\n CustomerCommentComponent,\n CreateFirebaseAccountModalComponent,\n CreditProductModalComponent,\n ],\n imports: [\n CustomersRoutingModule,\n SharedModule,\n CommonModule,\n FormsModule,\n ReactiveFormsModule,\n RouterModule,\n MomentModule,\n NgxSelectModule,\n MatAutocompleteModule,\n MatButtonModule,\n MatDialogModule,\n MatFormFieldModule,\n MatInputModule,\n MatSelectModule,\n MatStepperModule,\n MatChipsModule,\n NgbModule,\n SimpleNotificationsModule,\n ],\n providers: [\n CustomersService,\n ProductsService,\n TicketService,\n CustomersFiltersService,\n NetworkService,\n PaperworkService,\n SubscriptionsService,\n UsersService,\n ShopService,\n ],\n})\nexport class CustomersModule {}\n","
    \n \n
    \n
    \n \n En cours d'édition\n
    \n \n \n \n \n \n \n \n \n
    \n \n
    \n \n \n \n \n \n \n
    \n
    \n
    \n \n
    \n {{ comment?.login }}\n \n
    \n \n \n {{ 'pages.order_details.pending' | translate }}\n \n \n {{ 'pages.order_details.completed' | translate }}\n \n \n {{ 'pages.order_details.canceled' | translate }}\n \n \n {{ 'pages.order_details.error' | translate }}\n \n \n {{ 'pages.order.regularization' | translate }}\n \n \n {{ 'pages.order.preauthorized' | translate }}\n \n \n {{ 'pages.order.waiting_for_payment' | translate }}\n \n \n {{ 'pages.order.to_refund' | translate }}\n \n \n {{ 'pages.order.refunded' | translate }}\n \n \n
    \n \n
    \n \n \n {{ 'otherslabels.unspecified' | translate }}\n \n \n {{ 'pages.order_details.to_process' | translate }}\n \n \n {{ 'pages.order_details.processed' | translate }}\n \n \n {{ 'pages.order_details.autoprocessed' | translate }}\n \n \n {{ 'pages.order_details.processing' | translate }}\n \n \n
    \n
    \n \n
    \n","import { Component, Input } from '@angular/core';\nimport { OrderProcessStatus, OrderStatus } from '../../../models/order';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { TranslateService } from '@ngx-translate/core';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\n\n@Component({\n selector: 'tu-order-comment',\n templateUrl: './order-comment.component.html',\n styleUrls: ['./order-comment.component.scss'],\n})\nexport class OrderCommentComponent {\n public heroicons = heroicons;\n public OrderProcessStatus = OrderProcessStatus;\n public OrderStatus = OrderStatus;\n\n @Input()\n public comment: any;\n\n @Input()\n public commentToEdit: (id: number, message: string) => void;\n\n @Input()\n public editedComment: string;\n\n @Input()\n public commentToDelete: (id: number) => void;\n\n @Input()\n public markdownToHTML: (markdown: string) => string;\n\n constructor(public translate: TranslateService, public authService: AuthService) {}\n}\n","
    \n
    \n
    \n
    \n \n
    \n {{ 'pages.order_details.sale_id' | translate }}\n {{ order.order_identifier }}\n
    \n\n
    \n \n \n \n
    \n \n {{ status.label | translate }}\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 {{ 'pages.order_details.order_date_label' | translate }}\n

    \n\n \n \n {{ order.date | date : 'longDate' : undefined : translate.currentLang }}\n \n\n -\n\n \n {{ order.date | date : 'shortTime' : undefined : translate.currentLang }}\n \n \n
    \n\n \n

    \n {{ 'pages.order_details.checkout_date_label' | translate }}\n

    \n\n \n \n {{ latestCheckout.createdAt | date : 'longDate' : undefined : translate.currentLang }}\n\n -\n\n \n {{ latestCheckout.createdAt | date : 'shortTime' : undefined : translate.currentLang }}\n \n
    \n
    \n\n \n

    \n
    \n \n {{ order.user.firstname }} {{ order.user.lastname }}\n \n \n {{ order.user.email }}\n \n\n \n {{ order.user.customer_id }}\n \n
    \n\n \n

    \n\n
    \n {{ 'pages.order.origin' | translate }} : {{ order.origin_type }}\n
    \n\n \n {{ 'pages.order_details.checkout_btn_label' | translate }}\n \n
    \n
    \n\n \n
    \n {{ 'pages.order_details.process_title' | translate }}\n
    \n\n \n \n \n \n
    \n \n {{ status.label | translate }}\n \n
    \n \n \n
    \n\n \n
    \n\n \n \n {{ 'pages.order.autoprocessed' | translate }}\n \n \n \n \n \n\n
    \n
    \n \n {{ 'pages.order_details.products' | translate }}\n\n \n {{ getPaymentMethodLabel(order.payment_method) | translate }}\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 {{ 'pages.order_details.order_products' | translate }}\n
    {{ 'pages.order_details.products' | translate }}{{ 'pages.order_details.quantity' | translate }}{{ 'pages.order_details.unit_price' | translate }}\n {{ 'pages.order_details.sub_total' | translate }}\n {{ 'pages.order_details.recipient' | translate }}{{ 'pages.order_details.voucher' | translate }}{{ 'pages.order.delivery' | translate }}{{ 'pages.order_details.product_type' | translate }}{{ 'pages.order_details.booking_details' | translate }}
    \n
    \n

    {{ item.title }}

    \n

    {{ 'pages.order_details.origin' | translate }} : {{ getProductTrip(item.id).origin.label }}

    \n

    {{ 'pages.order_details.destination' | translate }} : {{ getProductTrip(item.id).destination.label }}

    \n

    {{ 'pages.order_details.via' | translate }} : {{ getProductTrip(item.id).via.label }}

    \n
    \n \n

    \n {{ 'pages.order_details.start_date' | translate }} :\n \n {{\n item.startingAt | date : 'MMMM' : undefined : translate.currentLang\n }}\n \n

    \n
      \n
    • \n \n {{ product.origin_name }} -\n {{\n product.origin_at\n | date : 'shortDate' : undefined : translate.currentLang\n }}\n {{\n product.origin_at\n | date : 'shortTime' : undefined : translate.currentLang\n }}\n ⟶\n {{ product.destination_name }} -\n {{\n product.destination_at\n | date : 'shortDate' : undefined : translate.currentLang\n }}\n {{\n product.destination_at\n | date : 'shortTime' : undefined : translate.currentLang\n }}\n \n \n {{ 'pages.ticket_details.number_seats' | translate }} :\n {{ product.seats }}\n \n \n {{ 'pages.ticket_details.number_seats_prm' | translate }} :\n {{ product.seats_prm }}\n \n
    • \n
    \n
    {{ item.qty }}\n {{\n item.price_wo_vat\n | currency\n : curr\n : 'symbol-narrow'\n : '1.2-2'\n : this.translate.currentLang\n }}\n \n 0; else elseBlock\">\n {{\n getDiscountVatPrice(item)\n | currency\n : curr\n : 'symbol-narrow'\n : '1.2-2'\n : this.translate.currentLang\n }}\n \n \n \n {{\n item.price_w_vat * item.qty\n | currency\n : curr\n : 'symbol-narrow'\n : '1.2-2'\n : this.translate.currentLang\n }}\n \n \n \n \n {{ item.user.customer_id }} - {{ item.user.fullname }}\n \n \n 0\">\n \n {{ item.voucher }}\n \n\n \n {{ getDiscountSourceItem(item.id) }}\n \n \n \n
    \n \n \n \n
    \n
    \n {{ (getProductBooking(item.id) ? 'pages.order_details.booking' : 'pages.order_details.default_product') | translate }}\n \n
    \n

    {{ 'pages.order_details.origin' | translate }} : {{ getProductBooking(item.id).tripDepartureLocationLabel }}

    \n

    {{ 'pages.order_details.destination' | translate }} : {{ getProductBooking(item.id).tripArrivalLocationLabel }}

    \n

    {{ 'pages.order_details.departure_date' | translate }} : {{ getProductBooking(item.id).tripDepartureDate | date: 'medium' }}

    \n

    {{ 'pages.order_details.arrival_date' | translate }} : {{ getProductBooking(item.id).tripArrivalDate | date: 'medium' }}

    \n
    \n
    \n
    \n
    \n
    \n {{ field.label }}\n
    \n \n \n {{ getValue(field.values, field.value_id) }}\n \n \n {{ formatFieldValue(field.type, field.value) }}\n \n \n {{ 'otherslabels.label_not_specified' | translate }}\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 {{ 'pages.order_details.use_voucher' | translate }}\n \n\n
      \n
    • {{ order.voucher }}
    • \n
    • {{ discount.sourceCode }}
    • \n
    \n
    \n\n \n \n\n \n \n \n \n \n \n\n \n \n \n \n \n\n \n 0\">\n \n \n \n \n\n \n \n \n \n \n
    \n {{ 'pages.order_details.order_footer' | translate }}\n
    {{ 'pages.order_details.order_footer_label' | translate }}{{ 'pages.order_details.order_footer_value' | translate }}
    \n {{ 'pages.order_details.tax_free_total' | translate }} :\n \n {{\n order.total_base\n | currency : curr : 'symbol-narrow' : '1.2-2' : this.translate.currentLang\n }}\n
    \n {{ 'pages.order_details.vat' | translate }} ({{ v[0] }}%)\n \n {{\n v[1]\n | currency\n : curr\n : 'symbol-narrow'\n : '1.2-2'\n : this.translate.currentLang\n }}\n
    \n {{ 'pages.order_details.total' | translate }} :\n \n {{\n order.total_vat\n | currency : curr : 'symbol-narrow' : '1.2-2' : this.translate.currentLang\n }}\n
    \n
    \n
    \n
    \n\n \n
    \n \n\n
    \n
    \n
    \n {{ 'pages.order_details.shipping_address' | translate }}\n
    \n
    \n

    \n {{\n order.shipping_address?.streetNumber\n ? order.shipping_address?.streetNumber + ' ' + order.shipping_address?.route\n : order.shipping_address?.route\n }}\n

    \n

    \n {{ order.shipping_address?.zipCode }} {{ order.shipping_address?.city }}\n

    \n

    {{ order.shipping_address?.country }}

    \n
    \n
    \n
    \n\n
    \n
    \n
    \n {{ 'pages.order_details.payment_plan_title' | translate }}\n
    \n
    \n \n \n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n
    \n {{ 'pages.order_details.payment_plan_title' | translate }}\n
    {{ 'pages.order_details.payment_plan' | translate }}{{ 'pages.order_details.amount' | translate }}
    \n {{ inst.date | date : 'shortDate' : undefined : translate.currentLang }}\n {{ inst.amount | currency : curr : 'symbol-narrow' : '1.2-2' }}
    \n
    \n
    \n
    \n \n \n \n \n \n \n \n\n\n\n","import {\n AfterViewChecked,\n Component,\n ElementRef,\n OnInit,\n ViewChild,\n ViewEncapsulation,\n} from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl, UntypedFormGroup, Validators } from '@angular/forms';\nimport { ActivatedRoute } from '@angular/router';\nimport { Option } from '@app/modules/shared/components/modal/modal.component';\nimport { CommentFormValues } from '@app/modules/shared/models/comment';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { TranslateService } from '@ngx-translate/core';\nimport { NotificationType, NotificationsService } from 'angular2-notifications';\nimport { firstValueFrom } from 'rxjs';\nimport { Comment } from '../../models/comment';\nimport { Item } from '../../models/item';\nimport {\n Order,\n OrderProcessStatus,\n OrderStatus,\n OrderDiscountType,\n OrderDiscount,\n Checkout,\n SchoolOrderArticle,\n Trip,\n ArticleBooking,\n} from '../../models/order';\nimport { OrdersService } from '../../services/orders.service';\nimport { match } from 'ts-pattern';\nimport { ModalService } from '@app/modules/shared/services/modal.service';\n\n@Component({\n selector: 'tu-order-detail',\n templateUrl: './order-detail.component.html',\n styleUrls: ['./order-detail.component.scss'],\n providers: [OrdersService],\n encapsulation: ViewEncapsulation.None,\n})\nexport class OrderDetailComponent implements OnInit, AfterViewChecked {\n @ViewChild('scrollBottom', { static: false }) private myScrollContainer: ElementRef;\n\n private version = 0; // No cache manager, because old repeatWhen don't increment the cache version\n private initialValues = {};\n\n public OrderStatus = OrderStatus;\n public OrderProcessStatus = OrderProcessStatus;\n\n // Provide access to the heroicons to the HTML template\n public heroicons = heroicons;\n\n private id = +this.route.snapshot.params['id'];\n\n public edit = {};\n public order: Order;\n public fieldForm: UntypedFormGroup;\n public articles: SchoolOrderArticle[] = [];\n\n get curr() {\n if (!this.order) return 'EUR';\n if (!this.authService.networks) return 'EUR';\n const network = this.authService.networks.find((n) => +n.id === +this.order.network_id);\n return network?.currency ?? 'EUR';\n }\n\n public comments: Comment[];\n public user_id = this.authService.user_id;\n public datePipeFormat: any;\n public zeroVat = true;\n public vats: {} = {};\n public entries = Object.entries;\n\n public get hasClickAndCollectItems(): boolean {\n return this.order?.items?.some((item) => item.support === 'COLLECT') ?? false;\n }\n\n public get hasHomeDeliveryItems(): boolean {\n return this.order?.items?.some((item) => item.support === 'SHIPPING') ?? false;\n }\n\n public handleOrderAction(event: Event) {\n const select = event.currentTarget;\n const isHTMLSelectElement = select instanceof HTMLSelectElement;\n\n if (!isHTMLSelectElement) {\n (select as HTMLSelectElement).value = 'NOOP';\n return this.notification.error(this.translate.instant('otherslabels.generic_error'));\n }\n\n const value = (select as HTMLSelectElement).value;\n if (value === 'NOOP') return;\n\n const email = this.order?.user?.email ?? '';\n\n const orderId = this.order?.order_identifier ?? '';\n\n const message = match(value)\n .with('SHIPPING', () => {\n return {\n title: this.translate.instant('pages.order_details.mail_shipping_title'),\n body: [\n this.translate.instant('pages.order_details.mail_shipping_line_1'),\n ``,\n this.translate.instant('pages.order_details.mail_shipping_line_2', { nb: orderId }),\n `${this.translate.instant(\n 'pages.order_details.mail_shipping_line_3'\n )} : <${this.translate.instant('pages.order_details.mail_shipping_to_complete')}>.`,\n `${this.translate.instant(\n 'pages.order_details.mail_shipping_line_4'\n )} : <${this.translate.instant('pages.order_details.mail_shipping_to_complete')}>.`,\n ``,\n `${this.translate.instant('pages.order_details.mail_shipping_line_5')},`,\n `<${this.translate.instant('pages.order_details.mail_shipping_to_complete')}>`,\n ].join('%0D%0A'),\n };\n })\n .with('COLLECT', () => {\n return {\n title: `${this.translate.instant('pages.order_details.mail_cc_title')},`,\n body: [\n this.translate.instant('pages.order_details.mail_shipping_line_1'),\n ``,\n this.translate.instant('pages.order_details.mail_shipping_line_2', { nb: orderId }),\n this.translate.instant('pages.order_details.mail_cc_line_3'),\n ``,\n `<${this.translate.instant('pages.order_details.mail_shipping_to_complete')}>.`,\n ``,\n `${this.translate.instant('pages.order_details.mail_shipping_line_5')},`,\n `<${this.translate.instant('pages.order_details.mail_shipping_to_complete')}>`,\n ].join('%0D%0A'),\n };\n })\n .run();\n\n const anchor = Object.assign(document.createElement('a'), {\n target: '_blank',\n href: `mailto:${email}?subject=${message.title}&body=${message.body}&content-type=text/html`,\n });\n\n document.body.appendChild(anchor);\n anchor.click();\n\n (select as HTMLSelectElement).value = 'NOOP';\n\n anchor.remove();\n }\n\n statusSelectOptions: Option[] = [\n { label: 'pages.order_details.completed', value: OrderStatus.completed },\n { label: 'pages.order.preauthorized', value: OrderStatus.preauthorized },\n { label: 'pages.order_details.pending', value: OrderStatus.pending },\n { label: 'pages.order_details.canceled', value: OrderStatus.canceled },\n { label: 'pages.order_details.error', value: OrderStatus.error },\n { label: 'pages.order.regularization', value: OrderStatus.regularization },\n { label: 'pages.order.waiting_for_payment', value: OrderStatus.waiting_for_payment },\n { label: 'pages.order.to_refund', value: OrderStatus.to_refund },\n { label: 'pages.order.refunded', value: OrderStatus.refunded },\n { label: 'pages.order.default_payment', value: OrderStatus.defaultPayment },\n ];\n selectedStatus: any;\n\n processStatusSelectOptions: Option[] = [\n { label: 'otherslabels.unspecified', value: 'NULL' },\n { label: 'pages.order_details.processed', value: OrderProcessStatus.processed },\n { label: 'pages.order_details.to_process', value: OrderProcessStatus.to_process },\n { label: 'pages.order_details.processing', value: OrderProcessStatus.processing },\n ];\n selectedProcessStatus: any;\n\n paymentMethods = {\n CARD: { type: 'info', label: 'pages.order.card' },\n TRANSFER: { type: 'warning', label: 'pages.order.transfer' },\n DEPOSIT: { type: 'success', label: 'pages.order.deposit' },\n CASH: { type: 'success', label: 'pages.order.cash' },\n };\n\n public notificationOptions = {\n timeOut: 5000,\n showProgressBar: true,\n pauseOnHover: false,\n clickToClose: false,\n };\n\n public checkouts: Checkout[] = [];\n\n // Get the latest checkout based on `createdAt` date\n public get latestCheckout(): Checkout | null {\n if (!this.checkouts?.length) return null;\n\n return this.checkouts\n .filter((checkout) => {\n // We only want to keep checkout from on operator\n return checkout.initiatorType === 'OPERATOR';\n })\n .reduce((latest, checkout) => {\n if (!latest) return checkout;\n\n return new Date(checkout.createdAt) > new Date(latest.createdAt) ? checkout : latest;\n });\n }\n\n public get showFooterHeader(): boolean {\n return (\n [OrderStatus.preauthorized, OrderStatus.completed].includes(this.order.status) &&\n this.order.process_status !== 'NULL'\n );\n }\n\n constructor(\n private orderService: OrdersService,\n private route: ActivatedRoute,\n private fb: UntypedFormBuilder,\n private notification: NotificationsService,\n private translate: TranslateService,\n private authService: AuthService,\n private modalService: ModalService\n ) {}\n\n async ngOnInit() {\n await this.fetchOrderDetails();\n this.translate.onLangChange.subscribe(() => this.setTranslation());\n this.datePipeFormat = this.translate.instant('date_format.date_format');\n }\n\n async fetchOrderDetails() {\n // TODO: Use getOrderDetails instead of getLegacyOrderDetails -> Currently missing infos in getOrderDetails\n this.orderService.getLegacyOrderDetails(this.id).then((order) => {\n const group = {};\n\n for (const item of order.items || []) {\n if (!item.fields.length) continue;\n\n const tmp = {};\n\n for (const field of item.fields) {\n tmp[field.field_id] = new UntypedFormControl(\n field.value || field.value_id,\n !!+field.mandatory ? Validators.required : []\n );\n }\n\n group[item.id] = this.fb.group(tmp);\n }\n\n /**\n * [\n * {\n * \"vat\": \"10.00\",\n * \"price\": 0.1\n * }\n * ]\n *\n * ===>\n *\n * {\n * \"10.00\": 0.1\n * }\n */\n const vats = order.vats.reduce(\n (vats, { vat, price }) => ({ ...vats, [vat]: (vats[vat] || 0) + price }),\n {}\n );\n\n const hasVatsOverZero = Object.keys(vats).some((vat) => parseFloat(vat) > 0);\n\n this.order = order;\n this.selectedStatus = order.status;\n this.selectedProcessStatus = order.process_status;\n\n this.vats = vats;\n this.zeroVat = !hasVatsOverZero;\n\n this.fieldForm = this.fb.group(group);\n this.initialValues = this.fieldForm.value;\n });\n\n this.orderService.getOrderDetails(this.id).then((order) => {\n this.checkouts = order.checkouts;\n this.articles = order.articles;\n });\n\n firstValueFrom(this.orderService.getOrderComments(this.id, this.version)).then((comments) => {\n this.comments = comments;\n });\n }\n\n ngAfterViewChecked() {\n this.scrollToBottom();\n }\n\n scrollToBottom(): void {\n try {\n this.myScrollContainer.nativeElement.scrollTop =\n this.myScrollContainer.nativeElement.scrollHeight;\n } catch (err) {}\n }\n\n public getVats(o: Order): Array {\n return Object.keys(o.vats);\n }\n\n public getValue(values: any[], id: any) {\n const option = values.find((val: { id: any }) => val.id === id);\n return option ? option.value : '-';\n }\n\n public formatFieldValue(type: 'DATE' | 'INPUT', value: string) {\n if (type === 'INPUT') return value.trim();\n\n try {\n return new Date(value).toLocaleDateString('fr-FR', {\n day: '2-digit',\n month: '2-digit',\n year: 'numeric',\n });\n } catch {\n return value.trim();\n }\n }\n\n public getPaymentMethodConfig(paymentMethod) {\n const configs = {\n CARD: { type: 'information', label: this.translate.instant('pages.order.card') },\n TRANSFER: { type: 'warning', label: this.translate.instant('pages.order.transfer') },\n DEPOSIT: { type: 'success', label: this.translate.instant('pages.order.deposit') },\n CASH: { type: 'success', label: this.translate.instant('pages.order.cash') },\n };\n return configs[paymentMethod];\n }\n\n public getPaymentMethodLabel(paymentMethod: string) {\n const paymentMethodConfig = this.paymentMethods[paymentMethod];\n\n return paymentMethodConfig?.label;\n }\n\n public getPaymentMethodCssClass(paymentMethod: string) {\n const paymentMethodConfig = this.paymentMethods[paymentMethod];\n\n return paymentMethodConfig?.type;\n }\n\n public async saveEdition(fieldItem) {\n const fields = this.fieldForm.controls[fieldItem.id].value;\n\n try {\n for (const [field, value] of Object.entries(fields)) {\n const articleId = fieldItem.fields.find((articleField) => {\n return articleField.field_id === field;\n }).order_item_id;\n\n await this.orderService.updateArticleFieldValue(\n this.order.id,\n articleId,\n fieldItem.id,\n field,\n value\n );\n }\n\n let i = this.order.items.findIndex((item) => item.id === fieldItem.id);\n\n if (typeof i !== 'undefined') {\n this.order.items[i].fields.forEach((field, f) => {\n if (field.type === 'SELECT') {\n this.order.items[i].fields[f].value_id =\n this.fieldForm.value[fieldItem.id][field.field_id];\n } else {\n this.order.items[i].fields[f].value =\n this.fieldForm.value[fieldItem.id][field.field_id];\n }\n });\n this.edit[fieldItem.id] = false;\n }\n } catch {\n this.notification.error(this.translate.instant('pages.customer_details.generic_error'));\n }\n }\n\n public cancelEdition(item_id: string) {\n this.fieldForm.controls[item_id].reset(this.initialValues[item_id]);\n this.edit[item_id] = false;\n }\n\n public isInvoiceDownloading = false;\n\n public async printOrder(options: { legacy: boolean; version?: string } = { legacy: false }) {\n const anchor = document.createElement('a');\n\n try {\n this.isInvoiceDownloading = true;\n\n const orderId = options.legacy ? this.order.order_identifier : this.order.id;\n const invoiceFile = await firstValueFrom(this.orderService.getOrderInvoice(orderId, options));\n\n if (invoiceFile instanceof Blob) {\n Object.assign(anchor, {\n style: `display: none;`,\n href: URL.createObjectURL(invoiceFile),\n download: `Commande_N°${this.order.order_identifier}.pdf`,\n });\n\n document.body.appendChild(anchor);\n anchor.click();\n } else {\n const { default: html2pdf } = await import('html2pdf.js');\n\n html2pdf()\n .from(invoiceFile)\n .set({\n margin: 0,\n filename: `Commande_N°${this.order.order_identifier}.pdf`,\n image: { type: 'jpeg', quality: 0.98 },\n html2canvas: { scale: 2, useCORS: true },\n jsPDF: { unit: 'in', format: 'a4', orientation: 'p' },\n })\n .save();\n }\n\n this.notification.success(\n this.translate.instant('pages.order_details.invoice_download_success')\n );\n } catch (error) {\n console.error(error);\n this.notification.error(this.translate.instant(`otherslabels.notif_download_ko`));\n } finally {\n anchor.remove();\n this.isInvoiceDownloading = false;\n }\n }\n\n public async updateStatus(status: OrderStatus) {\n const response$ = this.orderService.updateStatus(this.order.id, status);\n\n const response = await firstValueFrom(response$);\n this.order.status = response.status;\n\n return response.status;\n }\n\n public updateProcessStatus(process_status: OrderProcessStatus) {\n this.orderService.updateProcessStatus(Number(this.order.id), process_status).subscribe((r) => {\n this.order.process_status = r.process_status;\n this.notification.info(this.translate.instant(`otherslabels.update_done`));\n });\n }\n\n private setTranslation() {\n setTimeout(() => {\n this.datePipeFormat = this.translate.instant('date_format.date_format');\n });\n }\n\n public getCssClass(status: OrderStatus) {\n const classes = {\n COMPLETED: 'success',\n PENDING: 'info',\n CANCELED: 'warning',\n ERROR: 'danger',\n PROCESSED: 'success',\n PROCESSING: 'warning',\n TO_PROCESS: 'primary',\n REGULARIZATION: 'reg',\n PREAUTHORIZED: 'success',\n WAITING_FOR_PAYMENT: 'waiting_payment',\n TO_REFUND: 'primary',\n REFUNDED: 'refunded',\n PAYMENT_ISSUE: 'danger',\n NULL: 'default',\n };\n\n return classes[status];\n }\n\n public supportTitle(item) {\n const support = this.determineSupport(item);\n\n const titles = {\n DEMATERIALIZED: this.translate.instant('pages.order_details.support_demat'),\n PHYSICAL: this.translate.instant('pages.order_details.support_physical'),\n MEDIA: this.translate.instant('pages.order_details.support_media'),\n COLLECT: this.translate.instant('pages.order_details.support_collect'),\n SHIPPING: this.translate.instant('pages.order_details.support_shipping'),\n };\n\n return titles[support];\n }\n\n public supportIcon(item) {\n const support = this.determineSupport(item);\n\n const icons = {\n DEMATERIALIZED: heroicons.outline.deviceMobile,\n PHYSICAL: heroicons.outline.creditCard,\n MEDIA: heroicons.outline.cash,\n COLLECT: heroicons.outline.cursorClick,\n SHIPPING: heroicons.outline.home,\n };\n\n try {\n return icons[support];\n } catch {\n return icons['DEMATERIALIZED'];\n }\n }\n\n public postComment = async (comment: CommentFormValues) => {\n const request = await firstValueFrom(\n this.orderService.postOrderComment(this.id, comment.message)\n );\n\n if (request.success) {\n this.comments = await firstValueFrom(\n this.orderService.getOrderComments(this.id, this.version)\n );\n }\n\n return request;\n };\n\n public editComment = async (comment: CommentFormValues) => {\n const request = await firstValueFrom(\n this.orderService.editOrderComment(+comment.id, comment.message)\n );\n\n if (request.success) {\n this.comments = await firstValueFrom(\n this.orderService.getOrderComments(this.id, this.version)\n );\n }\n\n return request;\n };\n\n public deleteComment = async (id: number) => {\n const request = await firstValueFrom(this.orderService.deleteOrderComment(id));\n\n if (request.success) {\n this.comments = await firstValueFrom(\n this.orderService.getOrderComments(this.id, this.version)\n );\n }\n\n return request;\n };\n\n /**\n * Get the order item support for which it was bought.\n * If the support isn't set (newer feature) we fallback\n * on the old value which is `dematerialized` 1 or 0\n */\n public determineSupport(item: Item) {\n if (item.support) return item.support;\n\n return item.dematerialized === '1' ? 'DEMATERIALIZED' : 'PHYSICAL';\n }\n\n public getDiscountVatPrice(item: Item): number {\n if (item.discounted_qty > 0) {\n const discountList = this.getDiscountItemList(item.id);\n\n if (discountList.length) {\n let totalPrice = 0;\n let promoQty = 0;\n\n // For each discount we calculate the price of the product and his quantity\n for (const discount of discountList) {\n const { promo, qty } = this.extractDiscountValue(discount.value);\n\n // adds the price with the centimes promotional quantity\n if (discount.type === OrderDiscountType.ARTICLE_AMOUNT) {\n // promo is in centimes and price_w_vat in euros\n totalPrice += (item.price_w_vat - promo / 100) * qty;\n }\n\n // adds the price with the ratio promotional quantity\n if (discount.type === OrderDiscountType.ARTICLE_RATIO) {\n totalPrice += item.price_w_vat * (1 - promo) * qty;\n }\n\n promoQty += qty;\n }\n\n // adds the price with the rest of the quantity\n totalPrice += item.price_w_vat * (item.qty - promoQty);\n\n return totalPrice;\n }\n }\n\n // Default return without discount\n return item.price_w_vat * item.qty;\n }\n\n public getDiscountSourceItem(itemId: string): string {\n return this.getDiscountItemList(itemId)\n .map(({ sourceCode }) => sourceCode)\n .join(', ');\n }\n\n private getDiscountItemList(itemId: string): OrderDiscount[] {\n const regex = new RegExp(`@${itemId}`);\n return this.order.discounts.filter(({ value }) => regex.test(value));\n }\n\n private extractDiscountValue(value: string): { promo: number; id: number; qty: number } {\n // The value is construct like this \"Price@ItemId×Qty\"\n // Warning × is a special character this is normal\n const [promo, id, qty] = value.split(/[@×]/);\n\n return {\n promo: parseFloat(promo),\n id: parseInt(id),\n qty: parseInt(qty),\n };\n }\n\n public async openCheckoutModalConfirmation() {\n try {\n const choice = await this.modalService.open(\n this.translate.instant('pages.order_details.checkout_modal_title'),\n '',\n [\n {\n label: this.translate.instant('pages.order_details.checkout_modal_cancel'),\n value: 'CANCEL',\n },\n {\n label: this.translate.instant('pages.order_details.checkout_modal_confirm'),\n value: 'CONFIRM',\n },\n ]\n );\n\n // If the user doesn't confirm the action we don't do anything\n if (choice !== 'CONFIRM') return;\n\n const success = await this.orderService.checkoutOrder(this.order.id);\n\n const message = match(success)\n .with(true, () => ({\n title: this.translate.instant('pages.order_details.checkout_success_message'),\n type: NotificationType.Success,\n }))\n .otherwise(() => ({\n title: this.translate.instant('pages.order_details.checkout_error_message'),\n type: NotificationType.Error,\n }));\n\n if (success) {\n // Even if the order is already in the status Completed, we might as well set it to Completed\n await this.updateStatus(OrderStatus.completed);\n this.selectedStatus = OrderStatus.completed;\n }\n\n await this.fetchOrderDetails();\n\n this.notification.create(message.title, '', message.type);\n } catch (error) {\n // TODO: Find a way to dispatch the error within Sentry\n console.error(error);\n this.notification.error(\n this.translate.instant('pages.order_details.checkout_error_message'),\n ''\n );\n }\n }\n\n public getProductTrip(productId: number): Trip | null {\n const article = this.articles.find(({ product }) => +product.id === +productId);\n\n if (article && article.trip) {\n return article.trip;\n }\n\n return null;\n }\n\n public getProductBooking(productId: number): ArticleBooking | null {\n const article = this.articles.find(({ product }) => +product.id === +productId);\n\n if (article && article.reservation) {\n return article.reservation;\n }\n\n return null;\n }\n}\n","\n
    \n {{ 'pages.order.filters' | translate }}\n
    \n\n
    \n \n
    \n
    \n\n\n
    \n {{ 'pages.order.sales_filter' | translate }}\n
    \n\n \n
    \n {{ 'pages.order.filters' | translate }}\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 {{ 'pages.order.filters' | translate }}\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 {{ 'pages.order.filters' | translate }}\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 {{ productItem.id }} | {{ productItem.name }}\n \n \n\n
    \n \n \n \n
    \n
    \n
    \n \n\n \n {{ 'pages.order.filters' | translate }}\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 {{ 'pages.order.reset_form' | translate }}\n \n\n \n \n \n
    \n\n\n \n\n","import { NgModule } from '@angular/core';\nimport { RouterModule, Routes } from '@angular/router';\nimport { AuthGuard } from '@app/modules/shared/guards/auth.guard';\nimport { PermissionGuard } from '@app/modules/shared/guards/permission.guard';\nimport { OrderDetailComponent } from './components/order-detail/order-detail.component';\nimport { OrdersComponent } from './components/orders.component';\n\nconst routes: Routes = [\n {\n path: '',\n canActivate: [AuthGuard],\n canActivateChild: [PermissionGuard],\n data: {\n title: 'pages.list_menu.sales',\n permissions: {\n One: ['ACCESS_ORDERS', 'ACCESS_SCHOOL_ORDERS'],\n redirectTo: '/403',\n },\n },\n children: [\n {\n path: '',\n canActivate: [AuthGuard],\n component: OrdersComponent,\n data: {\n title: '',\n },\n },\n {\n path: ':id',\n canActivate: [AuthGuard],\n component: OrderDetailComponent,\n data: {\n title: 'pages.order_details.sale_details',\n },\n },\n ],\n },\n];\n\n@NgModule({\n imports: [RouterModule.forChild(routes)],\n exports: [RouterModule],\n})\nexport class OrdersRoutingModule {}\n","import { Component, OnInit, ViewChild } from '@angular/core';\nimport { UntypedFormBuilder, UntypedFormControl } from '@angular/forms';\nimport { CustomersService } from '@app/modules/customers/services/customers.service';\nimport { TableServerComponent } from '@app/modules/shared/components/table-server/table-server.component';\nimport { ColumnConfig } from '@app/modules/shared/models/server-pagination';\nimport { AuthService } from '@app/modules/shared/services/auth.service';\nimport { FilterService } from '@app/modules/shared/services/filter.service';\nimport { QueryParamsService } from '@app/modules/shared/services/queryParams.service';\nimport { ServerPaginationService } from '@app/modules/shared/services/server-pagination.service';\nimport { ProductItem } from '@app/modules/shop/models/product-item';\nimport { ShopService } from '@app/modules/shop/services/shop.service';\nimport { heroicons } from '@assets/icons/heroicons';\nimport { TranslateService } from '@ngx-translate/core';\nimport { Subscription, debounceTime, distinctUntilChanged } from 'rxjs';\nimport { Order, OrderFilters } from '../models/order';\nimport { OrdersService } from '../services/orders.service';\nimport { Article } from '@app/modules/orders/models/article';\n\n@Component({\n selector: 'tu-orders',\n templateUrl: './orders.component.html',\n styleUrls: ['./orders.component.scss'],\n providers: [OrdersService],\n})\nexport class OrdersComponent implements OnInit {\n constructor(\n private fb: UntypedFormBuilder,\n private authService: AuthService,\n private customersService: CustomersService,\n private shopService: ShopService,\n private filterService: FilterService,\n private translate: TranslateService,\n private queryParamsService: QueryParamsService,\n public serverPaginationService: ServerPaginationService\n ) {}\n\n @ViewChild('table', { static: true }) table: TableServerComponent;\n\n get shouldShowOrderSearch(): boolean {\n return window['__CONFIG__']?.features?.orderSearch ?? true;\n }\n\n public filters: OrderFilters = {};\n\n public filterOptionsForm = this.fb.group({\n order_status: [null],\n order_process_status: [null],\n shipping_mode: [null],\n order_installment: [null],\n order_price: [null],\n payment_method: [null],\n virtual_provider: [null],\n origin_type: [null],\n product_id: [null],\n attribute: [null],\n attribute_search: [null],\n });\n\n public productItems: ProductItem[] = [];\n public filteredProductItems: ProductItem[] = [];\n\n public productFilterControl = new UntypedFormControl();\n\n public attributes;\n\n get selectedAttribute() {\n const hasAttributes = Boolean(this.attributes);\n const hasSelectedAttribute = Boolean(this.filterOptionsForm.value.attribute);\n\n if (!hasAttributes || !hasSelectedAttribute) return false;\n\n const attribute = this.attributes.find(\n (attribute) => +attribute.id === +this.filterOptionsForm.value.attribute\n );\n\n return {\n ...attribute,\n values: attribute?.values?.sort((a: string, b: string) => a.localeCompare(b)) ?? undefined,\n };\n }\n\n public computeLink(order: Order) {\n return `/orders/${order.id}`;\n }\n\n public get columnsConfig(): ColumnConfig[] {\n return [\n {\n key: 'articles',\n label: this.translate.instant('pages.order.detail'),\n type: 'details',\n config: {\n table: [\n {\n key: 'productTitle',\n label: this.translate.instant('pages.order.article_title'),\n type: 'link',\n isSortable: false,\n config: {\n href: (_, row) => {\n if ('productId' in row) {\n return `/shop/products/edit/${row.productId}`;\n }\n },\n },\n },\n {\n key: 'quantity',\n label: this.translate.instant('pages.order.article_quantity'),\n type: 'text',\n isSortable: false,\n },\n {\n key: 'productUnitPrice',\n label: this.translate.instant('pages.order.article_unit_price'),\n type: 'price',\n isSortable: false,\n config: {\n currency: (order) => {\n if ('network_id' in order) {\n const network = this.authService.networks.find(\n (network) => +network.id === +order.network_id\n );\n return network?.currency ?? 'EUR';\n }\n },\n locale: () => this.translate.currentLang,\n },\n },\n {\n key: '',\n label: this.translate.instant('pages.order.article_total_price'),\n type: 'sumQuantityPrice',\n isSortable: false,\n config: {\n quantityKey: 'quantity',\n priceKey: 'productUnitPrice',\n currency: (order) => {\n if ('network_id' in order) {\n const network = this.authService.networks.find(\n (network) => +network.id === +order.network_id\n );\n return network?.currency ?? 'EUR';\n }\n },\n locale: () => this.translate.currentLang,\n },\n },\n ],\n },\n },\n {\n key: 'order_identifier',\n label: this.translate.instant('pages.order.sale_id'),\n type: 'text',\n isSortable: true,\n },\n {\n key: 'date',\n label: this.translate.instant('pages.order.date'),\n type: 'date',\n },\n {\n key: 'user.fullname',\n label: this.translate.instant('pages.order.client'),\n type: 'link',\n isSortable: true,\n config: {\n href: (_, row) => {\n return `/customers/${row.user.customer_id}`;\n },\n },\n },\n {\n key: 'user.email',\n label: this.translate.instant('pages.order.mail_address'),\n type: 'link',\n config: {\n href: (_, row) => `mailto:${row.user.email}`,\n },\n },\n {\n key: 'total_vat',\n label: this.translate.instant('pages.order.total'),\n type: 'price',\n config: {\n currency: (order) => {\n const network = this.authService.networks.find(\n (network) => +network.id === +order.network_id\n );\n return network?.currency ?? 'EUR';\n },\n locale: () => this.translate.currentLang,\n },\n isSortable: true,\n },\n {\n key: 'installments',\n label: this.translate.instant('pages.order.payment'),\n type: 'text',\n modifier: (installements) => {\n return `${installements} ${this.translate.instant('pages.order.times')}`;\n },\n isSortable: true,\n },\n {\n key: 'payment_method',\n label: this.translate.instant('pages.order.payment_method'),\n type: 'badge',\n isSortable: true,\n config: {\n CARD: { type: 'information', label: this.translate.instant('pages.order.card') },\n TRANSFER: { type: 'warning', label: this.translate.instant('pages.order.transfer') },\n DEPOSIT: { type: 'success', label: this.translate.instant('pages.order.deposit') },\n CASH: { type: 'success', label: this.translate.instant('pages.order.cash') },\n },\n },\n {\n key: 'wasCustomerAuthAsGuest',\n label: this.translate.instant('pages.order.guest_mode'),\n type: 'badge',\n modifier(wasCustomerAuthAsGuest: boolean) {\n return wasCustomerAuthAsGuest ? 'YES' : 'NO';\n },\n config: {\n YES: { type: 'information', label: this.translate.instant('otherslabels.yes') },\n NO: { type: 'information', label: this.translate.instant('otherslabels.no') },\n },\n },\n {\n key: 'item_subscription_count',\n label: this.translate.instant('pages.order.tacit'),\n type: 'badge',\n isSortable: true,\n modifier(count: string) {\n return Number.parseInt(count, 10) ? 'YES' : 'NO';\n },\n config: {\n YES: { type: 'success', label: this.translate.instant('otherslabels.yes') },\n NO: { type: 'danger', label: this.translate.instant('otherslabels.no') },\n },\n },\n {\n key: 'status',\n label: this.translate.instant('pages.order.status'),\n type: 'badge',\n isSortable: true,\n config: {\n CANCELED: { type: 'warning', label: this.translate.instant('pages.order.cancelled') },\n ERROR: { type: 'danger', label: this.translate.instant('pages.order.refused') },\n COMPLETED: { type: 'success', label: this.translate.instant('pages.order.ok') },\n PENDING: { type: 'information', label: this.translate.instant('pages.order.unfinished') },\n REGULARIZATION: {\n type: 'warning',\n label: this.translate.instant('pages.order.regularization'),\n },\n PROCESSING: { type: 'warning', label: this.translate.instant('pages.order.processing') },\n PREAUTHORIZED: {\n type: 'success',\n label: this.translate.instant('pages.order.preauthorized'),\n },\n WAITING_FOR_PAYMENT: {\n type: 'warning',\n label: this.translate.instant('pages.order.waiting_for_payment'),\n },\n TO_REFUND: {\n type: 'information',\n label: this.translate.instant('pages.order.to_refund'),\n },\n REFUNDED: {\n type: 'success',\n label: this.translate.instant('pages.order.refunded'),\n },\n PAYMENT_ISSUE: {\n type: 'danger',\n label: this.translate.instant('pages.order.default_payment'),\n },\n },\n },\n {\n key: 'process_status',\n label: this.translate.instant('pages.order.process_status'),\n type: 'badge',\n isSortable: true,\n modifier(status, order) {\n return order.status === 'COMPLETED' ||\n order.status === 'PREAUTHORIZED' ||\n order.status === 'WAITING_FOR_PAYMENT'\n ? status\n : 'NULL';\n },\n config: {\n TO_PROCESS: {\n type: 'information',\n label: this.translate.instant('pages.order.to_process'),\n },\n PROCESSING: { type: 'warning', label: this.translate.instant('pages.order.processing') },\n PROCESSED: {\n type: 'success',\n label: this.translate.instant('pages.order.processed'),\n },\n AUTOPROCESSED: {\n type: 'information',\n label: this.translate.instant('pages.order.autoprocessed'),\n },\n NULL: { type: 'hidden', label: '' },\n },\n },\n {\n key: 'supports',\n label: this.translate.instant('pages.order.delivery'),\n type: 'icon',\n config: {\n label: (supports: string[]) => {\n const map = {\n PHYSICAL: this.translate.instant('pages.order.support_physical'),\n DEMATERIALIZED: this.translate.instant('pages.order.support_demat'),\n COLLECT: this.translate.instant('pages.order.support_collect'),\n MEDIA: this.translate.instant('pages.order.support_media'),\n SHIPPING: this.translate.instant('pages.order.support_shipping'),\n };\n\n const labels = supports\n .sort()\n .map((support) => map[support])\n .join(', ');\n\n return supports.length > 1\n ? `${this.translate.instant('pages.order.support_mix')} : ${labels}`\n : `${this.translate.instant('pages.order.delivery')} : ${labels}`;\n },\n svgPaths: (supports: string[]) => {\n const map = {\n PHYSICAL: heroicons.outline.creditCard,\n DEMATERIALIZED: heroicons.outline.deviceMobile,\n COLLECT: heroicons.outline.cursorClick,\n MEDIA: heroicons.outline.cash,\n SHIPPING: heroicons.outline.home,\n };\n\n return supports.sort().map((support) => {\n try {\n return map[support];\n } catch {\n return map['DEMATERIALIZED'];\n }\n });\n },\n },\n isSortable: true,\n },\n {\n key: 'user.customer_id',\n label: this.translate.instant('pages.order.user_id'),\n type: 'text',\n isDisplayed: () => false,\n isSortable: true,\n },\n {\n key: 'network_id',\n label: this.translate.instant('pages.order.network'),\n type: 'text',\n modifier: (id) => this.authService.networks.find((n) => +n.id === +id).name,\n isSortable: true,\n isExportable: this.authService.networks.length > 1,\n isDisplayed: () => this.authService.networks.length > 1,\n },\n {\n key: 'origin_type',\n label: this.translate.instant('pages.order.origin'),\n type: 'text',\n modifier: (origin) => {\n return origin || '-';\n },\n isSortable: true,\n },\n {\n key: 'voucher',\n label: this.translate.instant('pages.order.voucher'),\n type: 'text',\n isSortable: false,\n },\n {\n key: '',\n label: this.translate.instant('pages.order.promotional_codes'),\n type: 'text',\n modifier: (_, order: Order) => {\n if (order.discounts) {\n return order.discounts.length;\n }\n\n return 0;\n },\n isSortable: false,\n },\n {\n key: 'virtual_provider',\n label: this.translate.instant('otherslabels.col_platform'),\n type: 'image',\n modifier: (_, order) => {\n return this.determineVirtualProvider(order);\n },\n config: {\n alt: (value: string) => value,\n src(value: string) {\n if (value?.toLowerCase().includes('instantsystem')) {\n return '/assets/img/providers/instant_system.png';\n }\n\n switch (value) {\n case 'tixipass':\n return '/assets/img/logo_tixipass.png';\n case 'sncf':\n return '/assets/img/providers/sncf.png';\n case 'smt_nfc':\n return '/assets/img/providers/smt_nfc.png';\n case 'modalis':\n return '/assets/img/providers/modalis.png';\n case 'otipass':\n return '/assets/img/providers/otipass.png';\n default:\n return '';\n }\n },\n },\n isSortable: true,\n isDisplayed: () => this.authService.networks.some((network) => network.universal),\n },\n {\n key: 'payment_token',\n label: this.translate.instant('pages.order.media_transaction'),\n type: 'text',\n },\n ];\n }\n\n public computeFilters() {\n const globalFilters = this.filterService.filters;\n\n const {\n shipping_mode,\n order_status,\n order_process_status,\n order_installment,\n order_price,\n virtual_provider,\n payment_method,\n origin_type,\n product_id,\n attribute,\n attribute_search,\n } = this.filterOptionsForm.value;\n\n //TODO set network as string in whole app\n const filters: Record = {\n network_id: globalFilters.network,\n from: globalFilters.from,\n to: globalFilters.to,\n query: globalFilters.query,\n status: order_status,\n processingStatus: order_process_status,\n supports: shipping_mode,\n installments: order_installment,\n provider: virtual_provider,\n paymentMethod: payment_method,\n originType: origin_type,\n productId: product_id,\n attribute: attribute,\n attributeSearch: attribute_search,\n };\n\n // FIXME: This help prevent boolean casting on the server side\n if (['string', 'number'].includes(typeof order_price)) {\n filters.total = +order_price === 0 ? '000' : String(+order_price * 100);\n }\n\n this.filters = filters;\n\n this.queryParamsService.localFilterParams = this.filterOptionsForm.value;\n }\n\n ngOnInit() {\n if (!this.shouldShowOrderSearch) {\n this.filterService.reset();\n }\n\n const params = this.queryParamsService.localFilterParams;\n\n const formParams: Record = {};\n\n for (const property of Object.keys(this.filterOptionsForm.value)) {\n formParams[property] = params[property] || null;\n }\n\n this.filterOptionsForm.setValue(formParams);\n\n this.computeFilters();\n\n let unsubscribeToAttribueChanges: Subscription;\n\n this.customersService.getAttributes().then((attributes) => {\n // We only want to filter on disability_benefits or social_fund (at least, for now)\n const attributeFiltersKeys = ['disability_benefits', 'social_fund'];\n\n this.attributes = attributes.filter((attribute) =>\n attributeFiltersKeys.includes(attribute.key)\n );\n\n if (unsubscribeToAttribueChanges?.unsubscribe) {\n unsubscribeToAttribueChanges.unsubscribe();\n }\n\n unsubscribeToAttribueChanges =\n this.filterOptionsForm.controls.attribute.valueChanges.subscribe((value) => {\n const attribute = this.attributes.find((attribute) => attribute.id === value);\n this.filterOptionsForm.controls.attribute_search.setValue(attribute?.values?.[0] || null);\n });\n });\n\n this.shopService.getProductsItem().subscribe((productItems) => {\n if (!Array.isArray(productItems)) return;\n\n this.productItems = productItems;\n });\n\n this.productFilterControl.valueChanges\n .pipe(debounceTime(200), distinctUntilChanged())\n .subscribe((product) => {\n if (product) {\n // Since some products have the same name, we also need to use product ID as autocomplete value\n // '|' is used as a separator between the ID and name when a product is actually selected\n // The product filter will work as long as no product contains a pipe in its name\n const productData = product.split(' | ');\n\n const isProductSelected = productData.length === 2;\n\n const filterValue = !isProductSelected\n ? productData[0].toLowerCase()\n : productData[1].toLowerCase();\n\n this.filteredProductItems = this.productItems.filter((productItem) => {\n return (\n productItem.name.toLowerCase().includes(filterValue) ||\n `${productItem.id}`.includes(filterValue)\n );\n });\n\n const productItemId = isProductSelected ? productData[0] : null;\n this.filterOptionsForm.controls.product_id.setValue(productItemId);\n } else {\n this.filteredProductItems = this.productItems;\n this.filterOptionsForm.controls.product_id.setValue(null);\n }\n });\n }\n\n // Export API\n get exportInformation() {\n const hasCheckedRows = this.table.selectedRows.length;\n const checkedRowsIds = hasCheckedRows ? this.table.selectedRows.map((row) => row.id) : [];\n\n const ordersFilters = this.filterOptionsForm.value;\n\n const filters = {};\n\n // Add filters only if needed\n if (ordersFilters.order_status) {\n filters['status'] = ordersFilters.order_status;\n }\n\n if (ordersFilters.order_process_status) {\n filters['processingStatus'] = ordersFilters.order_process_status;\n }\n\n if (ordersFilters.shipping_mode) {\n filters['supports'] = ordersFilters.shipping_mode;\n }\n\n if (ordersFilters.order_installment !== null) {\n filters['installments'] = ordersFilters.order_installment;\n }\n\n if (ordersFilters.order_price !== null) {\n filters['total'] = ordersFilters.order_price * 100;\n }\n\n if (ordersFilters.virtual_provider) {\n filters['provider'] = ordersFilters.virtual_provider;\n }\n\n if (ordersFilters.payment_method) {\n filters['paymentMethod'] = ordersFilters.payment_method;\n }\n\n if (ordersFilters.product_id) {\n filters['productId'] = ordersFilters.product_id;\n }\n\n if (ordersFilters.origin_type) {\n filters['originType'] = ordersFilters.origin_type;\n }\n\n const isAttributeFilterSelected = ordersFilters.attribute && ordersFilters.attribute_search;\n\n if (isAttributeFilterSelected) {\n filters['attributeId'] = ordersFilters.attribute.toString();\n filters['attributeValue'] = ordersFilters.attribute_search.toString();\n }\n\n return {\n exportId: 'cms-orders-v2',\n checkedRowsIds: checkedRowsIds,\n hasCheckedAll: this.table.allSelected,\n filters: filters,\n };\n }\n\n private determineVirtualProvider(order: Order): string {\n return Boolean(order.universal)\n ? 'tixipass'\n : Boolean(order.client)\n ? order.client.slug\n : order.issuer;\n }\n}\n","import { CommonModule } from '@angular/common';\nimport { NgModule } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatAutocompleteModule } from '@angular/material/autocomplete';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSelectModule } from '@angular/material/select';\nimport { RouterModule } from '@angular/router';\nimport { SharedModule } from '@app/modules/shared/shared.module';\nimport { NgbModule } from '@ng-bootstrap/ng-bootstrap';\nimport { SimpleNotificationsModule } from 'angular2-notifications';\nimport { MomentModule } from 'ngx-moment';\nimport { ShopService } from '../shop/services/shop.service';\nimport { OrderDetailComponent } from './components/order-detail/order-detail.component';\nimport { OrdersComponent } from './components/orders.component';\nimport { OrdersRoutingModule } from './orders.routing';\nimport { OrdersService } from './services/orders.service';\nimport { MatMenuModule } from '@angular/material/menu';\nimport { OrderCommentComponent } from './components/order-detail/order-comment/order-comment.component';\n\n@NgModule({\n declarations: [OrdersComponent, OrderDetailComponent, OrderCommentComponent],\n imports: [\n OrdersRoutingModule,\n SharedModule,\n CommonModule,\n FormsModule,\n ReactiveFormsModule,\n RouterModule,\n MomentModule,\n NgbModule,\n SimpleNotificationsModule.forRoot(),\n MatSelectModule,\n MatInputModule,\n MatAutocompleteModule,\n MatMenuModule,\n ],\n providers: [OrdersService, ShopService, MatSelectModule],\n})\nexport class OrdersModule {}\n","import { NgModule } from '@angular/core';\nimport { RouterModule, Routes } from '@angular/router';\nimport { AppLayoutComponent } from './app-layout/app-layout.component';\nimport { AuthGuard } from './modules/shared/guards/auth.guard';\nimport { HomeGuard } from './modules/shared/guards/home.guard';\nimport { PermissionGuard } from './modules/shared/guards/permission.guard';\nimport { AuthService } from './modules/shared/services/auth.service';\nimport { BadgeService } from './modules/shared/services/badge.service';\nimport { ConfigGuard } from './modules/shared/guards/config.guard';\nimport { CustomersModule } from './modules/customers/customers.module';\nimport { OrdersModule } from './modules/orders/orders.module';\n\nconst routes: Routes = [\n {\n path: 'login',\n loadChildren: () => import('./modules/auth/auth.module').then((m) => m.AuthModule),\n },\n {\n path: 'maintenance',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/maintenance/maintenance.module').then((m) => m.MaintenanceModule),\n },\n {\n path: '',\n canLoad: [AuthGuard],\n canActivateChild: [PermissionGuard],\n component: AppLayoutComponent,\n children: [\n {\n path: '',\n pathMatch: 'full',\n canActivate: [HomeGuard],\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/forbidden/forbidden.module').then((m) => m.ForbiddenModule),\n },\n {\n path: 'account-management',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/account-management/account-management.module').then(\n (m) => m.AccountManagementModule\n ),\n },\n {\n path: 'documents',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/document/document.module').then((m) => m.DocumentModule),\n },\n {\n path: 'graphic-documents',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/graphic-document/graphic-document.module').then(\n (m) => m.GraphicDocumentModule\n ),\n },\n {\n path: 'dashboard',\n\n canLoad: [AuthGuard],\n canActivate: [ConfigGuard],\n canActivateChild: [ConfigGuard],\n loadChildren: () =>\n import('./modules/dashboard/dashboard.module').then((m) => m.DashboardModule),\n data: {\n features: ['dashboard'],\n },\n },\n {\n path: 'dashboard-next',\n\n canLoad: [AuthGuard],\n canActivate: [ConfigGuard],\n canActivateChild: [ConfigGuard],\n loadChildren: () =>\n import('./modules/dashboard-next/dashboard.module').then((m) => m.DashboardModule),\n data: {\n features: ['dashboard'],\n },\n },\n {\n path: 'dashboard-ng',\n\n canLoad: [AuthGuard],\n canActivate: [ConfigGuard],\n canActivateChild: [ConfigGuard],\n loadChildren: () =>\n import('./modules/dashboard-ng/dashboard-ng.module').then((m) => m.DashboardNgModule),\n data: {\n features: ['dashboard'],\n },\n },\n {\n path: 'orders',\n canLoad: [AuthGuard],\n loadChildren: () => OrdersModule,\n },\n {\n path: 'subscriptions',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/subscriptions/subscriptions.module').then((m) => m.SubscriptionsModule),\n },\n {\n path: 'customers',\n canLoad: [AuthGuard],\n loadChildren: () => CustomersModule,\n },\n {\n path: 'validations',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/validations/validations.module').then((m) => m.ValidationsModule),\n },\n {\n path: 'ticket',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/ticket/ticket.module').then((m) => m.TicketModule),\n },\n {\n path: 'geolocation',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/geolocation/geolocation.module').then((m) => m.GeolocationModule),\n },\n {\n path: 'controls',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/controls/controls.module').then((m) => m.ControlsModule),\n },\n {\n path: 'colors',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/colors/colors.module').then((m) => m.ColorsModule),\n },\n {\n path: 'salesterms',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/salesterms/salesterms.module').then((m) => m.SalesTermsModule),\n },\n {\n path: 'legal-notice',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/legal-notice/legal-notice.module').then((m) => m.LegalNoticeModule),\n },\n {\n path: 'privacy-policy',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/privacy-policy/privacy-policy.module').then(\n (m) => m.PrivacyPolicyModule\n ),\n },\n {\n path: 'contact',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/contact/contact.module').then((m) => m.ContactModule),\n },\n {\n path: 'reports',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/lemonway-reports/lemonway-reports.module').then(\n (m) => m.LemonwayReportsModule\n ),\n },\n {\n path: 'payment',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/payment/payment.module').then((m) => m.PaymentModule),\n },\n {\n path: 'fleet',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/fleet/fleet.module').then((m) => m.FleetModule),\n },\n {\n path: 'flyers',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/flyers/flyers.module').then((m) => m.FlyersModule),\n },\n {\n path: 'providers',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/providers/providers.module').then((m) => m.ProvidersModule),\n },\n {\n path: 'graphics',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/graphics/graphics.module').then((m) => m.GraphicsModule),\n },\n {\n path: 'users',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/users/users.module').then((m) => m.UsersModule),\n },\n {\n path: 'networks',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/network/network.module').then((m) => m.NetworkModule),\n },\n {\n path: 'clients',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/clients/clients.module').then((m) => m.ClientsModule),\n },\n {\n path: 'gtfs',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/gtfs/gtfs.module').then((m) => m.GtfsModule),\n },\n {\n path: 'shop',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/shop/shop.module').then((m) => m.ShopModule),\n },\n {\n path: 'vehicle',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/vehicle/vehicle.module').then((m) => m.VehicleModule),\n },\n {\n path: 'trips',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/trips/trips.module').then((m) => m.TripsModule),\n },\n {\n path: 'booking',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/booking/booking.module').then((m) => m.BookingModule),\n },\n {\n path: 'logs',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/audit/audit.module').then((m) => m.AuditModule),\n },\n {\n path: 'submissions',\n\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/submissions/submissions.module').then((m) => m.SubmissionsModule),\n },\n {\n path: 'school',\n\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/school/school.module').then((m) => m.SchoolModule),\n },\n {\n path: 'fields',\n canLoad: [AuthGuard],\n loadChildren: () => import('./modules/fields/fields.module').then((m) => m.FieldsModule),\n },\n {\n path: 'tap-and-protect',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/tap-and-protect/tap-and-protect.module').then(\n (m) => m.TapAndProtectModule\n ),\n },\n {\n path: 'validators',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/validators/validators.module').then((m) => m.ValidatorsModule),\n },\n {\n path: 'provisioning',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/provisioning/provisioning.module').then((m) => m.ProvisioningModule),\n },\n {\n path: 'platform',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/platform/platform.module').then((m) => m.PlatformModule),\n },\n {\n path: 'changelog',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/changelog/changelog.module').then((m) => m.ChangelogModule),\n },\n {\n path: 'agencies',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/agencies/agencies.module').then((m) => m.AgenciesModule),\n },\n {\n path: 'maintenance-mode',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/maintenance-mode/maintenance-mode.module').then(\n (m) => m.MaintenanceModeModule\n ),\n },\n {\n path: 'matrix-od',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/matrix-od/matrix-od.module').then((m) => m.MatrixOdModule),\n },\n {\n path: '403',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/forbidden/forbidden.module').then((m) => m.ForbiddenModule),\n },\n {\n path: '404',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/notfound/notfound.module').then((m) => m.NotfoundModule),\n },\n {\n path: '500',\n canLoad: [AuthGuard],\n loadChildren: () =>\n import('./modules/internal-server-error/internal-server-error.module').then(\n (m) => m.InternalServerErrorModule\n ),\n },\n {\n path: '**',\n redirectTo: '/404',\n },\n ],\n },\n];\n\n@NgModule({\n imports: [RouterModule.forRoot(routes)],\n exports: [RouterModule],\n providers: [AuthService, BadgeService],\n})\nexport class AppRoutingModule {}\n","import { HttpErrorResponse, HttpHandler, HttpInterceptor, HttpRequest } from '@angular/common/http';\nimport { Injectable } from '@angular/core';\nimport { Router } from '@angular/router';\nimport { throwError } from 'rxjs';\nimport { catchError, map, tap } from 'rxjs/operators';\nimport { AuthService } from '../modules/shared/services/auth.service';\nimport { environment as env } from '@env/environment';\n\n@Injectable()\nexport class CommonInterceptor implements HttpInterceptor {\n constructor(private readonly router: Router, private readonly auth: AuthService) {}\n\n private isInError = false;\n\n intercept(request: HttpRequest, next: HttpHandler) {\n const token = this.auth.token;\n let headers = request.headers;\n\n if (token) {\n headers = headers.set('Authorization', `Bearer ${token}`);\n }\n\n const modifiedRequest = request.clone({ headers, withCredentials: true });\n\n return next.handle(modifiedRequest).pipe(\n tap((event: any) => {\n // FIXME: We should check if event is an instance of HttpResponse\n this.auth.checkLoggedIn(event.response);\n }),\n catchError(async (response: HttpErrorResponse) => {\n const code = response?.error?.error?.code;\n\n /**\n * The session is considered scuffed if:\n * - The token is invalid\n * - The session is expired\n **/\n const isSessionScuffed = ['TOKEN_INVALID', 'SESSION_EXPIRED'].includes(code);\n\n // If the error isn't tied to the session, go on with your life\n if (!isSessionScuffed) {\n throw response.error;\n }\n\n // Otherwise, show up the session expired modal\n try {\n const action = await this.auth.sessionTimeoutModal();\n\n if (action === 'REDIRECT') {\n this.auth.clearStorage();\n this.router.navigate(['/login'], { queryParams: { returnUrl: this.auth.redirectUrl } });\n return throwError(() => response?.message ?? 403);\n }\n } catch (error) {\n console.info('Error while handling session timeout', error);\n\n this.auth.clearStorage();\n const authUrl = new URL(env.config.newAuthenticationUrl);\n authUrl.searchParams.set('redirect', this.auth?.redirectUrl ?? window.location.href);\n location.href = authUrl.toString();\n }\n }),\n map((response: any) => {\n const body = response?.body;\n\n if (body?.status === 'ko') {\n const code = body?.response?.errorCode;\n const message = body?.response?.errorMessage;\n\n throw new Error(code ?? message);\n }\n\n return response;\n })\n );\n }\n}\n","import { APP_BASE_HREF, registerLocaleData } from '@angular/common';\nimport { HttpClient, HttpClientModule, HTTP_INTERCEPTORS } from '@angular/common/http';\nimport localeFrExtra from '@angular/common/locales/extra/fr';\nimport localeFr from '@angular/common/locales/fr';\nimport { APP_INITIALIZER, ErrorHandler, LOCALE_ID, NgModule } from '@angular/core';\nimport { BrowserModule } from '@angular/platform-browser';\nimport { BrowserAnimationsModule } from '@angular/platform-browser/animations';\nimport { SharedModule } from '@app/modules/shared/shared.module';\nimport { NgbModule } from '@ng-bootstrap/ng-bootstrap';\nimport { TranslateLoader, TranslateModule } from '@ngx-translate/core';\nimport { TranslateHttpLoader } from '@ngx-translate/http-loader';\nimport * as Sentry from '@sentry/angular-ivy';\n\nimport 'moment/locale/fr';\nimport { AppLayoutComponent } from './app-layout/app-layout.component';\n// Components\nimport { AppComponent } from './app.component';\n// Modules\nimport { AppRoutingModule } from './app.routing';\nimport { CommonInterceptor } from './interceptors/common.interceptor';\nimport { AdminlabsService } from './services/adminlabs.service';\nimport { Router } from '@angular/router';\nimport { AuthService } from './modules/shared/services/auth.service';\n\nregisterLocaleData(localeFr, 'fr', localeFrExtra);\n\nexport function createTranslateLoader(http: HttpClient) {\n return new TranslateHttpLoader(http, 'assets/i18n/', '.json');\n}\n\n@NgModule({\n declarations: [AppComponent, AppLayoutComponent],\n imports: [\n HttpClientModule,\n BrowserModule,\n BrowserAnimationsModule,\n SharedModule.forRoot(),\n AppRoutingModule,\n NgbModule,\n TranslateModule.forRoot({\n loader: {\n provide: TranslateLoader,\n useFactory: createTranslateLoader,\n deps: [HttpClient],\n },\n }),\n ],\n providers: [\n // HACK: Set in global module to persiste data between pages and component\n AuthService,\n\n AdminlabsService,\n { provide: LOCALE_ID, useValue: 'fr-FR' },\n { provide: APP_BASE_HREF, useValue: '/' },\n { provide: HTTP_INTERCEPTORS, useClass: CommonInterceptor, multi: true },\n {\n provide: ErrorHandler,\n useValue: Sentry.createErrorHandler({\n showDialog: false,\n }),\n },\n {\n provide: Sentry.TraceService,\n deps: [Router],\n },\n {\n provide: APP_INITIALIZER,\n useFactory: () => () => {},\n deps: [Sentry.TraceService],\n multi: true,\n },\n ],\n exports: [TranslateModule, SharedModule, NgbModule],\n bootstrap: [AppComponent],\n})\nexport class AppModule {}\n","import { enableProdMode } from '@angular/core';\nimport * as Sentry from '@sentry/angular-ivy';\nimport { platformBrowserDynamic } from '@angular/platform-browser-dynamic';\n\nimport './polyfills';\nimport pkg from '../package.json';\nimport { AppModule } from './app';\nimport defaultConfig from './config.json';\nimport { environment } from './environments/environment';\n\nconst isLocalhost = window.location.hostname === 'localhost';\n\nSentry.init({\n release: pkg.version,\n dsn: 'https://e014f9c742eaad122527aa5278b9a942@o458744.ingest.sentry.io/4506316562694144',\n enabled: !isLocalhost,\n integrations: [\n new Sentry.BrowserTracing({\n // Set 'tracePropagationTargets' to control for which URLs distributed tracing should be enabled\n tracePropagationTargets: [\n 'localhost',\n 'admin.airwebpass.com',\n 'airwebpass.com',\n 'na.admin.airwebpass.com',\n 'bo.ticket.run.airweb.dev',\n ],\n routingInstrumentation: Sentry.routingInstrumentation,\n traceFetch: false,\n }),\n ],\n\n // Performance Monitoring\n tracesSampleRate: 0.5, // Capture 100% of the transactions\n // Session Replay\n // replaysSessionSampleRate: 0.5, // This sets the sample rate at 10%. You may want to change it to 100% while in development and then sample at a lower rate in production.\n // replaysOnErrorSampleRate: 1.0, // If you're not already sampling the entire session, change the sample rate to 100% when sampling sessions where errors occur.\n});\n\nif (environment.production) {\n enableProdMode();\n}\n\nif (environment.config.clarityCode !== 'fake-code') {\n // @ts-ignore\n tarteaucitron.user.clarity = environment.config.clarityCode;\n // @ts-ignore\n (tarteaucitron.job = tarteaucitron.job || []).push('clarity');\n}\n\nasync function main() {\n try {\n window['__CONFIG__'] = {\n features: {\n globalSearch: true,\n dashboard: true,\n customerSearch: true,\n orderSearch: true,\n globalSearchPreview: true,\n },\n };\n\n if (environment.production) {\n try {\n const url = new URL(environment.config.configURL);\n url.searchParams.set('cache', Date.now().toString());\n window['__CONFIG__'] = await fetch(url.toString()).then((response) => response.json());\n } catch (e) {\n window['__CONFIG__'] = defaultConfig;\n console.log('failed to fetch global config', { e });\n }\n }\n } finally {\n return platformBrowserDynamic().bootstrapModule(AppModule);\n }\n}\n\nmain();\n","import { getActiveTransaction, spanToJSON } from '@sentry/core';\nimport { logger } from '@sentry/utils';\nimport { DEBUG_BUILD } from '../common/debug-build.js';\nimport { WINDOW } from './types.js';\n\n/**\n * Add a listener that cancels and finishes a transaction when the global\n * document is hidden.\n */\nfunction registerBackgroundTabDetection() {\n if (WINDOW.document) {\n WINDOW.document.addEventListener('visibilitychange', () => {\n // eslint-disable-next-line deprecation/deprecation\n const activeTransaction = getActiveTransaction() ;\n if (WINDOW.document.hidden && activeTransaction) {\n const statusType = 'cancelled';\n\n const { op, status } = spanToJSON(activeTransaction);\n\n DEBUG_BUILD &&\n logger.log(`[Tracing] Transaction: ${statusType} -> since tab moved to the background, op: ${op}`);\n // We should not set status if it is already set, this prevent important statuses like\n // error or data loss from being overwritten on transaction.\n if (!status) {\n activeTransaction.setStatus(statusType);\n }\n // TODO: Can we rewrite this to an attribute?\n // eslint-disable-next-line deprecation/deprecation\n activeTransaction.setTag('visibilitychange', 'document.hidden');\n activeTransaction.end();\n }\n });\n } else {\n DEBUG_BUILD && logger.warn('[Tracing] Could not set up background tab detection due to lack of global document');\n }\n}\n\nexport { registerBackgroundTabDetection };\n","import { getActiveTransaction } from './utils.js';\n\n/**\n * Adds a measurement to the current active transaction.\n */\nfunction setMeasurement(name, value, unit) {\n // eslint-disable-next-line deprecation/deprecation\n const transaction = getActiveTransaction();\n if (transaction) {\n // eslint-disable-next-line deprecation/deprecation\n transaction.setMeasurement(name, value, unit);\n }\n}\n\nexport { setMeasurement };\n","import { Injectable, EventEmitter, InjectionToken, Inject, Directive, ElementRef, ChangeDetectorRef, Input, Pipe, NgModule } from '@angular/core';\nimport { of, isObservable, forkJoin, concat, defer } from 'rxjs';\nimport { take, shareReplay, map, concatMap, switchMap } from 'rxjs/operators';\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.loader.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/**\n * @abstract\n */\nimport * as ɵngcc0 from '@angular/core';\nclass TranslateLoader {\n}\nif (false) {\n /**\n * @abstract\n * @param {?} lang\n * @return {?}\n */\n TranslateLoader.prototype.getTranslation = function (lang) { };\n}\n/**\n * This loader is just a placeholder that does nothing, in case you don't need a loader at all\n */\nclass TranslateFakeLoader extends TranslateLoader {\n /**\n * @param {?} lang\n * @return {?}\n */\n getTranslation(lang) {\n return of({});\n }\n}\nTranslateFakeLoader.ɵfac = /*@__PURE__*/ function () { let ɵTranslateFakeLoader_BaseFactory; return function TranslateFakeLoader_Factory(t) { return (ɵTranslateFakeLoader_BaseFactory || (ɵTranslateFakeLoader_BaseFactory = ɵngcc0.ɵɵgetInheritedFactory(TranslateFakeLoader)))(t || TranslateFakeLoader); }; }();\nTranslateFakeLoader.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: TranslateFakeLoader, factory: TranslateFakeLoader.ɵfac });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateFakeLoader, [{\n type: Injectable\n }], null, null); })();\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/missing-translation-handler.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/**\n * @record\n */\nfunction MissingTranslationHandlerParams() { }\nif (false) {\n /**\n * the key that's missing in translation files\n * @type {?}\n */\n MissingTranslationHandlerParams.prototype.key;\n /**\n * an instance of the service that was unable to translate the key.\n * @type {?}\n */\n MissingTranslationHandlerParams.prototype.translateService;\n /**\n * interpolation params that were passed along for translating the given key.\n * @type {?|undefined}\n */\n MissingTranslationHandlerParams.prototype.interpolateParams;\n}\n/**\n * @abstract\n */\nclass MissingTranslationHandler {\n}\nif (false) {\n /**\n * A function that handles missing translations.\n *\n * @abstract\n * @param {?} params context for resolving a missing translation\n * @return {?} a value or an observable\n * If it returns a value, then this value is used.\n * If it return an observable, the value returned by this observable will be used (except if the method was \"instant\").\n * If it doesn't return then the key will be used as a value\n */\n MissingTranslationHandler.prototype.handle = function (params) { };\n}\n/**\n * This handler is just a placeholder that does nothing, in case you don't need a missing translation handler at all\n */\nclass FakeMissingTranslationHandler {\n /**\n * @param {?} params\n * @return {?}\n */\n handle(params) {\n return params.key;\n }\n}\nFakeMissingTranslationHandler.ɵfac = function FakeMissingTranslationHandler_Factory(t) { return new (t || FakeMissingTranslationHandler)(); };\nFakeMissingTranslationHandler.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: FakeMissingTranslationHandler, factory: FakeMissingTranslationHandler.ɵfac });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(FakeMissingTranslationHandler, [{\n type: Injectable\n }], null, null); })();\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/util.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/* tslint:disable */\n/**\n * Determines if two objects or two values are equivalent.\n *\n * Two objects or values are considered equivalent if at least one of the following is true:\n *\n * * Both objects or values pass `===` comparison.\n * * Both objects or values are of the same type and all of their properties are equal by\n * comparing them with `equals`.\n *\n * @param {?} o1 Object or value to compare.\n * @param {?} o2 Object or value to compare.\n * @return {?} true if arguments are equal.\n */\nfunction equals(o1, o2) {\n if (o1 === o2)\n return true;\n if (o1 === null || o2 === null)\n return false;\n if (o1 !== o1 && o2 !== o2)\n return true; // NaN === NaN\n // NaN === NaN\n /** @type {?} */\n let t1 = typeof o1;\n /** @type {?} */\n let t2 = typeof o2;\n /** @type {?} */\n let length;\n /** @type {?} */\n let key;\n /** @type {?} */\n let keySet;\n if (t1 == t2 && t1 == 'object') {\n if (Array.isArray(o1)) {\n if (!Array.isArray(o2))\n return false;\n if ((length = o1.length) == o2.length) {\n for (key = 0; key < length; key++) {\n if (!equals(o1[key], o2[key]))\n return false;\n }\n return true;\n }\n }\n else {\n if (Array.isArray(o2)) {\n return false;\n }\n keySet = Object.create(null);\n for (key in o1) {\n if (!equals(o1[key], o2[key])) {\n return false;\n }\n keySet[key] = true;\n }\n for (key in o2) {\n if (!(key in keySet) && typeof o2[key] !== 'undefined') {\n return false;\n }\n }\n return true;\n }\n }\n return false;\n}\n/* tslint:enable */\n/**\n * @param {?} value\n * @return {?}\n */\nfunction isDefined(value) {\n return typeof value !== 'undefined' && value !== null;\n}\n/**\n * @param {?} item\n * @return {?}\n */\nfunction isObject(item) {\n return (item && typeof item === 'object' && !Array.isArray(item));\n}\n/**\n * @param {?} target\n * @param {?} source\n * @return {?}\n */\nfunction mergeDeep(target, source) {\n /** @type {?} */\n let output = Object.assign({}, target);\n if (isObject(target) && isObject(source)) {\n Object.keys(source).forEach((/**\n * @param {?} key\n * @return {?}\n */\n (key) => {\n if (isObject(source[key])) {\n if (!(key in target)) {\n Object.assign(output, { [key]: source[key] });\n }\n else {\n output[key] = mergeDeep(target[key], source[key]);\n }\n }\n else {\n Object.assign(output, { [key]: source[key] });\n }\n }));\n }\n return output;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.parser.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/**\n * @abstract\n */\nclass TranslateParser {\n}\nif (false) {\n /**\n * Interpolates a string to replace parameters\n * \"This is a {{ key }}\" ==> \"This is a value\", with params = { key: \"value\" }\n * @abstract\n * @param {?} expr\n * @param {?=} params\n * @return {?}\n */\n TranslateParser.prototype.interpolate = function (expr, params) { };\n /**\n * Gets a value from an object by composed key\n * parser.getValue({ key1: { keyA: 'valueI' }}, 'key1.keyA') ==> 'valueI'\n * @abstract\n * @param {?} target\n * @param {?} key\n * @return {?}\n */\n TranslateParser.prototype.getValue = function (target, key) { };\n}\nclass TranslateDefaultParser extends TranslateParser {\n constructor() {\n super(...arguments);\n this.templateMatcher = /{{\\s?([^{}\\s]*)\\s?}}/g;\n }\n /**\n * @param {?} expr\n * @param {?=} params\n * @return {?}\n */\n interpolate(expr, params) {\n /** @type {?} */\n let result;\n if (typeof expr === 'string') {\n result = this.interpolateString(expr, params);\n }\n else if (typeof expr === 'function') {\n result = this.interpolateFunction(expr, params);\n }\n else {\n // this should not happen, but an unrelated TranslateService test depends on it\n result = (/** @type {?} */ (expr));\n }\n return result;\n }\n /**\n * @param {?} target\n * @param {?} key\n * @return {?}\n */\n getValue(target, key) {\n /** @type {?} */\n let keys = typeof key === 'string' ? key.split('.') : [key];\n key = '';\n do {\n key += keys.shift();\n if (isDefined(target) && isDefined(target[key]) && (typeof target[key] === 'object' || !keys.length)) {\n target = target[key];\n key = '';\n }\n else if (!keys.length) {\n target = undefined;\n }\n else {\n key += '.';\n }\n } while (keys.length);\n return target;\n }\n /**\n * @private\n * @param {?} fn\n * @param {?=} params\n * @return {?}\n */\n interpolateFunction(fn, params) {\n return fn(params);\n }\n /**\n * @private\n * @param {?} expr\n * @param {?=} params\n * @return {?}\n */\n interpolateString(expr, params) {\n if (!params) {\n return expr;\n }\n return expr.replace(this.templateMatcher, (/**\n * @param {?} substring\n * @param {?} b\n * @return {?}\n */\n (substring, b) => {\n /** @type {?} */\n let r = this.getValue(params, b);\n return isDefined(r) ? r : substring;\n }));\n }\n}\nTranslateDefaultParser.ɵfac = /*@__PURE__*/ function () { let ɵTranslateDefaultParser_BaseFactory; return function TranslateDefaultParser_Factory(t) { return (ɵTranslateDefaultParser_BaseFactory || (ɵTranslateDefaultParser_BaseFactory = ɵngcc0.ɵɵgetInheritedFactory(TranslateDefaultParser)))(t || TranslateDefaultParser); }; }();\nTranslateDefaultParser.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: TranslateDefaultParser, factory: TranslateDefaultParser.ɵfac });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateDefaultParser, [{\n type: Injectable\n }], null, null); })();\nif (false) {\n /** @type {?} */\n TranslateDefaultParser.prototype.templateMatcher;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.compiler.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/**\n * @abstract\n */\nclass TranslateCompiler {\n}\nif (false) {\n /**\n * @abstract\n * @param {?} value\n * @param {?} lang\n * @return {?}\n */\n TranslateCompiler.prototype.compile = function (value, lang) { };\n /**\n * @abstract\n * @param {?} translations\n * @param {?} lang\n * @return {?}\n */\n TranslateCompiler.prototype.compileTranslations = function (translations, lang) { };\n}\n/**\n * This compiler is just a placeholder that does nothing, in case you don't need a compiler at all\n */\nclass TranslateFakeCompiler extends TranslateCompiler {\n /**\n * @param {?} value\n * @param {?} lang\n * @return {?}\n */\n compile(value, lang) {\n return value;\n }\n /**\n * @param {?} translations\n * @param {?} lang\n * @return {?}\n */\n compileTranslations(translations, lang) {\n return translations;\n }\n}\nTranslateFakeCompiler.ɵfac = /*@__PURE__*/ function () { let ɵTranslateFakeCompiler_BaseFactory; return function TranslateFakeCompiler_Factory(t) { return (ɵTranslateFakeCompiler_BaseFactory || (ɵTranslateFakeCompiler_BaseFactory = ɵngcc0.ɵɵgetInheritedFactory(TranslateFakeCompiler)))(t || TranslateFakeCompiler); }; }();\nTranslateFakeCompiler.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: TranslateFakeCompiler, factory: TranslateFakeCompiler.ɵfac });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateFakeCompiler, [{\n type: Injectable\n }], null, null); })();\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.store.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\nclass TranslateStore {\n constructor() {\n /**\n * The lang currently used\n */\n this.currentLang = this.defaultLang;\n /**\n * a list of translations per lang\n */\n this.translations = {};\n /**\n * an array of langs\n */\n this.langs = [];\n /**\n * An EventEmitter to listen to translation change events\n * onTranslationChange.subscribe((params: TranslationChangeEvent) => {\n * // do something\n * });\n */\n this.onTranslationChange = new EventEmitter();\n /**\n * An EventEmitter to listen to lang change events\n * onLangChange.subscribe((params: LangChangeEvent) => {\n * // do something\n * });\n */\n this.onLangChange = new EventEmitter();\n /**\n * An EventEmitter to listen to default lang change events\n * onDefaultLangChange.subscribe((params: DefaultLangChangeEvent) => {\n * // do something\n * });\n */\n this.onDefaultLangChange = new EventEmitter();\n }\n}\nif (false) {\n /**\n * The default lang to fallback when translations are missing on the current lang\n * @type {?}\n */\n TranslateStore.prototype.defaultLang;\n /**\n * The lang currently used\n * @type {?}\n */\n TranslateStore.prototype.currentLang;\n /**\n * a list of translations per lang\n * @type {?}\n */\n TranslateStore.prototype.translations;\n /**\n * an array of langs\n * @type {?}\n */\n TranslateStore.prototype.langs;\n /**\n * An EventEmitter to listen to translation change events\n * onTranslationChange.subscribe((params: TranslationChangeEvent) => {\n * // do something\n * });\n * @type {?}\n */\n TranslateStore.prototype.onTranslationChange;\n /**\n * An EventEmitter to listen to lang change events\n * onLangChange.subscribe((params: LangChangeEvent) => {\n * // do something\n * });\n * @type {?}\n */\n TranslateStore.prototype.onLangChange;\n /**\n * An EventEmitter to listen to default lang change events\n * onDefaultLangChange.subscribe((params: DefaultLangChangeEvent) => {\n * // do something\n * });\n * @type {?}\n */\n TranslateStore.prototype.onDefaultLangChange;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.service.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/** @type {?} */\nconst USE_STORE = new InjectionToken('USE_STORE');\n/** @type {?} */\nconst USE_DEFAULT_LANG = new InjectionToken('USE_DEFAULT_LANG');\n/** @type {?} */\nconst DEFAULT_LANGUAGE = new InjectionToken('DEFAULT_LANGUAGE');\n/** @type {?} */\nconst USE_EXTEND = new InjectionToken('USE_EXTEND');\n/**\n * @record\n */\nfunction TranslationChangeEvent() { }\nif (false) {\n /** @type {?} */\n TranslationChangeEvent.prototype.translations;\n /** @type {?} */\n TranslationChangeEvent.prototype.lang;\n}\n/**\n * @record\n */\nfunction LangChangeEvent() { }\nif (false) {\n /** @type {?} */\n LangChangeEvent.prototype.lang;\n /** @type {?} */\n LangChangeEvent.prototype.translations;\n}\n/**\n * @record\n */\nfunction DefaultLangChangeEvent() { }\nif (false) {\n /** @type {?} */\n DefaultLangChangeEvent.prototype.lang;\n /** @type {?} */\n DefaultLangChangeEvent.prototype.translations;\n}\nclass TranslateService {\n /**\n *\n * @param {?} store an instance of the store (that is supposed to be unique)\n * @param {?} currentLoader An instance of the loader currently used\n * @param {?} compiler An instance of the compiler currently used\n * @param {?} parser An instance of the parser currently used\n * @param {?} missingTranslationHandler A handler for missing translations.\n * @param {?=} useDefaultLang whether we should use default language translation when current language translation is missing.\n * @param {?=} isolate whether this service should use the store or not\n * @param {?=} extend To make a child module extend (and use) translations from parent modules.\n * @param {?=} defaultLanguage Set the default language using configuration\n */\n constructor(store, currentLoader, compiler, parser, missingTranslationHandler, useDefaultLang = true, isolate = false, extend = false, defaultLanguage) {\n this.store = store;\n this.currentLoader = currentLoader;\n this.compiler = compiler;\n this.parser = parser;\n this.missingTranslationHandler = missingTranslationHandler;\n this.useDefaultLang = useDefaultLang;\n this.isolate = isolate;\n this.extend = extend;\n this.pending = false;\n this._onTranslationChange = new EventEmitter();\n this._onLangChange = new EventEmitter();\n this._onDefaultLangChange = new EventEmitter();\n this._langs = [];\n this._translations = {};\n this._translationRequests = {};\n /** set the default language from configuration */\n if (defaultLanguage) {\n this.setDefaultLang(defaultLanguage);\n }\n }\n /**\n * An EventEmitter to listen to translation change events\n * onTranslationChange.subscribe((params: TranslationChangeEvent) => {\n * // do something\n * });\n * @return {?}\n */\n get onTranslationChange() {\n return this.isolate ? this._onTranslationChange : this.store.onTranslationChange;\n }\n /**\n * An EventEmitter to listen to lang change events\n * onLangChange.subscribe((params: LangChangeEvent) => {\n * // do something\n * });\n * @return {?}\n */\n get onLangChange() {\n return this.isolate ? this._onLangChange : this.store.onLangChange;\n }\n /**\n * An EventEmitter to listen to default lang change events\n * onDefaultLangChange.subscribe((params: DefaultLangChangeEvent) => {\n * // do something\n * });\n * @return {?}\n */\n get onDefaultLangChange() {\n return this.isolate ? this._onDefaultLangChange : this.store.onDefaultLangChange;\n }\n /**\n * The default lang to fallback when translations are missing on the current lang\n * @return {?}\n */\n get defaultLang() {\n return this.isolate ? this._defaultLang : this.store.defaultLang;\n }\n /**\n * @param {?} defaultLang\n * @return {?}\n */\n set defaultLang(defaultLang) {\n if (this.isolate) {\n this._defaultLang = defaultLang;\n }\n else {\n this.store.defaultLang = defaultLang;\n }\n }\n /**\n * The lang currently used\n * @return {?}\n */\n get currentLang() {\n return this.isolate ? this._currentLang : this.store.currentLang;\n }\n /**\n * @param {?} currentLang\n * @return {?}\n */\n set currentLang(currentLang) {\n if (this.isolate) {\n this._currentLang = currentLang;\n }\n else {\n this.store.currentLang = currentLang;\n }\n }\n /**\n * an array of langs\n * @return {?}\n */\n get langs() {\n return this.isolate ? this._langs : this.store.langs;\n }\n /**\n * @param {?} langs\n * @return {?}\n */\n set langs(langs) {\n if (this.isolate) {\n this._langs = langs;\n }\n else {\n this.store.langs = langs;\n }\n }\n /**\n * a list of translations per lang\n * @return {?}\n */\n get translations() {\n return this.isolate ? this._translations : this.store.translations;\n }\n /**\n * @param {?} translations\n * @return {?}\n */\n set translations(translations) {\n if (this.isolate) {\n this._translations = translations;\n }\n else {\n this.store.translations = translations;\n }\n }\n /**\n * Sets the default language to use as a fallback\n * @param {?} lang\n * @return {?}\n */\n setDefaultLang(lang) {\n if (lang === this.defaultLang) {\n return;\n }\n /** @type {?} */\n let pending = this.retrieveTranslations(lang);\n if (typeof pending !== \"undefined\") {\n // on init set the defaultLang immediately\n if (this.defaultLang == null) {\n this.defaultLang = lang;\n }\n pending.pipe(take(1))\n .subscribe((/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n this.changeDefaultLang(lang);\n }));\n }\n else { // we already have this language\n this.changeDefaultLang(lang);\n }\n }\n /**\n * Gets the default language used\n * @return {?}\n */\n getDefaultLang() {\n return this.defaultLang;\n }\n /**\n * Changes the lang currently used\n * @param {?} lang\n * @return {?}\n */\n use(lang) {\n // don't change the language if the language given is already selected\n if (lang === this.currentLang) {\n return of(this.translations[lang]);\n }\n /** @type {?} */\n let pending = this.retrieveTranslations(lang);\n if (typeof pending !== \"undefined\") {\n // on init set the currentLang immediately\n if (!this.currentLang) {\n this.currentLang = lang;\n }\n pending.pipe(take(1))\n .subscribe((/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n this.changeLang(lang);\n }));\n return pending;\n }\n else { // we have this language, return an Observable\n this.changeLang(lang);\n return of(this.translations[lang]);\n }\n }\n /**\n * Retrieves the given translations\n * @private\n * @param {?} lang\n * @return {?}\n */\n retrieveTranslations(lang) {\n /** @type {?} */\n let pending;\n // if this language is unavailable or extend is true, ask for it\n if (typeof this.translations[lang] === \"undefined\" || this.extend) {\n this._translationRequests[lang] = this._translationRequests[lang] || this.getTranslation(lang);\n pending = this._translationRequests[lang];\n }\n return pending;\n }\n /**\n * Gets an object of translations for a given language with the current loader\n * and passes it through the compiler\n * @param {?} lang\n * @return {?}\n */\n getTranslation(lang) {\n this.pending = true;\n /** @type {?} */\n const loadingTranslations = this.currentLoader.getTranslation(lang).pipe(shareReplay(1), take(1));\n this.loadingTranslations = loadingTranslations.pipe(map((/**\n * @param {?} res\n * @return {?}\n */\n (res) => this.compiler.compileTranslations(res, lang))), shareReplay(1), take(1));\n this.loadingTranslations\n .subscribe({\n next: (/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n this.translations[lang] = this.extend && this.translations[lang] ? Object.assign(Object.assign({}, res), this.translations[lang]) : res;\n this.updateLangs();\n this.pending = false;\n }),\n error: (/**\n * @param {?} err\n * @return {?}\n */\n (err) => {\n this.pending = false;\n })\n });\n return loadingTranslations;\n }\n /**\n * Manually sets an object of translations for a given language\n * after passing it through the compiler\n * @param {?} lang\n * @param {?} translations\n * @param {?=} shouldMerge\n * @return {?}\n */\n setTranslation(lang, translations, shouldMerge = false) {\n translations = this.compiler.compileTranslations(translations, lang);\n if ((shouldMerge || this.extend) && this.translations[lang]) {\n this.translations[lang] = mergeDeep(this.translations[lang], translations);\n }\n else {\n this.translations[lang] = translations;\n }\n this.updateLangs();\n this.onTranslationChange.emit({ lang: lang, translations: this.translations[lang] });\n }\n /**\n * Returns an array of currently available langs\n * @return {?}\n */\n getLangs() {\n return this.langs;\n }\n /**\n * Add available langs\n * @param {?} langs\n * @return {?}\n */\n addLangs(langs) {\n langs.forEach((/**\n * @param {?} lang\n * @return {?}\n */\n (lang) => {\n if (this.langs.indexOf(lang) === -1) {\n this.langs.push(lang);\n }\n }));\n }\n /**\n * Update the list of available langs\n * @private\n * @return {?}\n */\n updateLangs() {\n this.addLangs(Object.keys(this.translations));\n }\n /**\n * Returns the parsed result of the translations\n * @param {?} translations\n * @param {?} key\n * @param {?=} interpolateParams\n * @return {?}\n */\n getParsedResult(translations, key, interpolateParams) {\n /** @type {?} */\n let res;\n if (key instanceof Array) {\n /** @type {?} */\n let result = {};\n /** @type {?} */\n let observables = false;\n for (let k of key) {\n result[k] = this.getParsedResult(translations, k, interpolateParams);\n if (isObservable(result[k])) {\n observables = true;\n }\n }\n if (observables) {\n /** @type {?} */\n const sources = key.map((/**\n * @param {?} k\n * @return {?}\n */\n k => isObservable(result[k]) ? result[k] : of((/** @type {?} */ (result[k])))));\n return forkJoin(sources).pipe(map((/**\n * @param {?} arr\n * @return {?}\n */\n (arr) => {\n /** @type {?} */\n let obj = {};\n arr.forEach((/**\n * @param {?} value\n * @param {?} index\n * @return {?}\n */\n (value, index) => {\n obj[key[index]] = value;\n }));\n return obj;\n })));\n }\n return result;\n }\n if (translations) {\n res = this.parser.interpolate(this.parser.getValue(translations, key), interpolateParams);\n }\n if (typeof res === \"undefined\" && this.defaultLang != null && this.defaultLang !== this.currentLang && this.useDefaultLang) {\n res = this.parser.interpolate(this.parser.getValue(this.translations[this.defaultLang], key), interpolateParams);\n }\n if (typeof res === \"undefined\") {\n /** @type {?} */\n let params = { key, translateService: this };\n if (typeof interpolateParams !== 'undefined') {\n params.interpolateParams = interpolateParams;\n }\n res = this.missingTranslationHandler.handle(params);\n }\n return typeof res !== \"undefined\" ? res : key;\n }\n /**\n * Gets the translated value of a key (or an array of keys)\n * @param {?} key\n * @param {?=} interpolateParams\n * @return {?} the translated key, or an object of translated keys\n */\n get(key, interpolateParams) {\n if (!isDefined(key) || !key.length) {\n throw new Error(`Parameter \"key\" required`);\n }\n // check if we are loading a new translation to use\n if (this.pending) {\n return this.loadingTranslations.pipe(concatMap((/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n res = this.getParsedResult(res, key, interpolateParams);\n return isObservable(res) ? res : of(res);\n })));\n }\n else {\n /** @type {?} */\n let res = this.getParsedResult(this.translations[this.currentLang], key, interpolateParams);\n return isObservable(res) ? res : of(res);\n }\n }\n /**\n * Returns a stream of translated values of a key (or an array of keys) which updates\n * whenever the translation changes.\n * @param {?} key\n * @param {?=} interpolateParams\n * @return {?} A stream of the translated key, or an object of translated keys\n */\n getStreamOnTranslationChange(key, interpolateParams) {\n if (!isDefined(key) || !key.length) {\n throw new Error(`Parameter \"key\" required`);\n }\n return concat(defer((/**\n * @return {?}\n */\n () => this.get(key, interpolateParams))), this.onTranslationChange.pipe(switchMap((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n /** @type {?} */\n const res = this.getParsedResult(event.translations, key, interpolateParams);\n if (typeof res.subscribe === 'function') {\n return res;\n }\n else {\n return of(res);\n }\n }))));\n }\n /**\n * Returns a stream of translated values of a key (or an array of keys) which updates\n * whenever the language changes.\n * @param {?} key\n * @param {?=} interpolateParams\n * @return {?} A stream of the translated key, or an object of translated keys\n */\n stream(key, interpolateParams) {\n if (!isDefined(key) || !key.length) {\n throw new Error(`Parameter \"key\" required`);\n }\n return concat(defer((/**\n * @return {?}\n */\n () => this.get(key, interpolateParams))), this.onLangChange.pipe(switchMap((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n /** @type {?} */\n const res = this.getParsedResult(event.translations, key, interpolateParams);\n return isObservable(res) ? res : of(res);\n }))));\n }\n /**\n * Returns a translation instantly from the internal state of loaded translation.\n * All rules regarding the current language, the preferred language of even fallback languages will be used except any promise handling.\n * @param {?} key\n * @param {?=} interpolateParams\n * @return {?}\n */\n instant(key, interpolateParams) {\n if (!isDefined(key) || !key.length) {\n throw new Error(`Parameter \"key\" required`);\n }\n /** @type {?} */\n let res = this.getParsedResult(this.translations[this.currentLang], key, interpolateParams);\n if (isObservable(res)) {\n if (key instanceof Array) {\n /** @type {?} */\n let obj = {};\n key.forEach((/**\n * @param {?} value\n * @param {?} index\n * @return {?}\n */\n (value, index) => {\n obj[key[index]] = key[index];\n }));\n return obj;\n }\n return key;\n }\n else {\n return res;\n }\n }\n /**\n * Sets the translated value of a key, after compiling it\n * @param {?} key\n * @param {?} value\n * @param {?=} lang\n * @return {?}\n */\n set(key, value, lang = this.currentLang) {\n this.translations[lang][key] = this.compiler.compile(value, lang);\n this.updateLangs();\n this.onTranslationChange.emit({ lang: lang, translations: this.translations[lang] });\n }\n /**\n * Changes the current lang\n * @private\n * @param {?} lang\n * @return {?}\n */\n changeLang(lang) {\n this.currentLang = lang;\n this.onLangChange.emit({ lang: lang, translations: this.translations[lang] });\n // if there is no default lang, use the one that we just set\n if (this.defaultLang == null) {\n this.changeDefaultLang(lang);\n }\n }\n /**\n * Changes the default lang\n * @private\n * @param {?} lang\n * @return {?}\n */\n changeDefaultLang(lang) {\n this.defaultLang = lang;\n this.onDefaultLangChange.emit({ lang: lang, translations: this.translations[lang] });\n }\n /**\n * Allows to reload the lang file from the file\n * @param {?} lang\n * @return {?}\n */\n reloadLang(lang) {\n this.resetLang(lang);\n return this.getTranslation(lang);\n }\n /**\n * Deletes inner translation\n * @param {?} lang\n * @return {?}\n */\n resetLang(lang) {\n this._translationRequests[lang] = undefined;\n this.translations[lang] = undefined;\n }\n /**\n * Returns the language code name from the browser, e.g. \"de\"\n * @return {?}\n */\n getBrowserLang() {\n if (typeof window === 'undefined' || typeof window.navigator === 'undefined') {\n return undefined;\n }\n /** @type {?} */\n let browserLang = window.navigator.languages ? window.navigator.languages[0] : null;\n browserLang = browserLang || window.navigator.language || window.navigator.browserLanguage || window.navigator.userLanguage;\n if (typeof browserLang === 'undefined') {\n return undefined;\n }\n if (browserLang.indexOf('-') !== -1) {\n browserLang = browserLang.split('-')[0];\n }\n if (browserLang.indexOf('_') !== -1) {\n browserLang = browserLang.split('_')[0];\n }\n return browserLang;\n }\n /**\n * Returns the culture language code name from the browser, e.g. \"de-DE\"\n * @return {?}\n */\n getBrowserCultureLang() {\n if (typeof window === 'undefined' || typeof window.navigator === 'undefined') {\n return undefined;\n }\n /** @type {?} */\n let browserCultureLang = window.navigator.languages ? window.navigator.languages[0] : null;\n browserCultureLang = browserCultureLang || window.navigator.language || window.navigator.browserLanguage || window.navigator.userLanguage;\n return browserCultureLang;\n }\n}\nTranslateService.ɵfac = function TranslateService_Factory(t) { return new (t || TranslateService)(ɵngcc0.ɵɵinject(TranslateStore), ɵngcc0.ɵɵinject(TranslateLoader), ɵngcc0.ɵɵinject(TranslateCompiler), ɵngcc0.ɵɵinject(TranslateParser), ɵngcc0.ɵɵinject(MissingTranslationHandler), ɵngcc0.ɵɵinject(USE_DEFAULT_LANG), ɵngcc0.ɵɵinject(USE_STORE), ɵngcc0.ɵɵinject(USE_EXTEND), ɵngcc0.ɵɵinject(DEFAULT_LANGUAGE)); };\nTranslateService.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: TranslateService, factory: TranslateService.ɵfac });\n/** @nocollapse */\nTranslateService.ctorParameters = () => [\n { type: TranslateStore },\n { type: TranslateLoader },\n { type: TranslateCompiler },\n { type: TranslateParser },\n { type: MissingTranslationHandler },\n { type: Boolean, decorators: [{ type: Inject, args: [USE_DEFAULT_LANG,] }] },\n { type: Boolean, decorators: [{ type: Inject, args: [USE_STORE,] }] },\n { type: Boolean, decorators: [{ type: Inject, args: [USE_EXTEND,] }] },\n { type: String, decorators: [{ type: Inject, args: [DEFAULT_LANGUAGE,] }] }\n];\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateService, [{\n type: Injectable\n }], function () { return [{ type: TranslateStore }, { type: TranslateLoader }, { type: TranslateCompiler }, { type: TranslateParser }, { type: MissingTranslationHandler }, { type: Boolean, decorators: [{\n type: Inject,\n args: [USE_DEFAULT_LANG]\n }] }, { type: Boolean, decorators: [{\n type: Inject,\n args: [USE_STORE]\n }] }, { type: Boolean, decorators: [{\n type: Inject,\n args: [USE_EXTEND]\n }] }, { type: String, decorators: [{\n type: Inject,\n args: [DEFAULT_LANGUAGE]\n }] }]; }, null); })();\nif (false) {\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype.loadingTranslations;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype.pending;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._onTranslationChange;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._onLangChange;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._onDefaultLangChange;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._defaultLang;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._currentLang;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._langs;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._translations;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype._translationRequests;\n /** @type {?} */\n TranslateService.prototype.store;\n /** @type {?} */\n TranslateService.prototype.currentLoader;\n /** @type {?} */\n TranslateService.prototype.compiler;\n /** @type {?} */\n TranslateService.prototype.parser;\n /** @type {?} */\n TranslateService.prototype.missingTranslationHandler;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype.useDefaultLang;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype.isolate;\n /**\n * @type {?}\n * @private\n */\n TranslateService.prototype.extend;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.directive.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\nclass TranslateDirective {\n /**\n * @param {?} translateService\n * @param {?} element\n * @param {?} _ref\n */\n constructor(translateService, element, _ref) {\n this.translateService = translateService;\n this.element = element;\n this._ref = _ref;\n // subscribe to onTranslationChange event, in case the translations of the current lang change\n if (!this.onTranslationChangeSub) {\n this.onTranslationChangeSub = this.translateService.onTranslationChange.subscribe((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n if (event.lang === this.translateService.currentLang) {\n this.checkNodes(true, event.translations);\n }\n }));\n }\n // subscribe to onLangChange event, in case the language changes\n if (!this.onLangChangeSub) {\n this.onLangChangeSub = this.translateService.onLangChange.subscribe((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n this.checkNodes(true, event.translations);\n }));\n }\n // subscribe to onDefaultLangChange event, in case the default language changes\n if (!this.onDefaultLangChangeSub) {\n this.onDefaultLangChangeSub = this.translateService.onDefaultLangChange.subscribe((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n this.checkNodes(true);\n }));\n }\n }\n /**\n * @param {?} key\n * @return {?}\n */\n set translate(key) {\n if (key) {\n this.key = key;\n this.checkNodes();\n }\n }\n /**\n * @param {?} params\n * @return {?}\n */\n set translateParams(params) {\n if (!equals(this.currentParams, params)) {\n this.currentParams = params;\n this.checkNodes(true);\n }\n }\n /**\n * @return {?}\n */\n ngAfterViewChecked() {\n this.checkNodes();\n }\n /**\n * @param {?=} forceUpdate\n * @param {?=} translations\n * @return {?}\n */\n checkNodes(forceUpdate = false, translations) {\n /** @type {?} */\n let nodes = this.element.nativeElement.childNodes;\n // if the element is empty\n if (!nodes.length) {\n // we add the key as content\n this.setContent(this.element.nativeElement, this.key);\n nodes = this.element.nativeElement.childNodes;\n }\n for (let i = 0; i < nodes.length; ++i) {\n /** @type {?} */\n let node = nodes[i];\n if (node.nodeType === 3) { // node type 3 is a text node\n // node type 3 is a text node\n /** @type {?} */\n let key;\n if (forceUpdate) {\n node.lastKey = null;\n }\n if (isDefined(node.lookupKey)) {\n key = node.lookupKey;\n }\n else if (this.key) {\n key = this.key;\n }\n else {\n /** @type {?} */\n let content = this.getContent(node);\n /** @type {?} */\n let trimmedContent = content.trim();\n if (trimmedContent.length) {\n node.lookupKey = trimmedContent;\n // we want to use the content as a key, not the translation value\n if (content !== node.currentValue) {\n key = trimmedContent;\n // the content was changed from the user, we'll use it as a reference if needed\n node.originalContent = content || node.originalContent;\n }\n else if (node.originalContent) { // the content seems ok, but the lang has changed\n // the current content is the translation, not the key, use the last real content as key\n key = node.originalContent.trim();\n }\n else if (content !== node.currentValue) {\n // we want to use the content as a key, not the translation value\n key = trimmedContent;\n // the content was changed from the user, we'll use it as a reference if needed\n node.originalContent = content || node.originalContent;\n }\n }\n }\n this.updateValue(key, node, translations);\n }\n }\n }\n /**\n * @param {?} key\n * @param {?} node\n * @param {?} translations\n * @return {?}\n */\n updateValue(key, node, translations) {\n if (key) {\n if (node.lastKey === key && this.lastParams === this.currentParams) {\n return;\n }\n this.lastParams = this.currentParams;\n /** @type {?} */\n let onTranslation = (/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n if (res !== key) {\n node.lastKey = key;\n }\n if (!node.originalContent) {\n node.originalContent = this.getContent(node);\n }\n node.currentValue = isDefined(res) ? res : (node.originalContent || key);\n // we replace in the original content to preserve spaces that we might have trimmed\n this.setContent(node, this.key ? node.currentValue : node.originalContent.replace(key, node.currentValue));\n this._ref.markForCheck();\n });\n if (isDefined(translations)) {\n /** @type {?} */\n let res = this.translateService.getParsedResult(translations, key, this.currentParams);\n if (isObservable(res)) {\n res.subscribe(onTranslation);\n }\n else {\n onTranslation(res);\n }\n }\n else {\n this.translateService.get(key, this.currentParams).subscribe(onTranslation);\n }\n }\n }\n /**\n * @param {?} node\n * @return {?}\n */\n getContent(node) {\n return isDefined(node.textContent) ? node.textContent : node.data;\n }\n /**\n * @param {?} node\n * @param {?} content\n * @return {?}\n */\n setContent(node, content) {\n if (isDefined(node.textContent)) {\n node.textContent = content;\n }\n else {\n node.data = content;\n }\n }\n /**\n * @return {?}\n */\n ngOnDestroy() {\n if (this.onLangChangeSub) {\n this.onLangChangeSub.unsubscribe();\n }\n if (this.onDefaultLangChangeSub) {\n this.onDefaultLangChangeSub.unsubscribe();\n }\n if (this.onTranslationChangeSub) {\n this.onTranslationChangeSub.unsubscribe();\n }\n }\n}\nTranslateDirective.ɵfac = function TranslateDirective_Factory(t) { return new (t || TranslateDirective)(ɵngcc0.ɵɵdirectiveInject(TranslateService), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ElementRef), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef)); };\nTranslateDirective.ɵdir = /*@__PURE__*/ ɵngcc0.ɵɵdefineDirective({ type: TranslateDirective, selectors: [[\"\", \"translate\", \"\"], [\"\", \"ngx-translate\", \"\"]], inputs: { translate: \"translate\", translateParams: \"translateParams\" } });\n/** @nocollapse */\nTranslateDirective.ctorParameters = () => [\n { type: TranslateService },\n { type: ElementRef },\n { type: ChangeDetectorRef }\n];\nTranslateDirective.propDecorators = {\n translate: [{ type: Input }],\n translateParams: [{ type: Input }]\n};\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateDirective, [{\n type: Directive,\n args: [{\n selector: '[translate],[ngx-translate]'\n }]\n }], function () { return [{ type: TranslateService }, { type: ɵngcc0.ElementRef }, { type: ɵngcc0.ChangeDetectorRef }]; }, { translate: [{\n type: Input\n }], translateParams: [{\n type: Input\n }] }); })();\nif (false) {\n /** @type {?} */\n TranslateDirective.prototype.key;\n /** @type {?} */\n TranslateDirective.prototype.lastParams;\n /** @type {?} */\n TranslateDirective.prototype.currentParams;\n /** @type {?} */\n TranslateDirective.prototype.onLangChangeSub;\n /** @type {?} */\n TranslateDirective.prototype.onDefaultLangChangeSub;\n /** @type {?} */\n TranslateDirective.prototype.onTranslationChangeSub;\n /**\n * @type {?}\n * @private\n */\n TranslateDirective.prototype.translateService;\n /**\n * @type {?}\n * @private\n */\n TranslateDirective.prototype.element;\n /**\n * @type {?}\n * @private\n */\n TranslateDirective.prototype._ref;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: lib/translate.pipe.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\nclass TranslatePipe {\n /**\n * @param {?} translate\n * @param {?} _ref\n */\n constructor(translate, _ref) {\n this.translate = translate;\n this._ref = _ref;\n this.value = '';\n }\n /**\n * @param {?} key\n * @param {?=} interpolateParams\n * @param {?=} translations\n * @return {?}\n */\n updateValue(key, interpolateParams, translations) {\n /** @type {?} */\n let onTranslation = (/**\n * @param {?} res\n * @return {?}\n */\n (res) => {\n this.value = res !== undefined ? res : key;\n this.lastKey = key;\n this._ref.markForCheck();\n });\n if (translations) {\n /** @type {?} */\n let res = this.translate.getParsedResult(translations, key, interpolateParams);\n if (isObservable(res.subscribe)) {\n res.subscribe(onTranslation);\n }\n else {\n onTranslation(res);\n }\n }\n this.translate.get(key, interpolateParams).subscribe(onTranslation);\n }\n /**\n * @param {?} query\n * @param {...?} args\n * @return {?}\n */\n transform(query, ...args) {\n if (!query || !query.length) {\n return query;\n }\n // if we ask another time for the same key, return the last value\n if (equals(query, this.lastKey) && equals(args, this.lastParams)) {\n return this.value;\n }\n /** @type {?} */\n let interpolateParams;\n if (isDefined(args[0]) && args.length) {\n if (typeof args[0] === 'string' && args[0].length) {\n // we accept objects written in the template such as {n:1}, {'n':1}, {n:'v'}\n // which is why we might need to change it to real JSON objects such as {\"n\":1} or {\"n\":\"v\"}\n /** @type {?} */\n let validArgs = args[0]\n .replace(/(\\')?([a-zA-Z0-9_]+)(\\')?(\\s)?:/g, '\"$2\":')\n .replace(/:(\\s)?(\\')(.*?)(\\')/g, ':\"$3\"');\n try {\n interpolateParams = JSON.parse(validArgs);\n }\n catch (e) {\n throw new SyntaxError(`Wrong parameter in TranslatePipe. Expected a valid Object, received: ${args[0]}`);\n }\n }\n else if (typeof args[0] === 'object' && !Array.isArray(args[0])) {\n interpolateParams = args[0];\n }\n }\n // store the query, in case it changes\n this.lastKey = query;\n // store the params, in case they change\n this.lastParams = args;\n // set the value\n this.updateValue(query, interpolateParams);\n // if there is a subscription to onLangChange, clean it\n this._dispose();\n // subscribe to onTranslationChange event, in case the translations change\n if (!this.onTranslationChange) {\n this.onTranslationChange = this.translate.onTranslationChange.subscribe((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n if (this.lastKey && event.lang === this.translate.currentLang) {\n this.lastKey = null;\n this.updateValue(query, interpolateParams, event.translations);\n }\n }));\n }\n // subscribe to onLangChange event, in case the language changes\n if (!this.onLangChange) {\n this.onLangChange = this.translate.onLangChange.subscribe((/**\n * @param {?} event\n * @return {?}\n */\n (event) => {\n if (this.lastKey) {\n this.lastKey = null; // we want to make sure it doesn't return the same value until it's been updated\n this.updateValue(query, interpolateParams, event.translations);\n }\n }));\n }\n // subscribe to onDefaultLangChange event, in case the default language changes\n if (!this.onDefaultLangChange) {\n this.onDefaultLangChange = this.translate.onDefaultLangChange.subscribe((/**\n * @return {?}\n */\n () => {\n if (this.lastKey) {\n this.lastKey = null; // we want to make sure it doesn't return the same value until it's been updated\n this.updateValue(query, interpolateParams);\n }\n }));\n }\n return this.value;\n }\n /**\n * Clean any existing subscription to change events\n * @private\n * @return {?}\n */\n _dispose() {\n if (typeof this.onTranslationChange !== 'undefined') {\n this.onTranslationChange.unsubscribe();\n this.onTranslationChange = undefined;\n }\n if (typeof this.onLangChange !== 'undefined') {\n this.onLangChange.unsubscribe();\n this.onLangChange = undefined;\n }\n if (typeof this.onDefaultLangChange !== 'undefined') {\n this.onDefaultLangChange.unsubscribe();\n this.onDefaultLangChange = undefined;\n }\n }\n /**\n * @return {?}\n */\n ngOnDestroy() {\n this._dispose();\n }\n}\nTranslatePipe.ɵfac = function TranslatePipe_Factory(t) { return new (t || TranslatePipe)(ɵngcc0.ɵɵdirectiveInject(TranslateService, 16), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef, 16)); };\nTranslatePipe.ɵpipe = /*@__PURE__*/ ɵngcc0.ɵɵdefinePipe({ name: \"translate\", type: TranslatePipe, pure: false });\nTranslatePipe.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: TranslatePipe, factory: TranslatePipe.ɵfac });\n/** @nocollapse */\nTranslatePipe.ctorParameters = () => [\n { type: TranslateService },\n { type: ChangeDetectorRef }\n];\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslatePipe, [{\n type: Injectable\n }, {\n type: Pipe,\n args: [{\n name: 'translate',\n pure: false // required to update the value when the promise is resolved\n }]\n }], function () { return [{ type: TranslateService }, { type: ɵngcc0.ChangeDetectorRef }]; }, null); })();\nif (false) {\n /** @type {?} */\n TranslatePipe.prototype.value;\n /** @type {?} */\n TranslatePipe.prototype.lastKey;\n /** @type {?} */\n TranslatePipe.prototype.lastParams;\n /** @type {?} */\n TranslatePipe.prototype.onTranslationChange;\n /** @type {?} */\n TranslatePipe.prototype.onLangChange;\n /** @type {?} */\n TranslatePipe.prototype.onDefaultLangChange;\n /**\n * @type {?}\n * @private\n */\n TranslatePipe.prototype.translate;\n /**\n * @type {?}\n * @private\n */\n TranslatePipe.prototype._ref;\n}\n\n/**\n * @fileoverview added by tsickle\n * Generated from: public_api.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n/**\n * @record\n */\nfunction TranslateModuleConfig() { }\nif (false) {\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.loader;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.compiler;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.parser;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.missingTranslationHandler;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.isolate;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.extend;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.useDefaultLang;\n /** @type {?|undefined} */\n TranslateModuleConfig.prototype.defaultLanguage;\n}\nclass TranslateModule {\n /**\n * Use this method in your root module to provide the TranslateService\n * @param {?=} config\n * @return {?}\n */\n static forRoot(config = {}) {\n return {\n ngModule: TranslateModule,\n providers: [\n config.loader || { provide: TranslateLoader, useClass: TranslateFakeLoader },\n config.compiler || { provide: TranslateCompiler, useClass: TranslateFakeCompiler },\n config.parser || { provide: TranslateParser, useClass: TranslateDefaultParser },\n config.missingTranslationHandler || { provide: MissingTranslationHandler, useClass: FakeMissingTranslationHandler },\n TranslateStore,\n { provide: USE_STORE, useValue: config.isolate },\n { provide: USE_DEFAULT_LANG, useValue: config.useDefaultLang },\n { provide: USE_EXTEND, useValue: config.extend },\n { provide: DEFAULT_LANGUAGE, useValue: config.defaultLanguage },\n TranslateService\n ]\n };\n }\n /**\n * Use this method in your other (non root) modules to import the directive/pipe\n * @param {?=} config\n * @return {?}\n */\n static forChild(config = {}) {\n return {\n ngModule: TranslateModule,\n providers: [\n config.loader || { provide: TranslateLoader, useClass: TranslateFakeLoader },\n config.compiler || { provide: TranslateCompiler, useClass: TranslateFakeCompiler },\n config.parser || { provide: TranslateParser, useClass: TranslateDefaultParser },\n config.missingTranslationHandler || { provide: MissingTranslationHandler, useClass: FakeMissingTranslationHandler },\n { provide: USE_STORE, useValue: config.isolate },\n { provide: USE_DEFAULT_LANG, useValue: config.useDefaultLang },\n { provide: USE_EXTEND, useValue: config.extend },\n { provide: DEFAULT_LANGUAGE, useValue: config.defaultLanguage },\n TranslateService\n ]\n };\n }\n}\nTranslateModule.ɵfac = function TranslateModule_Factory(t) { return new (t || TranslateModule)(); };\nTranslateModule.ɵmod = /*@__PURE__*/ ɵngcc0.ɵɵdefineNgModule({ type: TranslateModule });\nTranslateModule.ɵinj = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjector({});\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(TranslateModule, [{\n type: NgModule,\n args: [{\n declarations: [\n TranslatePipe,\n TranslateDirective\n ],\n exports: [\n TranslatePipe,\n TranslateDirective\n ]\n }]\n }], null, null); })();\n(function () { (typeof ngJitMode === \"undefined\" || ngJitMode) && ɵngcc0.ɵɵsetNgModuleScope(TranslateModule, { declarations: [TranslatePipe, TranslateDirective], exports: [TranslatePipe, TranslateDirective] }); })();\n\n/**\n * @fileoverview added by tsickle\n * Generated from: ngx-translate-core.ts\n * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc\n */\n\nexport { DEFAULT_LANGUAGE, FakeMissingTranslationHandler, MissingTranslationHandler, TranslateCompiler, TranslateDefaultParser, TranslateDirective, TranslateFakeCompiler, TranslateFakeLoader, TranslateLoader, TranslateModule, TranslateParser, TranslatePipe, TranslateService, TranslateStore, USE_DEFAULT_LANG, USE_EXTEND, USE_STORE };\n\n","class TranslateHttpLoader {\n constructor(http, prefix = \"/assets/i18n/\", suffix = \".json\") {\n this.http = http;\n this.prefix = prefix;\n this.suffix = suffix;\n }\n /**\n * Gets the translations from the server\n */\n getTranslation(lang) {\n return this.http.get(`${this.prefix}${lang}${this.suffix}`);\n }\n}\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { TranslateHttpLoader };\n\n","// Adapted from https://github.com/Flet/prettier-bytes/\n// Changing 1000 bytes to 1024, so we can keep uppercase KB vs kB\n// ISC License (c) Dan Flettre https://github.com/Flet/prettier-bytes/blob/master/LICENSE\nmodule.exports = function prettierBytes (num) {\n if (typeof num !== 'number' || isNaN(num)) {\n throw new TypeError('Expected a number, got ' + typeof num)\n }\n\n var neg = num < 0\n var units = ['B', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']\n\n if (neg) {\n num = -num\n }\n\n if (num < 1) {\n return (neg ? '-' : '') + num + ' B'\n }\n\n var exponent = Math.min(Math.floor(Math.log(num) / Math.log(1024)), units.length - 1)\n num = Number(num / Math.pow(1024, exponent))\n var unit = units[exponent]\n\n if (num >= 10 || num % 1 === 0) {\n // Do not show decimals when the number is two-digit, or if the number has no\n // decimal component.\n return (neg ? '-' : '') + num.toFixed(0) + ' ' + unit\n } else {\n return (neg ? '-' : '') + num.toFixed(1) + ' ' + unit\n }\n}\n","/*!\n * Cropper.js v1.5.7\n * https://fengyuanchen.github.io/cropperjs\n *\n * Copyright 2015-present Chen Fengyuan\n * Released under the MIT license\n *\n * Date: 2020-05-23T05:23:00.081Z\n */\n\n(function (global, factory) {\n typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n typeof define === 'function' && define.amd ? define(factory) :\n (global = global || self, global.Cropper = factory());\n}(this, (function () { 'use strict';\n\n function _typeof(obj) {\n \"@babel/helpers - typeof\";\n\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\n function _classCallCheck(instance, Constructor) {\n if (!(instance instanceof Constructor)) {\n throw new TypeError(\"Cannot call a class as a function\");\n }\n }\n\n function _defineProperties(target, props) {\n for (var i = 0; i < props.length; i++) {\n var descriptor = props[i];\n descriptor.enumerable = descriptor.enumerable || false;\n descriptor.configurable = true;\n if (\"value\" in descriptor) descriptor.writable = true;\n Object.defineProperty(target, descriptor.key, descriptor);\n }\n }\n\n function _createClass(Constructor, protoProps, staticProps) {\n if (protoProps) _defineProperties(Constructor.prototype, protoProps);\n if (staticProps) _defineProperties(Constructor, staticProps);\n return Constructor;\n }\n\n function _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\n function ownKeys(object, enumerableOnly) {\n var keys = Object.keys(object);\n\n if (Object.getOwnPropertySymbols) {\n var symbols = Object.getOwnPropertySymbols(object);\n if (enumerableOnly) symbols = symbols.filter(function (sym) {\n return Object.getOwnPropertyDescriptor(object, sym).enumerable;\n });\n keys.push.apply(keys, symbols);\n }\n\n return keys;\n }\n\n function _objectSpread2(target) {\n for (var i = 1; i < arguments.length; i++) {\n var source = arguments[i] != null ? arguments[i] : {};\n\n if (i % 2) {\n ownKeys(Object(source), true).forEach(function (key) {\n _defineProperty(target, key, source[key]);\n });\n } else if (Object.getOwnPropertyDescriptors) {\n Object.defineProperties(target, Object.getOwnPropertyDescriptors(source));\n } else {\n ownKeys(Object(source)).forEach(function (key) {\n Object.defineProperty(target, key, Object.getOwnPropertyDescriptor(source, key));\n });\n }\n }\n\n return target;\n }\n\n function _toConsumableArray(arr) {\n return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread();\n }\n\n function _arrayWithoutHoles(arr) {\n if (Array.isArray(arr)) return _arrayLikeToArray(arr);\n }\n\n function _iterableToArray(iter) {\n if (typeof Symbol !== \"undefined\" && Symbol.iterator in Object(iter)) return Array.from(iter);\n }\n\n function _unsupportedIterableToArray(o, minLen) {\n if (!o) return;\n if (typeof o === \"string\") return _arrayLikeToArray(o, minLen);\n var n = Object.prototype.toString.call(o).slice(8, -1);\n if (n === \"Object\" && o.constructor) n = o.constructor.name;\n if (n === \"Map\" || n === \"Set\") return Array.from(o);\n if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);\n }\n\n function _arrayLikeToArray(arr, len) {\n if (len == null || len > arr.length) len = arr.length;\n\n for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];\n\n return arr2;\n }\n\n function _nonIterableSpread() {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n\n var IS_BROWSER = typeof window !== 'undefined' && typeof window.document !== 'undefined';\n var WINDOW = IS_BROWSER ? window : {};\n var IS_TOUCH_DEVICE = IS_BROWSER && WINDOW.document.documentElement ? 'ontouchstart' in WINDOW.document.documentElement : false;\n var HAS_POINTER_EVENT = IS_BROWSER ? 'PointerEvent' in WINDOW : false;\n var NAMESPACE = 'cropper'; // Actions\n\n var ACTION_ALL = 'all';\n var ACTION_CROP = 'crop';\n var ACTION_MOVE = 'move';\n var ACTION_ZOOM = 'zoom';\n var ACTION_EAST = 'e';\n var ACTION_WEST = 'w';\n var ACTION_SOUTH = 's';\n var ACTION_NORTH = 'n';\n var ACTION_NORTH_EAST = 'ne';\n var ACTION_NORTH_WEST = 'nw';\n var ACTION_SOUTH_EAST = 'se';\n var ACTION_SOUTH_WEST = 'sw'; // Classes\n\n var CLASS_CROP = \"\".concat(NAMESPACE, \"-crop\");\n var CLASS_DISABLED = \"\".concat(NAMESPACE, \"-disabled\");\n var CLASS_HIDDEN = \"\".concat(NAMESPACE, \"-hidden\");\n var CLASS_HIDE = \"\".concat(NAMESPACE, \"-hide\");\n var CLASS_INVISIBLE = \"\".concat(NAMESPACE, \"-invisible\");\n var CLASS_MODAL = \"\".concat(NAMESPACE, \"-modal\");\n var CLASS_MOVE = \"\".concat(NAMESPACE, \"-move\"); // Data keys\n\n var DATA_ACTION = \"\".concat(NAMESPACE, \"Action\");\n var DATA_PREVIEW = \"\".concat(NAMESPACE, \"Preview\"); // Drag modes\n\n var DRAG_MODE_CROP = 'crop';\n var DRAG_MODE_MOVE = 'move';\n var DRAG_MODE_NONE = 'none'; // Events\n\n var EVENT_CROP = 'crop';\n var EVENT_CROP_END = 'cropend';\n var EVENT_CROP_MOVE = 'cropmove';\n var EVENT_CROP_START = 'cropstart';\n var EVENT_DBLCLICK = 'dblclick';\n var EVENT_TOUCH_START = IS_TOUCH_DEVICE ? 'touchstart' : 'mousedown';\n var EVENT_TOUCH_MOVE = IS_TOUCH_DEVICE ? 'touchmove' : 'mousemove';\n var EVENT_TOUCH_END = IS_TOUCH_DEVICE ? 'touchend touchcancel' : 'mouseup';\n var EVENT_POINTER_DOWN = HAS_POINTER_EVENT ? 'pointerdown' : EVENT_TOUCH_START;\n var EVENT_POINTER_MOVE = HAS_POINTER_EVENT ? 'pointermove' : EVENT_TOUCH_MOVE;\n var EVENT_POINTER_UP = HAS_POINTER_EVENT ? 'pointerup pointercancel' : EVENT_TOUCH_END;\n var EVENT_READY = 'ready';\n var EVENT_RESIZE = 'resize';\n var EVENT_WHEEL = 'wheel';\n var EVENT_ZOOM = 'zoom'; // Mime types\n\n var MIME_TYPE_JPEG = 'image/jpeg'; // RegExps\n\n var REGEXP_ACTIONS = /^e|w|s|n|se|sw|ne|nw|all|crop|move|zoom$/;\n var REGEXP_DATA_URL = /^data:/;\n var REGEXP_DATA_URL_JPEG = /^data:image\\/jpeg;base64,/;\n var REGEXP_TAG_NAME = /^img|canvas$/i; // Misc\n\n var DEFAULTS = {\n // Define the view mode of the cropper\n viewMode: 0,\n // 0, 1, 2, 3\n // Define the dragging mode of the cropper\n dragMode: DRAG_MODE_CROP,\n // 'crop', 'move' or 'none'\n // Define the initial aspect ratio of the crop box\n initialAspectRatio: NaN,\n // Define the aspect ratio of the crop box\n aspectRatio: NaN,\n // An object with the previous cropping result data\n data: null,\n // A selector for adding extra containers to preview\n preview: '',\n // Re-render the cropper when resize the window\n responsive: true,\n // Restore the cropped area after resize the window\n restore: true,\n // Check if the current image is a cross-origin image\n checkCrossOrigin: true,\n // Check the current image's Exif Orientation information\n checkOrientation: true,\n // Show the black modal\n modal: true,\n // Show the dashed lines for guiding\n guides: true,\n // Show the center indicator for guiding\n center: true,\n // Show the white modal to highlight the crop box\n highlight: true,\n // Show the grid background\n background: true,\n // Enable to crop the image automatically when initialize\n autoCrop: true,\n // Define the percentage of automatic cropping area when initializes\n autoCropArea: 0.8,\n // Enable to move the image\n movable: true,\n // Enable to rotate the image\n rotatable: true,\n // Enable to scale the image\n scalable: true,\n // Enable to zoom the image\n zoomable: true,\n // Enable to zoom the image by dragging touch\n zoomOnTouch: true,\n // Enable to zoom the image by wheeling mouse\n zoomOnWheel: true,\n // Define zoom ratio when zoom the image by wheeling mouse\n wheelZoomRatio: 0.1,\n // Enable to move the crop box\n cropBoxMovable: true,\n // Enable to resize the crop box\n cropBoxResizable: true,\n // Toggle drag mode between \"crop\" and \"move\" when click twice on the cropper\n toggleDragModeOnDblclick: true,\n // Size limitation\n minCanvasWidth: 0,\n minCanvasHeight: 0,\n minCropBoxWidth: 0,\n minCropBoxHeight: 0,\n minContainerWidth: 200,\n minContainerHeight: 100,\n // Shortcuts of events\n ready: null,\n cropstart: null,\n cropmove: null,\n cropend: null,\n crop: null,\n zoom: null\n };\n\n var TEMPLATE = '
    ' + '
    ' + '
    ' + '
    ' + '
    ' + '
    ' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '
    ' + '
    ';\n\n /**\n * Check if the given value is not a number.\n */\n\n var isNaN = Number.isNaN || WINDOW.isNaN;\n /**\n * Check if the given value is a number.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a number, else `false`.\n */\n\n function isNumber(value) {\n return typeof value === 'number' && !isNaN(value);\n }\n /**\n * Check if the given value is a positive number.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a positive number, else `false`.\n */\n\n var isPositiveNumber = function isPositiveNumber(value) {\n return value > 0 && value < Infinity;\n };\n /**\n * Check if the given value is undefined.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is undefined, else `false`.\n */\n\n function isUndefined(value) {\n return typeof value === 'undefined';\n }\n /**\n * Check if the given value is an object.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is an object, else `false`.\n */\n\n function isObject(value) {\n return _typeof(value) === 'object' && value !== null;\n }\n var hasOwnProperty = Object.prototype.hasOwnProperty;\n /**\n * Check if the given value is a plain object.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a plain object, else `false`.\n */\n\n function isPlainObject(value) {\n if (!isObject(value)) {\n return false;\n }\n\n try {\n var _constructor = value.constructor;\n var prototype = _constructor.prototype;\n return _constructor && prototype && hasOwnProperty.call(prototype, 'isPrototypeOf');\n } catch (error) {\n return false;\n }\n }\n /**\n * Check if the given value is a function.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a function, else `false`.\n */\n\n function isFunction(value) {\n return typeof value === 'function';\n }\n var slice = Array.prototype.slice;\n /**\n * Convert array-like or iterable object to an array.\n * @param {*} value - The value to convert.\n * @returns {Array} Returns a new array.\n */\n\n function toArray(value) {\n return Array.from ? Array.from(value) : slice.call(value);\n }\n /**\n * Iterate the given data.\n * @param {*} data - The data to iterate.\n * @param {Function} callback - The process function for each element.\n * @returns {*} The original data.\n */\n\n function forEach(data, callback) {\n if (data && isFunction(callback)) {\n if (Array.isArray(data) || isNumber(data.length)\n /* array-like */\n ) {\n toArray(data).forEach(function (value, key) {\n callback.call(data, value, key, data);\n });\n } else if (isObject(data)) {\n Object.keys(data).forEach(function (key) {\n callback.call(data, data[key], key, data);\n });\n }\n }\n\n return data;\n }\n /**\n * Extend the given object.\n * @param {*} target - The target object to extend.\n * @param {*} args - The rest objects for merging to the target object.\n * @returns {Object} The extended object.\n */\n\n var assign = Object.assign || function assign(target) {\n for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {\n args[_key - 1] = arguments[_key];\n }\n\n if (isObject(target) && args.length > 0) {\n args.forEach(function (arg) {\n if (isObject(arg)) {\n Object.keys(arg).forEach(function (key) {\n target[key] = arg[key];\n });\n }\n });\n }\n\n return target;\n };\n var REGEXP_DECIMALS = /\\.\\d*(?:0|9){12}\\d*$/;\n /**\n * Normalize decimal number.\n * Check out {@link https://0.30000000000000004.com/}\n * @param {number} value - The value to normalize.\n * @param {number} [times=100000000000] - The times for normalizing.\n * @returns {number} Returns the normalized number.\n */\n\n function normalizeDecimalNumber(value) {\n var times = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 100000000000;\n return REGEXP_DECIMALS.test(value) ? Math.round(value * times) / times : value;\n }\n var REGEXP_SUFFIX = /^width|height|left|top|marginLeft|marginTop$/;\n /**\n * Apply styles to the given element.\n * @param {Element} element - The target element.\n * @param {Object} styles - The styles for applying.\n */\n\n function setStyle(element, styles) {\n var style = element.style;\n forEach(styles, function (value, property) {\n if (REGEXP_SUFFIX.test(property) && isNumber(value)) {\n value = \"\".concat(value, \"px\");\n }\n\n style[property] = value;\n });\n }\n /**\n * Check if the given element has a special class.\n * @param {Element} element - The element to check.\n * @param {string} value - The class to search.\n * @returns {boolean} Returns `true` if the special class was found.\n */\n\n function hasClass(element, value) {\n return element.classList ? element.classList.contains(value) : element.className.indexOf(value) > -1;\n }\n /**\n * Add classes to the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be added.\n */\n\n function addClass(element, value) {\n if (!value) {\n return;\n }\n\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n addClass(elem, value);\n });\n return;\n }\n\n if (element.classList) {\n element.classList.add(value);\n return;\n }\n\n var className = element.className.trim();\n\n if (!className) {\n element.className = value;\n } else if (className.indexOf(value) < 0) {\n element.className = \"\".concat(className, \" \").concat(value);\n }\n }\n /**\n * Remove classes from the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be removed.\n */\n\n function removeClass(element, value) {\n if (!value) {\n return;\n }\n\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n removeClass(elem, value);\n });\n return;\n }\n\n if (element.classList) {\n element.classList.remove(value);\n return;\n }\n\n if (element.className.indexOf(value) >= 0) {\n element.className = element.className.replace(value, '');\n }\n }\n /**\n * Add or remove classes from the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be toggled.\n * @param {boolean} added - Add only.\n */\n\n function toggleClass(element, value, added) {\n if (!value) {\n return;\n }\n\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n toggleClass(elem, value, added);\n });\n return;\n } // IE10-11 doesn't support the second parameter of `classList.toggle`\n\n\n if (added) {\n addClass(element, value);\n } else {\n removeClass(element, value);\n }\n }\n var REGEXP_CAMEL_CASE = /([a-z\\d])([A-Z])/g;\n /**\n * Transform the given string from camelCase to kebab-case\n * @param {string} value - The value to transform.\n * @returns {string} The transformed value.\n */\n\n function toParamCase(value) {\n return value.replace(REGEXP_CAMEL_CASE, '$1-$2').toLowerCase();\n }\n /**\n * Get data from the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to get.\n * @returns {string} The data value.\n */\n\n function getData(element, name) {\n if (isObject(element[name])) {\n return element[name];\n }\n\n if (element.dataset) {\n return element.dataset[name];\n }\n\n return element.getAttribute(\"data-\".concat(toParamCase(name)));\n }\n /**\n * Set data to the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to set.\n * @param {string} data - The data value.\n */\n\n function setData(element, name, data) {\n if (isObject(data)) {\n element[name] = data;\n } else if (element.dataset) {\n element.dataset[name] = data;\n } else {\n element.setAttribute(\"data-\".concat(toParamCase(name)), data);\n }\n }\n /**\n * Remove data from the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to remove.\n */\n\n function removeData(element, name) {\n if (isObject(element[name])) {\n try {\n delete element[name];\n } catch (error) {\n element[name] = undefined;\n }\n } else if (element.dataset) {\n // #128 Safari not allows to delete dataset property\n try {\n delete element.dataset[name];\n } catch (error) {\n element.dataset[name] = undefined;\n }\n } else {\n element.removeAttribute(\"data-\".concat(toParamCase(name)));\n }\n }\n var REGEXP_SPACES = /\\s\\s*/;\n\n var onceSupported = function () {\n var supported = false;\n\n if (IS_BROWSER) {\n var once = false;\n\n var listener = function listener() {};\n\n var options = Object.defineProperty({}, 'once', {\n get: function get() {\n supported = true;\n return once;\n },\n\n /**\n * This setter can fix a `TypeError` in strict mode\n * {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Errors/Getter_only}\n * @param {boolean} value - The value to set\n */\n set: function set(value) {\n once = value;\n }\n });\n WINDOW.addEventListener('test', listener, options);\n WINDOW.removeEventListener('test', listener, options);\n }\n\n return supported;\n }();\n /**\n * Remove event listener from the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Function} listener - The event listener.\n * @param {Object} options - The event options.\n */\n\n\n function removeListener(element, type, listener) {\n var options = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {};\n var handler = listener;\n type.trim().split(REGEXP_SPACES).forEach(function (event) {\n if (!onceSupported) {\n var listeners = element.listeners;\n\n if (listeners && listeners[event] && listeners[event][listener]) {\n handler = listeners[event][listener];\n delete listeners[event][listener];\n\n if (Object.keys(listeners[event]).length === 0) {\n delete listeners[event];\n }\n\n if (Object.keys(listeners).length === 0) {\n delete element.listeners;\n }\n }\n }\n\n element.removeEventListener(event, handler, options);\n });\n }\n /**\n * Add event listener to the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Function} listener - The event listener.\n * @param {Object} options - The event options.\n */\n\n function addListener(element, type, listener) {\n var options = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {};\n var _handler = listener;\n type.trim().split(REGEXP_SPACES).forEach(function (event) {\n if (options.once && !onceSupported) {\n var _element$listeners = element.listeners,\n listeners = _element$listeners === void 0 ? {} : _element$listeners;\n\n _handler = function handler() {\n delete listeners[event][listener];\n element.removeEventListener(event, _handler, options);\n\n for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {\n args[_key2] = arguments[_key2];\n }\n\n listener.apply(element, args);\n };\n\n if (!listeners[event]) {\n listeners[event] = {};\n }\n\n if (listeners[event][listener]) {\n element.removeEventListener(event, listeners[event][listener], options);\n }\n\n listeners[event][listener] = _handler;\n element.listeners = listeners;\n }\n\n element.addEventListener(event, _handler, options);\n });\n }\n /**\n * Dispatch event on the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Object} data - The additional event data.\n * @returns {boolean} Indicate if the event is default prevented or not.\n */\n\n function dispatchEvent(element, type, data) {\n var event; // Event and CustomEvent on IE9-11 are global objects, not constructors\n\n if (isFunction(Event) && isFunction(CustomEvent)) {\n event = new CustomEvent(type, {\n detail: data,\n bubbles: true,\n cancelable: true\n });\n } else {\n event = document.createEvent('CustomEvent');\n event.initCustomEvent(type, true, true, data);\n }\n\n return element.dispatchEvent(event);\n }\n /**\n * Get the offset base on the document.\n * @param {Element} element - The target element.\n * @returns {Object} The offset data.\n */\n\n function getOffset(element) {\n var box = element.getBoundingClientRect();\n return {\n left: box.left + (window.pageXOffset - document.documentElement.clientLeft),\n top: box.top + (window.pageYOffset - document.documentElement.clientTop)\n };\n }\n var location = WINDOW.location;\n var REGEXP_ORIGINS = /^(\\w+:)\\/\\/([^:/?#]*):?(\\d*)/i;\n /**\n * Check if the given URL is a cross origin URL.\n * @param {string} url - The target URL.\n * @returns {boolean} Returns `true` if the given URL is a cross origin URL, else `false`.\n */\n\n function isCrossOriginURL(url) {\n var parts = url.match(REGEXP_ORIGINS);\n return parts !== null && (parts[1] !== location.protocol || parts[2] !== location.hostname || parts[3] !== location.port);\n }\n /**\n * Add timestamp to the given URL.\n * @param {string} url - The target URL.\n * @returns {string} The result URL.\n */\n\n function addTimestamp(url) {\n var timestamp = \"timestamp=\".concat(new Date().getTime());\n return url + (url.indexOf('?') === -1 ? '?' : '&') + timestamp;\n }\n /**\n * Get transforms base on the given object.\n * @param {Object} obj - The target object.\n * @returns {string} A string contains transform values.\n */\n\n function getTransforms(_ref) {\n var rotate = _ref.rotate,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n translateX = _ref.translateX,\n translateY = _ref.translateY;\n var values = [];\n\n if (isNumber(translateX) && translateX !== 0) {\n values.push(\"translateX(\".concat(translateX, \"px)\"));\n }\n\n if (isNumber(translateY) && translateY !== 0) {\n values.push(\"translateY(\".concat(translateY, \"px)\"));\n } // Rotate should come first before scale to match orientation transform\n\n\n if (isNumber(rotate) && rotate !== 0) {\n values.push(\"rotate(\".concat(rotate, \"deg)\"));\n }\n\n if (isNumber(scaleX) && scaleX !== 1) {\n values.push(\"scaleX(\".concat(scaleX, \")\"));\n }\n\n if (isNumber(scaleY) && scaleY !== 1) {\n values.push(\"scaleY(\".concat(scaleY, \")\"));\n }\n\n var transform = values.length ? values.join(' ') : 'none';\n return {\n WebkitTransform: transform,\n msTransform: transform,\n transform: transform\n };\n }\n /**\n * Get the max ratio of a group of pointers.\n * @param {string} pointers - The target pointers.\n * @returns {number} The result ratio.\n */\n\n function getMaxZoomRatio(pointers) {\n var pointers2 = _objectSpread2({}, pointers);\n\n var ratios = [];\n forEach(pointers, function (pointer, pointerId) {\n delete pointers2[pointerId];\n forEach(pointers2, function (pointer2) {\n var x1 = Math.abs(pointer.startX - pointer2.startX);\n var y1 = Math.abs(pointer.startY - pointer2.startY);\n var x2 = Math.abs(pointer.endX - pointer2.endX);\n var y2 = Math.abs(pointer.endY - pointer2.endY);\n var z1 = Math.sqrt(x1 * x1 + y1 * y1);\n var z2 = Math.sqrt(x2 * x2 + y2 * y2);\n var ratio = (z2 - z1) / z1;\n ratios.push(ratio);\n });\n });\n ratios.sort(function (a, b) {\n return Math.abs(a) < Math.abs(b);\n });\n return ratios[0];\n }\n /**\n * Get a pointer from an event object.\n * @param {Object} event - The target event object.\n * @param {boolean} endOnly - Indicates if only returns the end point coordinate or not.\n * @returns {Object} The result pointer contains start and/or end point coordinates.\n */\n\n function getPointer(_ref2, endOnly) {\n var pageX = _ref2.pageX,\n pageY = _ref2.pageY;\n var end = {\n endX: pageX,\n endY: pageY\n };\n return endOnly ? end : _objectSpread2({\n startX: pageX,\n startY: pageY\n }, end);\n }\n /**\n * Get the center point coordinate of a group of pointers.\n * @param {Object} pointers - The target pointers.\n * @returns {Object} The center point coordinate.\n */\n\n function getPointersCenter(pointers) {\n var pageX = 0;\n var pageY = 0;\n var count = 0;\n forEach(pointers, function (_ref3) {\n var startX = _ref3.startX,\n startY = _ref3.startY;\n pageX += startX;\n pageY += startY;\n count += 1;\n });\n pageX /= count;\n pageY /= count;\n return {\n pageX: pageX,\n pageY: pageY\n };\n }\n /**\n * Get the max sizes in a rectangle under the given aspect ratio.\n * @param {Object} data - The original sizes.\n * @param {string} [type='contain'] - The adjust type.\n * @returns {Object} The result sizes.\n */\n\n function getAdjustedSizes(_ref4) // or 'cover'\n {\n var aspectRatio = _ref4.aspectRatio,\n height = _ref4.height,\n width = _ref4.width;\n var type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'contain';\n var isValidWidth = isPositiveNumber(width);\n var isValidHeight = isPositiveNumber(height);\n\n if (isValidWidth && isValidHeight) {\n var adjustedWidth = height * aspectRatio;\n\n if (type === 'contain' && adjustedWidth > width || type === 'cover' && adjustedWidth < width) {\n height = width / aspectRatio;\n } else {\n width = height * aspectRatio;\n }\n } else if (isValidWidth) {\n height = width / aspectRatio;\n } else if (isValidHeight) {\n width = height * aspectRatio;\n }\n\n return {\n width: width,\n height: height\n };\n }\n /**\n * Get the new sizes of a rectangle after rotated.\n * @param {Object} data - The original sizes.\n * @returns {Object} The result sizes.\n */\n\n function getRotatedSizes(_ref5) {\n var width = _ref5.width,\n height = _ref5.height,\n degree = _ref5.degree;\n degree = Math.abs(degree) % 180;\n\n if (degree === 90) {\n return {\n width: height,\n height: width\n };\n }\n\n var arc = degree % 90 * Math.PI / 180;\n var sinArc = Math.sin(arc);\n var cosArc = Math.cos(arc);\n var newWidth = width * cosArc + height * sinArc;\n var newHeight = width * sinArc + height * cosArc;\n return degree > 90 ? {\n width: newHeight,\n height: newWidth\n } : {\n width: newWidth,\n height: newHeight\n };\n }\n /**\n * Get a canvas which drew the given image.\n * @param {HTMLImageElement} image - The image for drawing.\n * @param {Object} imageData - The image data.\n * @param {Object} canvasData - The canvas data.\n * @param {Object} options - The options.\n * @returns {HTMLCanvasElement} The result canvas.\n */\n\n function getSourceCanvas(image, _ref6, _ref7, _ref8) {\n var imageAspectRatio = _ref6.aspectRatio,\n imageNaturalWidth = _ref6.naturalWidth,\n imageNaturalHeight = _ref6.naturalHeight,\n _ref6$rotate = _ref6.rotate,\n rotate = _ref6$rotate === void 0 ? 0 : _ref6$rotate,\n _ref6$scaleX = _ref6.scaleX,\n scaleX = _ref6$scaleX === void 0 ? 1 : _ref6$scaleX,\n _ref6$scaleY = _ref6.scaleY,\n scaleY = _ref6$scaleY === void 0 ? 1 : _ref6$scaleY;\n var aspectRatio = _ref7.aspectRatio,\n naturalWidth = _ref7.naturalWidth,\n naturalHeight = _ref7.naturalHeight;\n var _ref8$fillColor = _ref8.fillColor,\n fillColor = _ref8$fillColor === void 0 ? 'transparent' : _ref8$fillColor,\n _ref8$imageSmoothingE = _ref8.imageSmoothingEnabled,\n imageSmoothingEnabled = _ref8$imageSmoothingE === void 0 ? true : _ref8$imageSmoothingE,\n _ref8$imageSmoothingQ = _ref8.imageSmoothingQuality,\n imageSmoothingQuality = _ref8$imageSmoothingQ === void 0 ? 'low' : _ref8$imageSmoothingQ,\n _ref8$maxWidth = _ref8.maxWidth,\n maxWidth = _ref8$maxWidth === void 0 ? Infinity : _ref8$maxWidth,\n _ref8$maxHeight = _ref8.maxHeight,\n maxHeight = _ref8$maxHeight === void 0 ? Infinity : _ref8$maxHeight,\n _ref8$minWidth = _ref8.minWidth,\n minWidth = _ref8$minWidth === void 0 ? 0 : _ref8$minWidth,\n _ref8$minHeight = _ref8.minHeight,\n minHeight = _ref8$minHeight === void 0 ? 0 : _ref8$minHeight;\n var canvas = document.createElement('canvas');\n var context = canvas.getContext('2d');\n var maxSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: maxWidth,\n height: maxHeight\n });\n var minSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: minWidth,\n height: minHeight\n }, 'cover');\n var width = Math.min(maxSizes.width, Math.max(minSizes.width, naturalWidth));\n var height = Math.min(maxSizes.height, Math.max(minSizes.height, naturalHeight)); // Note: should always use image's natural sizes for drawing as\n // imageData.naturalWidth === canvasData.naturalHeight when rotate % 180 === 90\n\n var destMaxSizes = getAdjustedSizes({\n aspectRatio: imageAspectRatio,\n width: maxWidth,\n height: maxHeight\n });\n var destMinSizes = getAdjustedSizes({\n aspectRatio: imageAspectRatio,\n width: minWidth,\n height: minHeight\n }, 'cover');\n var destWidth = Math.min(destMaxSizes.width, Math.max(destMinSizes.width, imageNaturalWidth));\n var destHeight = Math.min(destMaxSizes.height, Math.max(destMinSizes.height, imageNaturalHeight));\n var params = [-destWidth / 2, -destHeight / 2, destWidth, destHeight];\n canvas.width = normalizeDecimalNumber(width);\n canvas.height = normalizeDecimalNumber(height);\n context.fillStyle = fillColor;\n context.fillRect(0, 0, width, height);\n context.save();\n context.translate(width / 2, height / 2);\n context.rotate(rotate * Math.PI / 180);\n context.scale(scaleX, scaleY);\n context.imageSmoothingEnabled = imageSmoothingEnabled;\n context.imageSmoothingQuality = imageSmoothingQuality;\n context.drawImage.apply(context, [image].concat(_toConsumableArray(params.map(function (param) {\n return Math.floor(normalizeDecimalNumber(param));\n }))));\n context.restore();\n return canvas;\n }\n var fromCharCode = String.fromCharCode;\n /**\n * Get string from char code in data view.\n * @param {DataView} dataView - The data view for read.\n * @param {number} start - The start index.\n * @param {number} length - The read length.\n * @returns {string} The read result.\n */\n\n function getStringFromCharCode(dataView, start, length) {\n var str = '';\n length += start;\n\n for (var i = start; i < length; i += 1) {\n str += fromCharCode(dataView.getUint8(i));\n }\n\n return str;\n }\n var REGEXP_DATA_URL_HEAD = /^data:.*,/;\n /**\n * Transform Data URL to array buffer.\n * @param {string} dataURL - The Data URL to transform.\n * @returns {ArrayBuffer} The result array buffer.\n */\n\n function dataURLToArrayBuffer(dataURL) {\n var base64 = dataURL.replace(REGEXP_DATA_URL_HEAD, '');\n var binary = atob(base64);\n var arrayBuffer = new ArrayBuffer(binary.length);\n var uint8 = new Uint8Array(arrayBuffer);\n forEach(uint8, function (value, i) {\n uint8[i] = binary.charCodeAt(i);\n });\n return arrayBuffer;\n }\n /**\n * Transform array buffer to Data URL.\n * @param {ArrayBuffer} arrayBuffer - The array buffer to transform.\n * @param {string} mimeType - The mime type of the Data URL.\n * @returns {string} The result Data URL.\n */\n\n function arrayBufferToDataURL(arrayBuffer, mimeType) {\n var chunks = []; // Chunk Typed Array for better performance (#435)\n\n var chunkSize = 8192;\n var uint8 = new Uint8Array(arrayBuffer);\n\n while (uint8.length > 0) {\n // XXX: Babel's `toConsumableArray` helper will throw error in IE or Safari 9\n // eslint-disable-next-line prefer-spread\n chunks.push(fromCharCode.apply(null, toArray(uint8.subarray(0, chunkSize))));\n uint8 = uint8.subarray(chunkSize);\n }\n\n return \"data:\".concat(mimeType, \";base64,\").concat(btoa(chunks.join('')));\n }\n /**\n * Get orientation value from given array buffer.\n * @param {ArrayBuffer} arrayBuffer - The array buffer to read.\n * @returns {number} The read orientation value.\n */\n\n function resetAndGetOrientation(arrayBuffer) {\n var dataView = new DataView(arrayBuffer);\n var orientation; // Ignores range error when the image does not have correct Exif information\n\n try {\n var littleEndian;\n var app1Start;\n var ifdStart; // Only handle JPEG image (start by 0xFFD8)\n\n if (dataView.getUint8(0) === 0xFF && dataView.getUint8(1) === 0xD8) {\n var length = dataView.byteLength;\n var offset = 2;\n\n while (offset + 1 < length) {\n if (dataView.getUint8(offset) === 0xFF && dataView.getUint8(offset + 1) === 0xE1) {\n app1Start = offset;\n break;\n }\n\n offset += 1;\n }\n }\n\n if (app1Start) {\n var exifIDCode = app1Start + 4;\n var tiffOffset = app1Start + 10;\n\n if (getStringFromCharCode(dataView, exifIDCode, 4) === 'Exif') {\n var endianness = dataView.getUint16(tiffOffset);\n littleEndian = endianness === 0x4949;\n\n if (littleEndian || endianness === 0x4D4D\n /* bigEndian */\n ) {\n if (dataView.getUint16(tiffOffset + 2, littleEndian) === 0x002A) {\n var firstIFDOffset = dataView.getUint32(tiffOffset + 4, littleEndian);\n\n if (firstIFDOffset >= 0x00000008) {\n ifdStart = tiffOffset + firstIFDOffset;\n }\n }\n }\n }\n }\n\n if (ifdStart) {\n var _length = dataView.getUint16(ifdStart, littleEndian);\n\n var _offset;\n\n var i;\n\n for (i = 0; i < _length; i += 1) {\n _offset = ifdStart + i * 12 + 2;\n\n if (dataView.getUint16(_offset, littleEndian) === 0x0112\n /* Orientation */\n ) {\n // 8 is the offset of the current tag's value\n _offset += 8; // Get the original orientation value\n\n orientation = dataView.getUint16(_offset, littleEndian); // Override the orientation with its default value\n\n dataView.setUint16(_offset, 1, littleEndian);\n break;\n }\n }\n }\n } catch (error) {\n orientation = 1;\n }\n\n return orientation;\n }\n /**\n * Parse Exif Orientation value.\n * @param {number} orientation - The orientation to parse.\n * @returns {Object} The parsed result.\n */\n\n function parseOrientation(orientation) {\n var rotate = 0;\n var scaleX = 1;\n var scaleY = 1;\n\n switch (orientation) {\n // Flip horizontal\n case 2:\n scaleX = -1;\n break;\n // Rotate left 180°\n\n case 3:\n rotate = -180;\n break;\n // Flip vertical\n\n case 4:\n scaleY = -1;\n break;\n // Flip vertical and rotate right 90°\n\n case 5:\n rotate = 90;\n scaleY = -1;\n break;\n // Rotate right 90°\n\n case 6:\n rotate = 90;\n break;\n // Flip horizontal and rotate right 90°\n\n case 7:\n rotate = 90;\n scaleX = -1;\n break;\n // Rotate left 90°\n\n case 8:\n rotate = -90;\n break;\n }\n\n return {\n rotate: rotate,\n scaleX: scaleX,\n scaleY: scaleY\n };\n }\n\n var render = {\n render: function render() {\n this.initContainer();\n this.initCanvas();\n this.initCropBox();\n this.renderCanvas();\n\n if (this.cropped) {\n this.renderCropBox();\n }\n },\n initContainer: function initContainer() {\n var element = this.element,\n options = this.options,\n container = this.container,\n cropper = this.cropper;\n addClass(cropper, CLASS_HIDDEN);\n removeClass(element, CLASS_HIDDEN);\n var containerData = {\n width: Math.max(container.offsetWidth, Number(options.minContainerWidth) || 200),\n height: Math.max(container.offsetHeight, Number(options.minContainerHeight) || 100)\n };\n this.containerData = containerData;\n setStyle(cropper, {\n width: containerData.width,\n height: containerData.height\n });\n addClass(element, CLASS_HIDDEN);\n removeClass(cropper, CLASS_HIDDEN);\n },\n // Canvas (image wrapper)\n initCanvas: function initCanvas() {\n var containerData = this.containerData,\n imageData = this.imageData;\n var viewMode = this.options.viewMode;\n var rotated = Math.abs(imageData.rotate) % 180 === 90;\n var naturalWidth = rotated ? imageData.naturalHeight : imageData.naturalWidth;\n var naturalHeight = rotated ? imageData.naturalWidth : imageData.naturalHeight;\n var aspectRatio = naturalWidth / naturalHeight;\n var canvasWidth = containerData.width;\n var canvasHeight = containerData.height;\n\n if (containerData.height * aspectRatio > containerData.width) {\n if (viewMode === 3) {\n canvasWidth = containerData.height * aspectRatio;\n } else {\n canvasHeight = containerData.width / aspectRatio;\n }\n } else if (viewMode === 3) {\n canvasHeight = containerData.width / aspectRatio;\n } else {\n canvasWidth = containerData.height * aspectRatio;\n }\n\n var canvasData = {\n aspectRatio: aspectRatio,\n naturalWidth: naturalWidth,\n naturalHeight: naturalHeight,\n width: canvasWidth,\n height: canvasHeight\n };\n canvasData.left = (containerData.width - canvasWidth) / 2;\n canvasData.top = (containerData.height - canvasHeight) / 2;\n canvasData.oldLeft = canvasData.left;\n canvasData.oldTop = canvasData.top;\n this.canvasData = canvasData;\n this.limited = viewMode === 1 || viewMode === 2;\n this.limitCanvas(true, true);\n this.initialImageData = assign({}, imageData);\n this.initialCanvasData = assign({}, canvasData);\n },\n limitCanvas: function limitCanvas(sizeLimited, positionLimited) {\n var options = this.options,\n containerData = this.containerData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var viewMode = options.viewMode;\n var aspectRatio = canvasData.aspectRatio;\n var cropped = this.cropped && cropBoxData;\n\n if (sizeLimited) {\n var minCanvasWidth = Number(options.minCanvasWidth) || 0;\n var minCanvasHeight = Number(options.minCanvasHeight) || 0;\n\n if (viewMode > 1) {\n minCanvasWidth = Math.max(minCanvasWidth, containerData.width);\n minCanvasHeight = Math.max(minCanvasHeight, containerData.height);\n\n if (viewMode === 3) {\n if (minCanvasHeight * aspectRatio > minCanvasWidth) {\n minCanvasWidth = minCanvasHeight * aspectRatio;\n } else {\n minCanvasHeight = minCanvasWidth / aspectRatio;\n }\n }\n } else if (viewMode > 0) {\n if (minCanvasWidth) {\n minCanvasWidth = Math.max(minCanvasWidth, cropped ? cropBoxData.width : 0);\n } else if (minCanvasHeight) {\n minCanvasHeight = Math.max(minCanvasHeight, cropped ? cropBoxData.height : 0);\n } else if (cropped) {\n minCanvasWidth = cropBoxData.width;\n minCanvasHeight = cropBoxData.height;\n\n if (minCanvasHeight * aspectRatio > minCanvasWidth) {\n minCanvasWidth = minCanvasHeight * aspectRatio;\n } else {\n minCanvasHeight = minCanvasWidth / aspectRatio;\n }\n }\n }\n\n var _getAdjustedSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: minCanvasWidth,\n height: minCanvasHeight\n });\n\n minCanvasWidth = _getAdjustedSizes.width;\n minCanvasHeight = _getAdjustedSizes.height;\n canvasData.minWidth = minCanvasWidth;\n canvasData.minHeight = minCanvasHeight;\n canvasData.maxWidth = Infinity;\n canvasData.maxHeight = Infinity;\n }\n\n if (positionLimited) {\n if (viewMode > (cropped ? 0 : 1)) {\n var newCanvasLeft = containerData.width - canvasData.width;\n var newCanvasTop = containerData.height - canvasData.height;\n canvasData.minLeft = Math.min(0, newCanvasLeft);\n canvasData.minTop = Math.min(0, newCanvasTop);\n canvasData.maxLeft = Math.max(0, newCanvasLeft);\n canvasData.maxTop = Math.max(0, newCanvasTop);\n\n if (cropped && this.limited) {\n canvasData.minLeft = Math.min(cropBoxData.left, cropBoxData.left + (cropBoxData.width - canvasData.width));\n canvasData.minTop = Math.min(cropBoxData.top, cropBoxData.top + (cropBoxData.height - canvasData.height));\n canvasData.maxLeft = cropBoxData.left;\n canvasData.maxTop = cropBoxData.top;\n\n if (viewMode === 2) {\n if (canvasData.width >= containerData.width) {\n canvasData.minLeft = Math.min(0, newCanvasLeft);\n canvasData.maxLeft = Math.max(0, newCanvasLeft);\n }\n\n if (canvasData.height >= containerData.height) {\n canvasData.minTop = Math.min(0, newCanvasTop);\n canvasData.maxTop = Math.max(0, newCanvasTop);\n }\n }\n }\n } else {\n canvasData.minLeft = -canvasData.width;\n canvasData.minTop = -canvasData.height;\n canvasData.maxLeft = containerData.width;\n canvasData.maxTop = containerData.height;\n }\n }\n },\n renderCanvas: function renderCanvas(changed, transformed) {\n var canvasData = this.canvasData,\n imageData = this.imageData;\n\n if (transformed) {\n var _getRotatedSizes = getRotatedSizes({\n width: imageData.naturalWidth * Math.abs(imageData.scaleX || 1),\n height: imageData.naturalHeight * Math.abs(imageData.scaleY || 1),\n degree: imageData.rotate || 0\n }),\n naturalWidth = _getRotatedSizes.width,\n naturalHeight = _getRotatedSizes.height;\n\n var width = canvasData.width * (naturalWidth / canvasData.naturalWidth);\n var height = canvasData.height * (naturalHeight / canvasData.naturalHeight);\n canvasData.left -= (width - canvasData.width) / 2;\n canvasData.top -= (height - canvasData.height) / 2;\n canvasData.width = width;\n canvasData.height = height;\n canvasData.aspectRatio = naturalWidth / naturalHeight;\n canvasData.naturalWidth = naturalWidth;\n canvasData.naturalHeight = naturalHeight;\n this.limitCanvas(true, false);\n }\n\n if (canvasData.width > canvasData.maxWidth || canvasData.width < canvasData.minWidth) {\n canvasData.left = canvasData.oldLeft;\n }\n\n if (canvasData.height > canvasData.maxHeight || canvasData.height < canvasData.minHeight) {\n canvasData.top = canvasData.oldTop;\n }\n\n canvasData.width = Math.min(Math.max(canvasData.width, canvasData.minWidth), canvasData.maxWidth);\n canvasData.height = Math.min(Math.max(canvasData.height, canvasData.minHeight), canvasData.maxHeight);\n this.limitCanvas(false, true);\n canvasData.left = Math.min(Math.max(canvasData.left, canvasData.minLeft), canvasData.maxLeft);\n canvasData.top = Math.min(Math.max(canvasData.top, canvasData.minTop), canvasData.maxTop);\n canvasData.oldLeft = canvasData.left;\n canvasData.oldTop = canvasData.top;\n setStyle(this.canvas, assign({\n width: canvasData.width,\n height: canvasData.height\n }, getTransforms({\n translateX: canvasData.left,\n translateY: canvasData.top\n })));\n this.renderImage(changed);\n\n if (this.cropped && this.limited) {\n this.limitCropBox(true, true);\n }\n },\n renderImage: function renderImage(changed) {\n var canvasData = this.canvasData,\n imageData = this.imageData;\n var width = imageData.naturalWidth * (canvasData.width / canvasData.naturalWidth);\n var height = imageData.naturalHeight * (canvasData.height / canvasData.naturalHeight);\n assign(imageData, {\n width: width,\n height: height,\n left: (canvasData.width - width) / 2,\n top: (canvasData.height - height) / 2\n });\n setStyle(this.image, assign({\n width: imageData.width,\n height: imageData.height\n }, getTransforms(assign({\n translateX: imageData.left,\n translateY: imageData.top\n }, imageData))));\n\n if (changed) {\n this.output();\n }\n },\n initCropBox: function initCropBox() {\n var options = this.options,\n canvasData = this.canvasData;\n var aspectRatio = options.aspectRatio || options.initialAspectRatio;\n var autoCropArea = Number(options.autoCropArea) || 0.8;\n var cropBoxData = {\n width: canvasData.width,\n height: canvasData.height\n };\n\n if (aspectRatio) {\n if (canvasData.height * aspectRatio > canvasData.width) {\n cropBoxData.height = cropBoxData.width / aspectRatio;\n } else {\n cropBoxData.width = cropBoxData.height * aspectRatio;\n }\n }\n\n this.cropBoxData = cropBoxData;\n this.limitCropBox(true, true); // Initialize auto crop area\n\n cropBoxData.width = Math.min(Math.max(cropBoxData.width, cropBoxData.minWidth), cropBoxData.maxWidth);\n cropBoxData.height = Math.min(Math.max(cropBoxData.height, cropBoxData.minHeight), cropBoxData.maxHeight); // The width/height of auto crop area must large than \"minWidth/Height\"\n\n cropBoxData.width = Math.max(cropBoxData.minWidth, cropBoxData.width * autoCropArea);\n cropBoxData.height = Math.max(cropBoxData.minHeight, cropBoxData.height * autoCropArea);\n cropBoxData.left = canvasData.left + (canvasData.width - cropBoxData.width) / 2;\n cropBoxData.top = canvasData.top + (canvasData.height - cropBoxData.height) / 2;\n cropBoxData.oldLeft = cropBoxData.left;\n cropBoxData.oldTop = cropBoxData.top;\n this.initialCropBoxData = assign({}, cropBoxData);\n },\n limitCropBox: function limitCropBox(sizeLimited, positionLimited) {\n var options = this.options,\n containerData = this.containerData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData,\n limited = this.limited;\n var aspectRatio = options.aspectRatio;\n\n if (sizeLimited) {\n var minCropBoxWidth = Number(options.minCropBoxWidth) || 0;\n var minCropBoxHeight = Number(options.minCropBoxHeight) || 0;\n var maxCropBoxWidth = limited ? Math.min(containerData.width, canvasData.width, canvasData.width + canvasData.left, containerData.width - canvasData.left) : containerData.width;\n var maxCropBoxHeight = limited ? Math.min(containerData.height, canvasData.height, canvasData.height + canvasData.top, containerData.height - canvasData.top) : containerData.height; // The min/maxCropBoxWidth/Height must be less than container's width/height\n\n minCropBoxWidth = Math.min(minCropBoxWidth, containerData.width);\n minCropBoxHeight = Math.min(minCropBoxHeight, containerData.height);\n\n if (aspectRatio) {\n if (minCropBoxWidth && minCropBoxHeight) {\n if (minCropBoxHeight * aspectRatio > minCropBoxWidth) {\n minCropBoxHeight = minCropBoxWidth / aspectRatio;\n } else {\n minCropBoxWidth = minCropBoxHeight * aspectRatio;\n }\n } else if (minCropBoxWidth) {\n minCropBoxHeight = minCropBoxWidth / aspectRatio;\n } else if (minCropBoxHeight) {\n minCropBoxWidth = minCropBoxHeight * aspectRatio;\n }\n\n if (maxCropBoxHeight * aspectRatio > maxCropBoxWidth) {\n maxCropBoxHeight = maxCropBoxWidth / aspectRatio;\n } else {\n maxCropBoxWidth = maxCropBoxHeight * aspectRatio;\n }\n } // The minWidth/Height must be less than maxWidth/Height\n\n\n cropBoxData.minWidth = Math.min(minCropBoxWidth, maxCropBoxWidth);\n cropBoxData.minHeight = Math.min(minCropBoxHeight, maxCropBoxHeight);\n cropBoxData.maxWidth = maxCropBoxWidth;\n cropBoxData.maxHeight = maxCropBoxHeight;\n }\n\n if (positionLimited) {\n if (limited) {\n cropBoxData.minLeft = Math.max(0, canvasData.left);\n cropBoxData.minTop = Math.max(0, canvasData.top);\n cropBoxData.maxLeft = Math.min(containerData.width, canvasData.left + canvasData.width) - cropBoxData.width;\n cropBoxData.maxTop = Math.min(containerData.height, canvasData.top + canvasData.height) - cropBoxData.height;\n } else {\n cropBoxData.minLeft = 0;\n cropBoxData.minTop = 0;\n cropBoxData.maxLeft = containerData.width - cropBoxData.width;\n cropBoxData.maxTop = containerData.height - cropBoxData.height;\n }\n }\n },\n renderCropBox: function renderCropBox() {\n var options = this.options,\n containerData = this.containerData,\n cropBoxData = this.cropBoxData;\n\n if (cropBoxData.width > cropBoxData.maxWidth || cropBoxData.width < cropBoxData.minWidth) {\n cropBoxData.left = cropBoxData.oldLeft;\n }\n\n if (cropBoxData.height > cropBoxData.maxHeight || cropBoxData.height < cropBoxData.minHeight) {\n cropBoxData.top = cropBoxData.oldTop;\n }\n\n cropBoxData.width = Math.min(Math.max(cropBoxData.width, cropBoxData.minWidth), cropBoxData.maxWidth);\n cropBoxData.height = Math.min(Math.max(cropBoxData.height, cropBoxData.minHeight), cropBoxData.maxHeight);\n this.limitCropBox(false, true);\n cropBoxData.left = Math.min(Math.max(cropBoxData.left, cropBoxData.minLeft), cropBoxData.maxLeft);\n cropBoxData.top = Math.min(Math.max(cropBoxData.top, cropBoxData.minTop), cropBoxData.maxTop);\n cropBoxData.oldLeft = cropBoxData.left;\n cropBoxData.oldTop = cropBoxData.top;\n\n if (options.movable && options.cropBoxMovable) {\n // Turn to move the canvas when the crop box is equal to the container\n setData(this.face, DATA_ACTION, cropBoxData.width >= containerData.width && cropBoxData.height >= containerData.height ? ACTION_MOVE : ACTION_ALL);\n }\n\n setStyle(this.cropBox, assign({\n width: cropBoxData.width,\n height: cropBoxData.height\n }, getTransforms({\n translateX: cropBoxData.left,\n translateY: cropBoxData.top\n })));\n\n if (this.cropped && this.limited) {\n this.limitCanvas(true, true);\n }\n\n if (!this.disabled) {\n this.output();\n }\n },\n output: function output() {\n this.preview();\n dispatchEvent(this.element, EVENT_CROP, this.getData());\n }\n };\n\n var preview = {\n initPreview: function initPreview() {\n var element = this.element,\n crossOrigin = this.crossOrigin;\n var preview = this.options.preview;\n var url = crossOrigin ? this.crossOriginUrl : this.url;\n var alt = element.alt || 'The image to preview';\n var image = document.createElement('img');\n\n if (crossOrigin) {\n image.crossOrigin = crossOrigin;\n }\n\n image.src = url;\n image.alt = alt;\n this.viewBox.appendChild(image);\n this.viewBoxImage = image;\n\n if (!preview) {\n return;\n }\n\n var previews = preview;\n\n if (typeof preview === 'string') {\n previews = element.ownerDocument.querySelectorAll(preview);\n } else if (preview.querySelector) {\n previews = [preview];\n }\n\n this.previews = previews;\n forEach(previews, function (el) {\n var img = document.createElement('img'); // Save the original size for recover\n\n setData(el, DATA_PREVIEW, {\n width: el.offsetWidth,\n height: el.offsetHeight,\n html: el.innerHTML\n });\n\n if (crossOrigin) {\n img.crossOrigin = crossOrigin;\n }\n\n img.src = url;\n img.alt = alt;\n /**\n * Override img element styles\n * Add `display:block` to avoid margin top issue\n * Add `height:auto` to override `height` attribute on IE8\n * (Occur only when margin-top <= -height)\n */\n\n img.style.cssText = 'display:block;' + 'width:100%;' + 'height:auto;' + 'min-width:0!important;' + 'min-height:0!important;' + 'max-width:none!important;' + 'max-height:none!important;' + 'image-orientation:0deg!important;\"';\n el.innerHTML = '';\n el.appendChild(img);\n });\n },\n resetPreview: function resetPreview() {\n forEach(this.previews, function (element) {\n var data = getData(element, DATA_PREVIEW);\n setStyle(element, {\n width: data.width,\n height: data.height\n });\n element.innerHTML = data.html;\n removeData(element, DATA_PREVIEW);\n });\n },\n preview: function preview() {\n var imageData = this.imageData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var cropBoxWidth = cropBoxData.width,\n cropBoxHeight = cropBoxData.height;\n var width = imageData.width,\n height = imageData.height;\n var left = cropBoxData.left - canvasData.left - imageData.left;\n var top = cropBoxData.top - canvasData.top - imageData.top;\n\n if (!this.cropped || this.disabled) {\n return;\n }\n\n setStyle(this.viewBoxImage, assign({\n width: width,\n height: height\n }, getTransforms(assign({\n translateX: -left,\n translateY: -top\n }, imageData))));\n forEach(this.previews, function (element) {\n var data = getData(element, DATA_PREVIEW);\n var originalWidth = data.width;\n var originalHeight = data.height;\n var newWidth = originalWidth;\n var newHeight = originalHeight;\n var ratio = 1;\n\n if (cropBoxWidth) {\n ratio = originalWidth / cropBoxWidth;\n newHeight = cropBoxHeight * ratio;\n }\n\n if (cropBoxHeight && newHeight > originalHeight) {\n ratio = originalHeight / cropBoxHeight;\n newWidth = cropBoxWidth * ratio;\n newHeight = originalHeight;\n }\n\n setStyle(element, {\n width: newWidth,\n height: newHeight\n });\n setStyle(element.getElementsByTagName('img')[0], assign({\n width: width * ratio,\n height: height * ratio\n }, getTransforms(assign({\n translateX: -left * ratio,\n translateY: -top * ratio\n }, imageData))));\n });\n }\n };\n\n var events = {\n bind: function bind() {\n var element = this.element,\n options = this.options,\n cropper = this.cropper;\n\n if (isFunction(options.cropstart)) {\n addListener(element, EVENT_CROP_START, options.cropstart);\n }\n\n if (isFunction(options.cropmove)) {\n addListener(element, EVENT_CROP_MOVE, options.cropmove);\n }\n\n if (isFunction(options.cropend)) {\n addListener(element, EVENT_CROP_END, options.cropend);\n }\n\n if (isFunction(options.crop)) {\n addListener(element, EVENT_CROP, options.crop);\n }\n\n if (isFunction(options.zoom)) {\n addListener(element, EVENT_ZOOM, options.zoom);\n }\n\n addListener(cropper, EVENT_POINTER_DOWN, this.onCropStart = this.cropStart.bind(this));\n\n if (options.zoomable && options.zoomOnWheel) {\n addListener(cropper, EVENT_WHEEL, this.onWheel = this.wheel.bind(this), {\n passive: false,\n capture: true\n });\n }\n\n if (options.toggleDragModeOnDblclick) {\n addListener(cropper, EVENT_DBLCLICK, this.onDblclick = this.dblclick.bind(this));\n }\n\n addListener(element.ownerDocument, EVENT_POINTER_MOVE, this.onCropMove = this.cropMove.bind(this));\n addListener(element.ownerDocument, EVENT_POINTER_UP, this.onCropEnd = this.cropEnd.bind(this));\n\n if (options.responsive) {\n addListener(window, EVENT_RESIZE, this.onResize = this.resize.bind(this));\n }\n },\n unbind: function unbind() {\n var element = this.element,\n options = this.options,\n cropper = this.cropper;\n\n if (isFunction(options.cropstart)) {\n removeListener(element, EVENT_CROP_START, options.cropstart);\n }\n\n if (isFunction(options.cropmove)) {\n removeListener(element, EVENT_CROP_MOVE, options.cropmove);\n }\n\n if (isFunction(options.cropend)) {\n removeListener(element, EVENT_CROP_END, options.cropend);\n }\n\n if (isFunction(options.crop)) {\n removeListener(element, EVENT_CROP, options.crop);\n }\n\n if (isFunction(options.zoom)) {\n removeListener(element, EVENT_ZOOM, options.zoom);\n }\n\n removeListener(cropper, EVENT_POINTER_DOWN, this.onCropStart);\n\n if (options.zoomable && options.zoomOnWheel) {\n removeListener(cropper, EVENT_WHEEL, this.onWheel, {\n passive: false,\n capture: true\n });\n }\n\n if (options.toggleDragModeOnDblclick) {\n removeListener(cropper, EVENT_DBLCLICK, this.onDblclick);\n }\n\n removeListener(element.ownerDocument, EVENT_POINTER_MOVE, this.onCropMove);\n removeListener(element.ownerDocument, EVENT_POINTER_UP, this.onCropEnd);\n\n if (options.responsive) {\n removeListener(window, EVENT_RESIZE, this.onResize);\n }\n }\n };\n\n var handlers = {\n resize: function resize() {\n if (this.disabled) {\n return;\n }\n\n var options = this.options,\n container = this.container,\n containerData = this.containerData;\n var ratio = container.offsetWidth / containerData.width; // Resize when width changed or height changed\n\n if (ratio !== 1 || container.offsetHeight !== containerData.height) {\n var canvasData;\n var cropBoxData;\n\n if (options.restore) {\n canvasData = this.getCanvasData();\n cropBoxData = this.getCropBoxData();\n }\n\n this.render();\n\n if (options.restore) {\n this.setCanvasData(forEach(canvasData, function (n, i) {\n canvasData[i] = n * ratio;\n }));\n this.setCropBoxData(forEach(cropBoxData, function (n, i) {\n cropBoxData[i] = n * ratio;\n }));\n }\n }\n },\n dblclick: function dblclick() {\n if (this.disabled || this.options.dragMode === DRAG_MODE_NONE) {\n return;\n }\n\n this.setDragMode(hasClass(this.dragBox, CLASS_CROP) ? DRAG_MODE_MOVE : DRAG_MODE_CROP);\n },\n wheel: function wheel(event) {\n var _this = this;\n\n var ratio = Number(this.options.wheelZoomRatio) || 0.1;\n var delta = 1;\n\n if (this.disabled) {\n return;\n }\n\n event.preventDefault(); // Limit wheel speed to prevent zoom too fast (#21)\n\n if (this.wheeling) {\n return;\n }\n\n this.wheeling = true;\n setTimeout(function () {\n _this.wheeling = false;\n }, 50);\n\n if (event.deltaY) {\n delta = event.deltaY > 0 ? 1 : -1;\n } else if (event.wheelDelta) {\n delta = -event.wheelDelta / 120;\n } else if (event.detail) {\n delta = event.detail > 0 ? 1 : -1;\n }\n\n this.zoom(-delta * ratio, event);\n },\n cropStart: function cropStart(event) {\n var buttons = event.buttons,\n button = event.button;\n\n if (this.disabled // Handle mouse event and pointer event and ignore touch event\n || (event.type === 'mousedown' || event.type === 'pointerdown' && event.pointerType === 'mouse') && ( // No primary button (Usually the left button)\n isNumber(buttons) && buttons !== 1 || isNumber(button) && button !== 0 // Open context menu\n || event.ctrlKey)) {\n return;\n }\n\n var options = this.options,\n pointers = this.pointers;\n var action;\n\n if (event.changedTouches) {\n // Handle touch event\n forEach(event.changedTouches, function (touch) {\n pointers[touch.identifier] = getPointer(touch);\n });\n } else {\n // Handle mouse event and pointer event\n pointers[event.pointerId || 0] = getPointer(event);\n }\n\n if (Object.keys(pointers).length > 1 && options.zoomable && options.zoomOnTouch) {\n action = ACTION_ZOOM;\n } else {\n action = getData(event.target, DATA_ACTION);\n }\n\n if (!REGEXP_ACTIONS.test(action)) {\n return;\n }\n\n if (dispatchEvent(this.element, EVENT_CROP_START, {\n originalEvent: event,\n action: action\n }) === false) {\n return;\n } // This line is required for preventing page zooming in iOS browsers\n\n\n event.preventDefault();\n this.action = action;\n this.cropping = false;\n\n if (action === ACTION_CROP) {\n this.cropping = true;\n addClass(this.dragBox, CLASS_MODAL);\n }\n },\n cropMove: function cropMove(event) {\n var action = this.action;\n\n if (this.disabled || !action) {\n return;\n }\n\n var pointers = this.pointers;\n event.preventDefault();\n\n if (dispatchEvent(this.element, EVENT_CROP_MOVE, {\n originalEvent: event,\n action: action\n }) === false) {\n return;\n }\n\n if (event.changedTouches) {\n forEach(event.changedTouches, function (touch) {\n // The first parameter should not be undefined (#432)\n assign(pointers[touch.identifier] || {}, getPointer(touch, true));\n });\n } else {\n assign(pointers[event.pointerId || 0] || {}, getPointer(event, true));\n }\n\n this.change(event);\n },\n cropEnd: function cropEnd(event) {\n if (this.disabled) {\n return;\n }\n\n var action = this.action,\n pointers = this.pointers;\n\n if (event.changedTouches) {\n forEach(event.changedTouches, function (touch) {\n delete pointers[touch.identifier];\n });\n } else {\n delete pointers[event.pointerId || 0];\n }\n\n if (!action) {\n return;\n }\n\n event.preventDefault();\n\n if (!Object.keys(pointers).length) {\n this.action = '';\n }\n\n if (this.cropping) {\n this.cropping = false;\n toggleClass(this.dragBox, CLASS_MODAL, this.cropped && this.options.modal);\n }\n\n dispatchEvent(this.element, EVENT_CROP_END, {\n originalEvent: event,\n action: action\n });\n }\n };\n\n var change = {\n change: function change(event) {\n var options = this.options,\n canvasData = this.canvasData,\n containerData = this.containerData,\n cropBoxData = this.cropBoxData,\n pointers = this.pointers;\n var action = this.action;\n var aspectRatio = options.aspectRatio;\n var left = cropBoxData.left,\n top = cropBoxData.top,\n width = cropBoxData.width,\n height = cropBoxData.height;\n var right = left + width;\n var bottom = top + height;\n var minLeft = 0;\n var minTop = 0;\n var maxWidth = containerData.width;\n var maxHeight = containerData.height;\n var renderable = true;\n var offset; // Locking aspect ratio in \"free mode\" by holding shift key\n\n if (!aspectRatio && event.shiftKey) {\n aspectRatio = width && height ? width / height : 1;\n }\n\n if (this.limited) {\n minLeft = cropBoxData.minLeft;\n minTop = cropBoxData.minTop;\n maxWidth = minLeft + Math.min(containerData.width, canvasData.width, canvasData.left + canvasData.width);\n maxHeight = minTop + Math.min(containerData.height, canvasData.height, canvasData.top + canvasData.height);\n }\n\n var pointer = pointers[Object.keys(pointers)[0]];\n var range = {\n x: pointer.endX - pointer.startX,\n y: pointer.endY - pointer.startY\n };\n\n var check = function check(side) {\n switch (side) {\n case ACTION_EAST:\n if (right + range.x > maxWidth) {\n range.x = maxWidth - right;\n }\n\n break;\n\n case ACTION_WEST:\n if (left + range.x < minLeft) {\n range.x = minLeft - left;\n }\n\n break;\n\n case ACTION_NORTH:\n if (top + range.y < minTop) {\n range.y = minTop - top;\n }\n\n break;\n\n case ACTION_SOUTH:\n if (bottom + range.y > maxHeight) {\n range.y = maxHeight - bottom;\n }\n\n break;\n }\n };\n\n switch (action) {\n // Move crop box\n case ACTION_ALL:\n left += range.x;\n top += range.y;\n break;\n // Resize crop box\n\n case ACTION_EAST:\n if (range.x >= 0 && (right >= maxWidth || aspectRatio && (top <= minTop || bottom >= maxHeight))) {\n renderable = false;\n break;\n }\n\n check(ACTION_EAST);\n width += range.x;\n\n if (width < 0) {\n action = ACTION_WEST;\n width = -width;\n left -= width;\n }\n\n if (aspectRatio) {\n height = width / aspectRatio;\n top += (cropBoxData.height - height) / 2;\n }\n\n break;\n\n case ACTION_NORTH:\n if (range.y <= 0 && (top <= minTop || aspectRatio && (left <= minLeft || right >= maxWidth))) {\n renderable = false;\n break;\n }\n\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n\n if (height < 0) {\n action = ACTION_SOUTH;\n height = -height;\n top -= height;\n }\n\n if (aspectRatio) {\n width = height * aspectRatio;\n left += (cropBoxData.width - width) / 2;\n }\n\n break;\n\n case ACTION_WEST:\n if (range.x <= 0 && (left <= minLeft || aspectRatio && (top <= minTop || bottom >= maxHeight))) {\n renderable = false;\n break;\n }\n\n check(ACTION_WEST);\n width -= range.x;\n left += range.x;\n\n if (width < 0) {\n action = ACTION_EAST;\n width = -width;\n left -= width;\n }\n\n if (aspectRatio) {\n height = width / aspectRatio;\n top += (cropBoxData.height - height) / 2;\n }\n\n break;\n\n case ACTION_SOUTH:\n if (range.y >= 0 && (bottom >= maxHeight || aspectRatio && (left <= minLeft || right >= maxWidth))) {\n renderable = false;\n break;\n }\n\n check(ACTION_SOUTH);\n height += range.y;\n\n if (height < 0) {\n action = ACTION_NORTH;\n height = -height;\n top -= height;\n }\n\n if (aspectRatio) {\n width = height * aspectRatio;\n left += (cropBoxData.width - width) / 2;\n }\n\n break;\n\n case ACTION_NORTH_EAST:\n if (aspectRatio) {\n if (range.y <= 0 && (top <= minTop || right >= maxWidth)) {\n renderable = false;\n break;\n }\n\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n width = height * aspectRatio;\n } else {\n check(ACTION_NORTH);\n check(ACTION_EAST);\n\n if (range.x >= 0) {\n if (right < maxWidth) {\n width += range.x;\n } else if (range.y <= 0 && top <= minTop) {\n renderable = false;\n }\n } else {\n width += range.x;\n }\n\n if (range.y <= 0) {\n if (top > minTop) {\n height -= range.y;\n top += range.y;\n }\n } else {\n height -= range.y;\n top += range.y;\n }\n }\n\n if (width < 0 && height < 0) {\n action = ACTION_SOUTH_WEST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_NORTH_WEST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_SOUTH_EAST;\n height = -height;\n top -= height;\n }\n\n break;\n\n case ACTION_NORTH_WEST:\n if (aspectRatio) {\n if (range.y <= 0 && (top <= minTop || left <= minLeft)) {\n renderable = false;\n break;\n }\n\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n width = height * aspectRatio;\n left += cropBoxData.width - width;\n } else {\n check(ACTION_NORTH);\n check(ACTION_WEST);\n\n if (range.x <= 0) {\n if (left > minLeft) {\n width -= range.x;\n left += range.x;\n } else if (range.y <= 0 && top <= minTop) {\n renderable = false;\n }\n } else {\n width -= range.x;\n left += range.x;\n }\n\n if (range.y <= 0) {\n if (top > minTop) {\n height -= range.y;\n top += range.y;\n }\n } else {\n height -= range.y;\n top += range.y;\n }\n }\n\n if (width < 0 && height < 0) {\n action = ACTION_SOUTH_EAST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_NORTH_EAST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_SOUTH_WEST;\n height = -height;\n top -= height;\n }\n\n break;\n\n case ACTION_SOUTH_WEST:\n if (aspectRatio) {\n if (range.x <= 0 && (left <= minLeft || bottom >= maxHeight)) {\n renderable = false;\n break;\n }\n\n check(ACTION_WEST);\n width -= range.x;\n left += range.x;\n height = width / aspectRatio;\n } else {\n check(ACTION_SOUTH);\n check(ACTION_WEST);\n\n if (range.x <= 0) {\n if (left > minLeft) {\n width -= range.x;\n left += range.x;\n } else if (range.y >= 0 && bottom >= maxHeight) {\n renderable = false;\n }\n } else {\n width -= range.x;\n left += range.x;\n }\n\n if (range.y >= 0) {\n if (bottom < maxHeight) {\n height += range.y;\n }\n } else {\n height += range.y;\n }\n }\n\n if (width < 0 && height < 0) {\n action = ACTION_NORTH_EAST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_SOUTH_EAST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_NORTH_WEST;\n height = -height;\n top -= height;\n }\n\n break;\n\n case ACTION_SOUTH_EAST:\n if (aspectRatio) {\n if (range.x >= 0 && (right >= maxWidth || bottom >= maxHeight)) {\n renderable = false;\n break;\n }\n\n check(ACTION_EAST);\n width += range.x;\n height = width / aspectRatio;\n } else {\n check(ACTION_SOUTH);\n check(ACTION_EAST);\n\n if (range.x >= 0) {\n if (right < maxWidth) {\n width += range.x;\n } else if (range.y >= 0 && bottom >= maxHeight) {\n renderable = false;\n }\n } else {\n width += range.x;\n }\n\n if (range.y >= 0) {\n if (bottom < maxHeight) {\n height += range.y;\n }\n } else {\n height += range.y;\n }\n }\n\n if (width < 0 && height < 0) {\n action = ACTION_NORTH_WEST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_SOUTH_WEST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_NORTH_EAST;\n height = -height;\n top -= height;\n }\n\n break;\n // Move canvas\n\n case ACTION_MOVE:\n this.move(range.x, range.y);\n renderable = false;\n break;\n // Zoom canvas\n\n case ACTION_ZOOM:\n this.zoom(getMaxZoomRatio(pointers), event);\n renderable = false;\n break;\n // Create crop box\n\n case ACTION_CROP:\n if (!range.x || !range.y) {\n renderable = false;\n break;\n }\n\n offset = getOffset(this.cropper);\n left = pointer.startX - offset.left;\n top = pointer.startY - offset.top;\n width = cropBoxData.minWidth;\n height = cropBoxData.minHeight;\n\n if (range.x > 0) {\n action = range.y > 0 ? ACTION_SOUTH_EAST : ACTION_NORTH_EAST;\n } else if (range.x < 0) {\n left -= width;\n action = range.y > 0 ? ACTION_SOUTH_WEST : ACTION_NORTH_WEST;\n }\n\n if (range.y < 0) {\n top -= height;\n } // Show the crop box if is hidden\n\n\n if (!this.cropped) {\n removeClass(this.cropBox, CLASS_HIDDEN);\n this.cropped = true;\n\n if (this.limited) {\n this.limitCropBox(true, true);\n }\n }\n\n break;\n }\n\n if (renderable) {\n cropBoxData.width = width;\n cropBoxData.height = height;\n cropBoxData.left = left;\n cropBoxData.top = top;\n this.action = action;\n this.renderCropBox();\n } // Override\n\n\n forEach(pointers, function (p) {\n p.startX = p.endX;\n p.startY = p.endY;\n });\n }\n };\n\n var methods = {\n // Show the crop box manually\n crop: function crop() {\n if (this.ready && !this.cropped && !this.disabled) {\n this.cropped = true;\n this.limitCropBox(true, true);\n\n if (this.options.modal) {\n addClass(this.dragBox, CLASS_MODAL);\n }\n\n removeClass(this.cropBox, CLASS_HIDDEN);\n this.setCropBoxData(this.initialCropBoxData);\n }\n\n return this;\n },\n // Reset the image and crop box to their initial states\n reset: function reset() {\n if (this.ready && !this.disabled) {\n this.imageData = assign({}, this.initialImageData);\n this.canvasData = assign({}, this.initialCanvasData);\n this.cropBoxData = assign({}, this.initialCropBoxData);\n this.renderCanvas();\n\n if (this.cropped) {\n this.renderCropBox();\n }\n }\n\n return this;\n },\n // Clear the crop box\n clear: function clear() {\n if (this.cropped && !this.disabled) {\n assign(this.cropBoxData, {\n left: 0,\n top: 0,\n width: 0,\n height: 0\n });\n this.cropped = false;\n this.renderCropBox();\n this.limitCanvas(true, true); // Render canvas after crop box rendered\n\n this.renderCanvas();\n removeClass(this.dragBox, CLASS_MODAL);\n addClass(this.cropBox, CLASS_HIDDEN);\n }\n\n return this;\n },\n\n /**\n * Replace the image's src and rebuild the cropper\n * @param {string} url - The new URL.\n * @param {boolean} [hasSameSize] - Indicate if the new image has the same size as the old one.\n * @returns {Cropper} this\n */\n replace: function replace(url) {\n var hasSameSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;\n\n if (!this.disabled && url) {\n if (this.isImg) {\n this.element.src = url;\n }\n\n if (hasSameSize) {\n this.url = url;\n this.image.src = url;\n\n if (this.ready) {\n this.viewBoxImage.src = url;\n forEach(this.previews, function (element) {\n element.getElementsByTagName('img')[0].src = url;\n });\n }\n } else {\n if (this.isImg) {\n this.replaced = true;\n }\n\n this.options.data = null;\n this.uncreate();\n this.load(url);\n }\n }\n\n return this;\n },\n // Enable (unfreeze) the cropper\n enable: function enable() {\n if (this.ready && this.disabled) {\n this.disabled = false;\n removeClass(this.cropper, CLASS_DISABLED);\n }\n\n return this;\n },\n // Disable (freeze) the cropper\n disable: function disable() {\n if (this.ready && !this.disabled) {\n this.disabled = true;\n addClass(this.cropper, CLASS_DISABLED);\n }\n\n return this;\n },\n\n /**\n * Destroy the cropper and remove the instance from the image\n * @returns {Cropper} this\n */\n destroy: function destroy() {\n var element = this.element;\n\n if (!element[NAMESPACE]) {\n return this;\n }\n\n element[NAMESPACE] = undefined;\n\n if (this.isImg && this.replaced) {\n element.src = this.originalUrl;\n }\n\n this.uncreate();\n return this;\n },\n\n /**\n * Move the canvas with relative offsets\n * @param {number} offsetX - The relative offset distance on the x-axis.\n * @param {number} [offsetY=offsetX] - The relative offset distance on the y-axis.\n * @returns {Cropper} this\n */\n move: function move(offsetX) {\n var offsetY = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : offsetX;\n var _this$canvasData = this.canvasData,\n left = _this$canvasData.left,\n top = _this$canvasData.top;\n return this.moveTo(isUndefined(offsetX) ? offsetX : left + Number(offsetX), isUndefined(offsetY) ? offsetY : top + Number(offsetY));\n },\n\n /**\n * Move the canvas to an absolute point\n * @param {number} x - The x-axis coordinate.\n * @param {number} [y=x] - The y-axis coordinate.\n * @returns {Cropper} this\n */\n moveTo: function moveTo(x) {\n var y = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : x;\n var canvasData = this.canvasData;\n var changed = false;\n x = Number(x);\n y = Number(y);\n\n if (this.ready && !this.disabled && this.options.movable) {\n if (isNumber(x)) {\n canvasData.left = x;\n changed = true;\n }\n\n if (isNumber(y)) {\n canvasData.top = y;\n changed = true;\n }\n\n if (changed) {\n this.renderCanvas(true);\n }\n }\n\n return this;\n },\n\n /**\n * Zoom the canvas with a relative ratio\n * @param {number} ratio - The target ratio.\n * @param {Event} _originalEvent - The original event if any.\n * @returns {Cropper} this\n */\n zoom: function zoom(ratio, _originalEvent) {\n var canvasData = this.canvasData;\n ratio = Number(ratio);\n\n if (ratio < 0) {\n ratio = 1 / (1 - ratio);\n } else {\n ratio = 1 + ratio;\n }\n\n return this.zoomTo(canvasData.width * ratio / canvasData.naturalWidth, null, _originalEvent);\n },\n\n /**\n * Zoom the canvas to an absolute ratio\n * @param {number} ratio - The target ratio.\n * @param {Object} pivot - The zoom pivot point coordinate.\n * @param {Event} _originalEvent - The original event if any.\n * @returns {Cropper} this\n */\n zoomTo: function zoomTo(ratio, pivot, _originalEvent) {\n var options = this.options,\n canvasData = this.canvasData;\n var width = canvasData.width,\n height = canvasData.height,\n naturalWidth = canvasData.naturalWidth,\n naturalHeight = canvasData.naturalHeight;\n ratio = Number(ratio);\n\n if (ratio >= 0 && this.ready && !this.disabled && options.zoomable) {\n var newWidth = naturalWidth * ratio;\n var newHeight = naturalHeight * ratio;\n\n if (dispatchEvent(this.element, EVENT_ZOOM, {\n ratio: ratio,\n oldRatio: width / naturalWidth,\n originalEvent: _originalEvent\n }) === false) {\n return this;\n }\n\n if (_originalEvent) {\n var pointers = this.pointers;\n var offset = getOffset(this.cropper);\n var center = pointers && Object.keys(pointers).length ? getPointersCenter(pointers) : {\n pageX: _originalEvent.pageX,\n pageY: _originalEvent.pageY\n }; // Zoom from the triggering point of the event\n\n canvasData.left -= (newWidth - width) * ((center.pageX - offset.left - canvasData.left) / width);\n canvasData.top -= (newHeight - height) * ((center.pageY - offset.top - canvasData.top) / height);\n } else if (isPlainObject(pivot) && isNumber(pivot.x) && isNumber(pivot.y)) {\n canvasData.left -= (newWidth - width) * ((pivot.x - canvasData.left) / width);\n canvasData.top -= (newHeight - height) * ((pivot.y - canvasData.top) / height);\n } else {\n // Zoom from the center of the canvas\n canvasData.left -= (newWidth - width) / 2;\n canvasData.top -= (newHeight - height) / 2;\n }\n\n canvasData.width = newWidth;\n canvasData.height = newHeight;\n this.renderCanvas(true);\n }\n\n return this;\n },\n\n /**\n * Rotate the canvas with a relative degree\n * @param {number} degree - The rotate degree.\n * @returns {Cropper} this\n */\n rotate: function rotate(degree) {\n return this.rotateTo((this.imageData.rotate || 0) + Number(degree));\n },\n\n /**\n * Rotate the canvas to an absolute degree\n * @param {number} degree - The rotate degree.\n * @returns {Cropper} this\n */\n rotateTo: function rotateTo(degree) {\n degree = Number(degree);\n\n if (isNumber(degree) && this.ready && !this.disabled && this.options.rotatable) {\n this.imageData.rotate = degree % 360;\n this.renderCanvas(true, true);\n }\n\n return this;\n },\n\n /**\n * Scale the image on the x-axis.\n * @param {number} scaleX - The scale ratio on the x-axis.\n * @returns {Cropper} this\n */\n scaleX: function scaleX(_scaleX) {\n var scaleY = this.imageData.scaleY;\n return this.scale(_scaleX, isNumber(scaleY) ? scaleY : 1);\n },\n\n /**\n * Scale the image on the y-axis.\n * @param {number} scaleY - The scale ratio on the y-axis.\n * @returns {Cropper} this\n */\n scaleY: function scaleY(_scaleY) {\n var scaleX = this.imageData.scaleX;\n return this.scale(isNumber(scaleX) ? scaleX : 1, _scaleY);\n },\n\n /**\n * Scale the image\n * @param {number} scaleX - The scale ratio on the x-axis.\n * @param {number} [scaleY=scaleX] - The scale ratio on the y-axis.\n * @returns {Cropper} this\n */\n scale: function scale(scaleX) {\n var scaleY = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : scaleX;\n var imageData = this.imageData;\n var transformed = false;\n scaleX = Number(scaleX);\n scaleY = Number(scaleY);\n\n if (this.ready && !this.disabled && this.options.scalable) {\n if (isNumber(scaleX)) {\n imageData.scaleX = scaleX;\n transformed = true;\n }\n\n if (isNumber(scaleY)) {\n imageData.scaleY = scaleY;\n transformed = true;\n }\n\n if (transformed) {\n this.renderCanvas(true, true);\n }\n }\n\n return this;\n },\n\n /**\n * Get the cropped area position and size data (base on the original image)\n * @param {boolean} [rounded=false] - Indicate if round the data values or not.\n * @returns {Object} The result cropped data.\n */\n getData: function getData() {\n var rounded = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;\n var options = this.options,\n imageData = this.imageData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var data;\n\n if (this.ready && this.cropped) {\n data = {\n x: cropBoxData.left - canvasData.left,\n y: cropBoxData.top - canvasData.top,\n width: cropBoxData.width,\n height: cropBoxData.height\n };\n var ratio = imageData.width / imageData.naturalWidth;\n forEach(data, function (n, i) {\n data[i] = n / ratio;\n });\n\n if (rounded) {\n // In case rounding off leads to extra 1px in right or bottom border\n // we should round the top-left corner and the dimension (#343).\n var bottom = Math.round(data.y + data.height);\n var right = Math.round(data.x + data.width);\n data.x = Math.round(data.x);\n data.y = Math.round(data.y);\n data.width = right - data.x;\n data.height = bottom - data.y;\n }\n } else {\n data = {\n x: 0,\n y: 0,\n width: 0,\n height: 0\n };\n }\n\n if (options.rotatable) {\n data.rotate = imageData.rotate || 0;\n }\n\n if (options.scalable) {\n data.scaleX = imageData.scaleX || 1;\n data.scaleY = imageData.scaleY || 1;\n }\n\n return data;\n },\n\n /**\n * Set the cropped area position and size with new data\n * @param {Object} data - The new data.\n * @returns {Cropper} this\n */\n setData: function setData(data) {\n var options = this.options,\n imageData = this.imageData,\n canvasData = this.canvasData;\n var cropBoxData = {};\n\n if (this.ready && !this.disabled && isPlainObject(data)) {\n var transformed = false;\n\n if (options.rotatable) {\n if (isNumber(data.rotate) && data.rotate !== imageData.rotate) {\n imageData.rotate = data.rotate;\n transformed = true;\n }\n }\n\n if (options.scalable) {\n if (isNumber(data.scaleX) && data.scaleX !== imageData.scaleX) {\n imageData.scaleX = data.scaleX;\n transformed = true;\n }\n\n if (isNumber(data.scaleY) && data.scaleY !== imageData.scaleY) {\n imageData.scaleY = data.scaleY;\n transformed = true;\n }\n }\n\n if (transformed) {\n this.renderCanvas(true, true);\n }\n\n var ratio = imageData.width / imageData.naturalWidth;\n\n if (isNumber(data.x)) {\n cropBoxData.left = data.x * ratio + canvasData.left;\n }\n\n if (isNumber(data.y)) {\n cropBoxData.top = data.y * ratio + canvasData.top;\n }\n\n if (isNumber(data.width)) {\n cropBoxData.width = data.width * ratio;\n }\n\n if (isNumber(data.height)) {\n cropBoxData.height = data.height * ratio;\n }\n\n this.setCropBoxData(cropBoxData);\n }\n\n return this;\n },\n\n /**\n * Get the container size data.\n * @returns {Object} The result container data.\n */\n getContainerData: function getContainerData() {\n return this.ready ? assign({}, this.containerData) : {};\n },\n\n /**\n * Get the image position and size data.\n * @returns {Object} The result image data.\n */\n getImageData: function getImageData() {\n return this.sized ? assign({}, this.imageData) : {};\n },\n\n /**\n * Get the canvas position and size data.\n * @returns {Object} The result canvas data.\n */\n getCanvasData: function getCanvasData() {\n var canvasData = this.canvasData;\n var data = {};\n\n if (this.ready) {\n forEach(['left', 'top', 'width', 'height', 'naturalWidth', 'naturalHeight'], function (n) {\n data[n] = canvasData[n];\n });\n }\n\n return data;\n },\n\n /**\n * Set the canvas position and size with new data.\n * @param {Object} data - The new canvas data.\n * @returns {Cropper} this\n */\n setCanvasData: function setCanvasData(data) {\n var canvasData = this.canvasData;\n var aspectRatio = canvasData.aspectRatio;\n\n if (this.ready && !this.disabled && isPlainObject(data)) {\n if (isNumber(data.left)) {\n canvasData.left = data.left;\n }\n\n if (isNumber(data.top)) {\n canvasData.top = data.top;\n }\n\n if (isNumber(data.width)) {\n canvasData.width = data.width;\n canvasData.height = data.width / aspectRatio;\n } else if (isNumber(data.height)) {\n canvasData.height = data.height;\n canvasData.width = data.height * aspectRatio;\n }\n\n this.renderCanvas(true);\n }\n\n return this;\n },\n\n /**\n * Get the crop box position and size data.\n * @returns {Object} The result crop box data.\n */\n getCropBoxData: function getCropBoxData() {\n var cropBoxData = this.cropBoxData;\n var data;\n\n if (this.ready && this.cropped) {\n data = {\n left: cropBoxData.left,\n top: cropBoxData.top,\n width: cropBoxData.width,\n height: cropBoxData.height\n };\n }\n\n return data || {};\n },\n\n /**\n * Set the crop box position and size with new data.\n * @param {Object} data - The new crop box data.\n * @returns {Cropper} this\n */\n setCropBoxData: function setCropBoxData(data) {\n var cropBoxData = this.cropBoxData;\n var aspectRatio = this.options.aspectRatio;\n var widthChanged;\n var heightChanged;\n\n if (this.ready && this.cropped && !this.disabled && isPlainObject(data)) {\n if (isNumber(data.left)) {\n cropBoxData.left = data.left;\n }\n\n if (isNumber(data.top)) {\n cropBoxData.top = data.top;\n }\n\n if (isNumber(data.width) && data.width !== cropBoxData.width) {\n widthChanged = true;\n cropBoxData.width = data.width;\n }\n\n if (isNumber(data.height) && data.height !== cropBoxData.height) {\n heightChanged = true;\n cropBoxData.height = data.height;\n }\n\n if (aspectRatio) {\n if (widthChanged) {\n cropBoxData.height = cropBoxData.width / aspectRatio;\n } else if (heightChanged) {\n cropBoxData.width = cropBoxData.height * aspectRatio;\n }\n }\n\n this.renderCropBox();\n }\n\n return this;\n },\n\n /**\n * Get a canvas drawn the cropped image.\n * @param {Object} [options={}] - The config options.\n * @returns {HTMLCanvasElement} - The result canvas.\n */\n getCroppedCanvas: function getCroppedCanvas() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n\n if (!this.ready || !window.HTMLCanvasElement) {\n return null;\n }\n\n var canvasData = this.canvasData;\n var source = getSourceCanvas(this.image, this.imageData, canvasData, options); // Returns the source canvas if it is not cropped.\n\n if (!this.cropped) {\n return source;\n }\n\n var _this$getData = this.getData(),\n initialX = _this$getData.x,\n initialY = _this$getData.y,\n initialWidth = _this$getData.width,\n initialHeight = _this$getData.height;\n\n var ratio = source.width / Math.floor(canvasData.naturalWidth);\n\n if (ratio !== 1) {\n initialX *= ratio;\n initialY *= ratio;\n initialWidth *= ratio;\n initialHeight *= ratio;\n }\n\n var aspectRatio = initialWidth / initialHeight;\n var maxSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.maxWidth || Infinity,\n height: options.maxHeight || Infinity\n });\n var minSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.minWidth || 0,\n height: options.minHeight || 0\n }, 'cover');\n\n var _getAdjustedSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.width || (ratio !== 1 ? source.width : initialWidth),\n height: options.height || (ratio !== 1 ? source.height : initialHeight)\n }),\n width = _getAdjustedSizes.width,\n height = _getAdjustedSizes.height;\n\n width = Math.min(maxSizes.width, Math.max(minSizes.width, width));\n height = Math.min(maxSizes.height, Math.max(minSizes.height, height));\n var canvas = document.createElement('canvas');\n var context = canvas.getContext('2d');\n canvas.width = normalizeDecimalNumber(width);\n canvas.height = normalizeDecimalNumber(height);\n context.fillStyle = options.fillColor || 'transparent';\n context.fillRect(0, 0, width, height);\n var _options$imageSmoothi = options.imageSmoothingEnabled,\n imageSmoothingEnabled = _options$imageSmoothi === void 0 ? true : _options$imageSmoothi,\n imageSmoothingQuality = options.imageSmoothingQuality;\n context.imageSmoothingEnabled = imageSmoothingEnabled;\n\n if (imageSmoothingQuality) {\n context.imageSmoothingQuality = imageSmoothingQuality;\n } // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D.drawImage\n\n\n var sourceWidth = source.width;\n var sourceHeight = source.height; // Source canvas parameters\n\n var srcX = initialX;\n var srcY = initialY;\n var srcWidth;\n var srcHeight; // Destination canvas parameters\n\n var dstX;\n var dstY;\n var dstWidth;\n var dstHeight;\n\n if (srcX <= -initialWidth || srcX > sourceWidth) {\n srcX = 0;\n srcWidth = 0;\n dstX = 0;\n dstWidth = 0;\n } else if (srcX <= 0) {\n dstX = -srcX;\n srcX = 0;\n srcWidth = Math.min(sourceWidth, initialWidth + srcX);\n dstWidth = srcWidth;\n } else if (srcX <= sourceWidth) {\n dstX = 0;\n srcWidth = Math.min(initialWidth, sourceWidth - srcX);\n dstWidth = srcWidth;\n }\n\n if (srcWidth <= 0 || srcY <= -initialHeight || srcY > sourceHeight) {\n srcY = 0;\n srcHeight = 0;\n dstY = 0;\n dstHeight = 0;\n } else if (srcY <= 0) {\n dstY = -srcY;\n srcY = 0;\n srcHeight = Math.min(sourceHeight, initialHeight + srcY);\n dstHeight = srcHeight;\n } else if (srcY <= sourceHeight) {\n dstY = 0;\n srcHeight = Math.min(initialHeight, sourceHeight - srcY);\n dstHeight = srcHeight;\n }\n\n var params = [srcX, srcY, srcWidth, srcHeight]; // Avoid \"IndexSizeError\"\n\n if (dstWidth > 0 && dstHeight > 0) {\n var scale = width / initialWidth;\n params.push(dstX * scale, dstY * scale, dstWidth * scale, dstHeight * scale);\n } // All the numerical parameters should be integer for `drawImage`\n // https://github.com/fengyuanchen/cropper/issues/476\n\n\n context.drawImage.apply(context, [source].concat(_toConsumableArray(params.map(function (param) {\n return Math.floor(normalizeDecimalNumber(param));\n }))));\n return canvas;\n },\n\n /**\n * Change the aspect ratio of the crop box.\n * @param {number} aspectRatio - The new aspect ratio.\n * @returns {Cropper} this\n */\n setAspectRatio: function setAspectRatio(aspectRatio) {\n var options = this.options;\n\n if (!this.disabled && !isUndefined(aspectRatio)) {\n // 0 -> NaN\n options.aspectRatio = Math.max(0, aspectRatio) || NaN;\n\n if (this.ready) {\n this.initCropBox();\n\n if (this.cropped) {\n this.renderCropBox();\n }\n }\n }\n\n return this;\n },\n\n /**\n * Change the drag mode.\n * @param {string} mode - The new drag mode.\n * @returns {Cropper} this\n */\n setDragMode: function setDragMode(mode) {\n var options = this.options,\n dragBox = this.dragBox,\n face = this.face;\n\n if (this.ready && !this.disabled) {\n var croppable = mode === DRAG_MODE_CROP;\n var movable = options.movable && mode === DRAG_MODE_MOVE;\n mode = croppable || movable ? mode : DRAG_MODE_NONE;\n options.dragMode = mode;\n setData(dragBox, DATA_ACTION, mode);\n toggleClass(dragBox, CLASS_CROP, croppable);\n toggleClass(dragBox, CLASS_MOVE, movable);\n\n if (!options.cropBoxMovable) {\n // Sync drag mode to crop box when it is not movable\n setData(face, DATA_ACTION, mode);\n toggleClass(face, CLASS_CROP, croppable);\n toggleClass(face, CLASS_MOVE, movable);\n }\n }\n\n return this;\n }\n };\n\n var AnotherCropper = WINDOW.Cropper;\n\n var Cropper = /*#__PURE__*/function () {\n /**\n * Create a new Cropper.\n * @param {Element} element - The target element for cropping.\n * @param {Object} [options={}] - The configuration options.\n */\n function Cropper(element) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};\n\n _classCallCheck(this, Cropper);\n\n if (!element || !REGEXP_TAG_NAME.test(element.tagName)) {\n throw new Error('The first argument is required and must be an or element.');\n }\n\n this.element = element;\n this.options = assign({}, DEFAULTS, isPlainObject(options) && options);\n this.cropped = false;\n this.disabled = false;\n this.pointers = {};\n this.ready = false;\n this.reloading = false;\n this.replaced = false;\n this.sized = false;\n this.sizing = false;\n this.init();\n }\n\n _createClass(Cropper, [{\n key: \"init\",\n value: function init() {\n var element = this.element;\n var tagName = element.tagName.toLowerCase();\n var url;\n\n if (element[NAMESPACE]) {\n return;\n }\n\n element[NAMESPACE] = this;\n\n if (tagName === 'img') {\n this.isImg = true; // e.g.: \"img/picture.jpg\"\n\n url = element.getAttribute('src') || '';\n this.originalUrl = url; // Stop when it's a blank image\n\n if (!url) {\n return;\n } // e.g.: \"https://example.com/img/picture.jpg\"\n\n\n url = element.src;\n } else if (tagName === 'canvas' && window.HTMLCanvasElement) {\n url = element.toDataURL();\n }\n\n this.load(url);\n }\n }, {\n key: \"load\",\n value: function load(url) {\n var _this = this;\n\n if (!url) {\n return;\n }\n\n this.url = url;\n this.imageData = {};\n var element = this.element,\n options = this.options;\n\n if (!options.rotatable && !options.scalable) {\n options.checkOrientation = false;\n } // Only IE10+ supports Typed Arrays\n\n\n if (!options.checkOrientation || !window.ArrayBuffer) {\n this.clone();\n return;\n } // Detect the mime type of the image directly if it is a Data URL\n\n\n if (REGEXP_DATA_URL.test(url)) {\n // Read ArrayBuffer from Data URL of JPEG images directly for better performance\n if (REGEXP_DATA_URL_JPEG.test(url)) {\n this.read(dataURLToArrayBuffer(url));\n } else {\n // Only a JPEG image may contains Exif Orientation information,\n // the rest types of Data URLs are not necessary to check orientation at all.\n this.clone();\n }\n\n return;\n } // 1. Detect the mime type of the image by a XMLHttpRequest.\n // 2. Load the image as ArrayBuffer for reading orientation if its a JPEG image.\n\n\n var xhr = new XMLHttpRequest();\n var clone = this.clone.bind(this);\n this.reloading = true;\n this.xhr = xhr; // 1. Cross origin requests are only supported for protocol schemes:\n // http, https, data, chrome, chrome-extension.\n // 2. Access to XMLHttpRequest from a Data URL will be blocked by CORS policy\n // in some browsers as IE11 and Safari.\n\n xhr.onabort = clone;\n xhr.onerror = clone;\n xhr.ontimeout = clone;\n\n xhr.onprogress = function () {\n // Abort the request directly if it not a JPEG image for better performance\n if (xhr.getResponseHeader('content-type') !== MIME_TYPE_JPEG) {\n xhr.abort();\n }\n };\n\n xhr.onload = function () {\n _this.read(xhr.response);\n };\n\n xhr.onloadend = function () {\n _this.reloading = false;\n _this.xhr = null;\n }; // Bust cache when there is a \"crossOrigin\" property to avoid browser cache error\n\n\n if (options.checkCrossOrigin && isCrossOriginURL(url) && element.crossOrigin) {\n url = addTimestamp(url);\n }\n\n xhr.open('GET', url);\n xhr.responseType = 'arraybuffer';\n xhr.withCredentials = element.crossOrigin === 'use-credentials';\n xhr.send();\n }\n }, {\n key: \"read\",\n value: function read(arrayBuffer) {\n var options = this.options,\n imageData = this.imageData; // Reset the orientation value to its default value 1\n // as some iOS browsers will render image with its orientation\n\n var orientation = resetAndGetOrientation(arrayBuffer);\n var rotate = 0;\n var scaleX = 1;\n var scaleY = 1;\n\n if (orientation > 1) {\n // Generate a new URL which has the default orientation value\n this.url = arrayBufferToDataURL(arrayBuffer, MIME_TYPE_JPEG);\n\n var _parseOrientation = parseOrientation(orientation);\n\n rotate = _parseOrientation.rotate;\n scaleX = _parseOrientation.scaleX;\n scaleY = _parseOrientation.scaleY;\n }\n\n if (options.rotatable) {\n imageData.rotate = rotate;\n }\n\n if (options.scalable) {\n imageData.scaleX = scaleX;\n imageData.scaleY = scaleY;\n }\n\n this.clone();\n }\n }, {\n key: \"clone\",\n value: function clone() {\n var element = this.element,\n url = this.url;\n var crossOrigin = element.crossOrigin;\n var crossOriginUrl = url;\n\n if (this.options.checkCrossOrigin && isCrossOriginURL(url)) {\n if (!crossOrigin) {\n crossOrigin = 'anonymous';\n } // Bust cache when there is not a \"crossOrigin\" property (#519)\n\n\n crossOriginUrl = addTimestamp(url);\n }\n\n this.crossOrigin = crossOrigin;\n this.crossOriginUrl = crossOriginUrl;\n var image = document.createElement('img');\n\n if (crossOrigin) {\n image.crossOrigin = crossOrigin;\n }\n\n image.src = crossOriginUrl || url;\n image.alt = element.alt || 'The image to crop';\n this.image = image;\n image.onload = this.start.bind(this);\n image.onerror = this.stop.bind(this);\n addClass(image, CLASS_HIDE);\n element.parentNode.insertBefore(image, element.nextSibling);\n }\n }, {\n key: \"start\",\n value: function start() {\n var _this2 = this;\n\n var image = this.image;\n image.onload = null;\n image.onerror = null;\n this.sizing = true; // Match all browsers that use WebKit as the layout engine in iOS devices,\n // such as Safari for iOS, Chrome for iOS, and in-app browsers.\n\n var isIOSWebKit = WINDOW.navigator && /(?:iPad|iPhone|iPod).*?AppleWebKit/i.test(WINDOW.navigator.userAgent);\n\n var done = function done(naturalWidth, naturalHeight) {\n assign(_this2.imageData, {\n naturalWidth: naturalWidth,\n naturalHeight: naturalHeight,\n aspectRatio: naturalWidth / naturalHeight\n });\n _this2.sizing = false;\n _this2.sized = true;\n\n _this2.build();\n }; // Most modern browsers (excepts iOS WebKit)\n\n\n if (image.naturalWidth && !isIOSWebKit) {\n done(image.naturalWidth, image.naturalHeight);\n return;\n }\n\n var sizingImage = document.createElement('img');\n var body = document.body || document.documentElement;\n this.sizingImage = sizingImage;\n\n sizingImage.onload = function () {\n done(sizingImage.width, sizingImage.height);\n\n if (!isIOSWebKit) {\n body.removeChild(sizingImage);\n }\n };\n\n sizingImage.src = image.src; // iOS WebKit will convert the image automatically\n // with its orientation once append it into DOM (#279)\n\n if (!isIOSWebKit) {\n sizingImage.style.cssText = 'left:0;' + 'max-height:none!important;' + 'max-width:none!important;' + 'min-height:0!important;' + 'min-width:0!important;' + 'opacity:0;' + 'position:absolute;' + 'top:0;' + 'z-index:-1;';\n body.appendChild(sizingImage);\n }\n }\n }, {\n key: \"stop\",\n value: function stop() {\n var image = this.image;\n image.onload = null;\n image.onerror = null;\n image.parentNode.removeChild(image);\n this.image = null;\n }\n }, {\n key: \"build\",\n value: function build() {\n if (!this.sized || this.ready) {\n return;\n }\n\n var element = this.element,\n options = this.options,\n image = this.image; // Create cropper elements\n\n var container = element.parentNode;\n var template = document.createElement('div');\n template.innerHTML = TEMPLATE;\n var cropper = template.querySelector(\".\".concat(NAMESPACE, \"-container\"));\n var canvas = cropper.querySelector(\".\".concat(NAMESPACE, \"-canvas\"));\n var dragBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-drag-box\"));\n var cropBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-crop-box\"));\n var face = cropBox.querySelector(\".\".concat(NAMESPACE, \"-face\"));\n this.container = container;\n this.cropper = cropper;\n this.canvas = canvas;\n this.dragBox = dragBox;\n this.cropBox = cropBox;\n this.viewBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-view-box\"));\n this.face = face;\n canvas.appendChild(image); // Hide the original image\n\n addClass(element, CLASS_HIDDEN); // Inserts the cropper after to the current image\n\n container.insertBefore(cropper, element.nextSibling); // Show the image if is hidden\n\n if (!this.isImg) {\n removeClass(image, CLASS_HIDE);\n }\n\n this.initPreview();\n this.bind();\n options.initialAspectRatio = Math.max(0, options.initialAspectRatio) || NaN;\n options.aspectRatio = Math.max(0, options.aspectRatio) || NaN;\n options.viewMode = Math.max(0, Math.min(3, Math.round(options.viewMode))) || 0;\n addClass(cropBox, CLASS_HIDDEN);\n\n if (!options.guides) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-dashed\")), CLASS_HIDDEN);\n }\n\n if (!options.center) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-center\")), CLASS_HIDDEN);\n }\n\n if (options.background) {\n addClass(cropper, \"\".concat(NAMESPACE, \"-bg\"));\n }\n\n if (!options.highlight) {\n addClass(face, CLASS_INVISIBLE);\n }\n\n if (options.cropBoxMovable) {\n addClass(face, CLASS_MOVE);\n setData(face, DATA_ACTION, ACTION_ALL);\n }\n\n if (!options.cropBoxResizable) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-line\")), CLASS_HIDDEN);\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-point\")), CLASS_HIDDEN);\n }\n\n this.render();\n this.ready = true;\n this.setDragMode(options.dragMode);\n\n if (options.autoCrop) {\n this.crop();\n }\n\n this.setData(options.data);\n\n if (isFunction(options.ready)) {\n addListener(element, EVENT_READY, options.ready, {\n once: true\n });\n }\n\n dispatchEvent(element, EVENT_READY);\n }\n }, {\n key: \"unbuild\",\n value: function unbuild() {\n if (!this.ready) {\n return;\n }\n\n this.ready = false;\n this.unbind();\n this.resetPreview();\n this.cropper.parentNode.removeChild(this.cropper);\n removeClass(this.element, CLASS_HIDDEN);\n }\n }, {\n key: \"uncreate\",\n value: function uncreate() {\n if (this.ready) {\n this.unbuild();\n this.ready = false;\n this.cropped = false;\n } else if (this.sizing) {\n this.sizingImage.onload = null;\n this.sizing = false;\n this.sized = false;\n } else if (this.reloading) {\n this.xhr.onabort = null;\n this.xhr.abort();\n } else if (this.image) {\n this.stop();\n }\n }\n /**\n * Get the no conflict cropper class.\n * @returns {Cropper} The cropper class.\n */\n\n }], [{\n key: \"noConflict\",\n value: function noConflict() {\n window.Cropper = AnotherCropper;\n return Cropper;\n }\n /**\n * Change the default options.\n * @param {Object} options - The new default options.\n */\n\n }, {\n key: \"setDefaults\",\n value: function setDefaults(options) {\n assign(DEFAULTS, isPlainObject(options) && options);\n }\n }]);\n\n return Cropper;\n }();\n\n assign(Cropper.prototype, render, preview, events, handlers, change, methods);\n\n return Cropper;\n\n})));\n","import { EventEmitter, Component, forwardRef, ChangeDetectionStrategy, ViewChild, Input, Output, NgModule, CUSTOM_ELEMENTS_SCHEMA, NO_ERRORS_SCHEMA } from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport * as editor from 'jsoneditor';\nimport { NG_VALUE_ACCESSOR, FormsModule } from '@angular/forms';\n\nimport * as ɵngcc0 from '@angular/core';\n\nconst _c0 = [\"jsonEditorContainer\"];\nclass JsonEditorOptions {\n constructor() {\n this.enableSort = true;\n this.enableTransform = true;\n this.escapeUnicode = false;\n this.expandAll = false;\n this.sortObjectKeys = false;\n this.history = true;\n this.mode = 'tree';\n this.search = true;\n this.indentation = 2;\n }\n}\n\nclass JsonEditorComponent {\n constructor() {\n this.id = 'angjsoneditor' + Math.floor(Math.random() * 1000000);\n this.disabled = false;\n this.isFocused = false;\n this.optionsChanged = false;\n this._data = {};\n this.options = new JsonEditorOptions();\n this.debug = false;\n this.change = new EventEmitter();\n this.jsonChange = new EventEmitter();\n // Implemented as part of ControlValueAccessor.\n this.onTouched = () => {\n };\n // Implemented as part of ControlValueAccessor.\n this.onChangeModel = (e) => {\n };\n }\n set data(value) {\n this._data = value;\n if (this.editor) {\n this.editor.destroy();\n this.ngOnInit();\n }\n }\n ngOnInit() {\n let optionsBefore = this.options;\n if (!this.optionsChanged && this.editor) {\n optionsBefore = this.editor.options;\n }\n if (!this.options.onChangeJSON && this.jsonChange) {\n this.options.onChangeJSON = this.onChangeJSON.bind(this);\n }\n if (!this.options.onChange && this.change) {\n this.options.onChange = this.onChange.bind(this);\n }\n const optionsCopy = Object.assign({}, optionsBefore);\n // expandAll is an option only supported by ang-jsoneditor and not by the the original jsoneditor.\n delete optionsCopy.expandAll;\n if (this.debug) {\n console.log(optionsCopy, this._data);\n }\n if (!this.jsonEditorContainer.nativeElement) {\n console.error(`Can't find the ElementRef reference for jsoneditor)`);\n }\n if (optionsCopy.mode === 'text' || optionsCopy.mode === 'code') {\n optionsCopy.onChangeJSON = null;\n }\n this.editor = new editor(this.jsonEditorContainer.nativeElement, optionsCopy, this._data);\n if (this.options.expandAll) {\n this.editor.expandAll();\n }\n }\n ngOnDestroy() {\n this.destroy();\n }\n /**\n * ngModel\n * ControlValueAccessor\n */\n // ControlValueAccessor implementation\n writeValue(value) {\n this.data = value;\n }\n // Implemented as part of ControlValueAccessor\n registerOnChange(fn) {\n this.onChangeModel = fn;\n }\n // Implemented as part of ControlValueAccessor.\n registerOnTouched(fn) {\n this.onTouched = fn;\n }\n // Implemented as part of ControlValueAccessor.\n setDisabledState(isDisabled) {\n this.disabled = isDisabled;\n }\n onChange(e) {\n if (this.editor) {\n try {\n const json = this.editor.get();\n this.onChangeModel(json);\n this.change.emit(json);\n }\n catch (e) {\n if (this.debug) {\n console.log(e);\n }\n }\n }\n }\n onChangeJSON(e) {\n if (this.editor) {\n try {\n this.jsonChange.emit(this.editor.get());\n }\n catch (e) {\n if (this.debug) {\n console.log(e);\n }\n }\n }\n }\n /**\n * JSON EDITOR FUNCTIONS\n */\n collapseAll() {\n this.editor.collapseAll();\n }\n expandAll() {\n this.editor.expandAll();\n }\n focus() {\n this.editor.focus();\n }\n get() {\n return this.editor.get();\n }\n getMode() {\n return this.editor.getMode();\n }\n getName() {\n return this.editor.getName();\n }\n getText() {\n return this.editor.getText();\n }\n set(json) {\n this.editor.set(json);\n }\n setMode(mode) {\n this.editor.setMode(mode);\n }\n setName(name) {\n this.editor.setName(name);\n }\n setSelection(start, end) {\n this.editor.setSelection(start, end);\n }\n getSelection() {\n return this.editor.getSelection();\n }\n getValidateSchema() {\n return this.editor.validateSchema;\n }\n setSchema(schema, schemaRefs) {\n this.editor.setSchema(schema, schemaRefs);\n }\n search(query) {\n this.editor.search(query);\n }\n setOptions(newOptions) {\n if (this.editor) {\n this.editor.destroy();\n }\n this.optionsChanged = true;\n this.options = newOptions;\n this.ngOnInit();\n }\n update(json) {\n this.editor.update(json);\n }\n destroy() {\n this.editor.destroy();\n }\n getEditor() {\n return this.editor;\n }\n isValidJson() {\n try {\n JSON.parse(this.getText());\n return true;\n }\n catch (e) {\n return false;\n }\n }\n}\nJsonEditorComponent.ɵfac = function JsonEditorComponent_Factory(t) { return new (t || JsonEditorComponent)(); };\nJsonEditorComponent.ɵcmp = /*@__PURE__*/ ɵngcc0.ɵɵdefineComponent({ type: JsonEditorComponent, selectors: [[\"json-editor\"]], viewQuery: function JsonEditorComponent_Query(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵviewQuery(_c0, 7);\n } if (rf & 2) {\n let _t;\n ɵngcc0.ɵɵqueryRefresh(_t = ɵngcc0.ɵɵloadQuery()) && (ctx.jsonEditorContainer = _t.first);\n } }, inputs: { options: \"options\", debug: \"debug\", data: \"data\" }, outputs: { change: \"change\", jsonChange: \"jsonChange\" }, features: [ɵngcc0.ɵɵProvidersFeature([\n {\n provide: NG_VALUE_ACCESSOR,\n useExisting: forwardRef(() => JsonEditorComponent),\n multi: true\n }\n ])], decls: 2, vars: 1, consts: [[3, \"id\"], [\"jsonEditorContainer\", \"\"]], template: function JsonEditorComponent_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"div\", 0, 1);\n } if (rf & 2) {\n ɵngcc0.ɵɵproperty(\"id\", ctx.id);\n } }, encapsulation: 2, changeDetection: 0 });\nJsonEditorComponent.ctorParameters = () => [];\nJsonEditorComponent.propDecorators = {\n jsonEditorContainer: [{ type: ViewChild, args: ['jsonEditorContainer', { static: true },] }],\n options: [{ type: Input }],\n data: [{ type: Input, args: ['data',] }],\n debug: [{ type: Input }],\n change: [{ type: Output }],\n jsonChange: [{ type: Output }]\n};\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(JsonEditorComponent, [{\n type: Component,\n args: [{\n // tslint:disable-next-line:component-selector\n selector: 'json-editor',\n template: `
    `,\n providers: [\n {\n provide: NG_VALUE_ACCESSOR,\n useExisting: forwardRef(() => JsonEditorComponent),\n multi: true\n }\n ],\n preserveWhitespaces: false,\n changeDetection: ChangeDetectionStrategy.OnPush\n }]\n }], function () { return []; }, { options: [{\n type: Input\n }], debug: [{\n type: Input\n }], change: [{\n type: Output\n }], jsonChange: [{\n type: Output\n }], data: [{\n type: Input,\n args: ['data']\n }], jsonEditorContainer: [{\n type: ViewChild,\n args: ['jsonEditorContainer', { static: true }]\n }] }); })();\n\nclass NgJsonEditorModule {\n static forRoot() {\n return {\n ngModule: NgJsonEditorModule,\n providers: []\n };\n }\n}\nNgJsonEditorModule.ɵfac = function NgJsonEditorModule_Factory(t) { return new (t || NgJsonEditorModule)(); };\nNgJsonEditorModule.ɵmod = /*@__PURE__*/ ɵngcc0.ɵɵdefineNgModule({ type: NgJsonEditorModule });\nNgJsonEditorModule.ɵinj = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjector({ imports: [CommonModule,\n FormsModule] });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(NgJsonEditorModule, [{\n type: NgModule,\n args: [{\n imports: [\n CommonModule,\n FormsModule\n ],\n declarations: [\n JsonEditorComponent\n ],\n exports: [\n JsonEditorComponent\n ],\n schemas: [\n CUSTOM_ELEMENTS_SCHEMA,\n NO_ERRORS_SCHEMA\n ]\n }]\n }], null, null); })();\n(function () { (typeof ngJitMode === \"undefined\" || ngJitMode) && ɵngcc0.ɵɵsetNgModuleScope(NgJsonEditorModule, { declarations: function () { return [JsonEditorComponent]; }, imports: function () { return [CommonModule,\n FormsModule]; }, exports: function () { return [JsonEditorComponent]; } }); })();\n\n/**\n * Generated bundle index. Do not edit.\n */\n\nexport { JsonEditorComponent, JsonEditorOptions, NgJsonEditorModule };\n\n","import { trigger, state, style, transition, animate } from '@angular/animations';\nimport { EventEmitter, Inject, Injectable, TemplateRef, ChangeDetectorRef, NgZone, Input, Component, ViewEncapsulation, ChangeDetectionStrategy, Output, InjectionToken, NgModule } from '@angular/core';\nimport { DomSanitizer } from '@angular/platform-browser';\nimport { Subject } from 'rxjs';\nimport { CommonModule } from '@angular/common';\n\nimport * as ɵngcc0 from '@angular/core';\nimport * as ɵngcc1 from '@angular/platform-browser';\nimport * as ɵngcc2 from '@angular/common';\n\nfunction NotificationComponent_div_1_div_1_ng_container_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementContainer(0);\n} }\nfunction NotificationComponent_div_1_div_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 8);\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_1_div_1_ng_container_1_Template, 1, 0, \"ng-container\", 9);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const ctx_r3 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngTemplateOutlet\", ctx_r3.title)(\"ngTemplateOutletContext\", ctx_r3.item.context);\n} }\nfunction NotificationComponent_div_1_ng_template_2_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"div\", 10);\n} if (rf & 2) {\n const ctx_r5 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵproperty(\"innerHTML\", ctx_r5.title, ɵngcc0.ɵɵsanitizeHtml);\n} }\nfunction NotificationComponent_div_1_div_4_ng_container_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementContainer(0);\n} }\nfunction NotificationComponent_div_1_div_4_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 11);\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_1_div_4_ng_container_1_Template, 1, 0, \"ng-container\", 9);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const ctx_r6 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngTemplateOutlet\", ctx_r6.content)(\"ngTemplateOutletContext\", ctx_r6.item.context);\n} }\nfunction NotificationComponent_div_1_ng_template_5_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"div\", 12);\n} if (rf & 2) {\n const ctx_r8 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵproperty(\"innerHTML\", ctx_r8.content, ɵngcc0.ɵɵsanitizeHtml);\n} }\nfunction NotificationComponent_div_1_div_7_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"div\", 13);\n} if (rf & 2) {\n const ctx_r9 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵproperty(\"innerHTML\", ctx_r9.safeSvg, ɵngcc0.ɵɵsanitizeHtml);\n} }\nfunction NotificationComponent_div_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\");\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_1_div_1_Template, 2, 2, \"div\", 3);\n ɵngcc0.ɵɵtemplate(2, NotificationComponent_div_1_ng_template_2_Template, 1, 1, \"ng-template\", null, 4, ɵngcc0.ɵɵtemplateRefExtractor);\n ɵngcc0.ɵɵtemplate(4, NotificationComponent_div_1_div_4_Template, 2, 2, \"div\", 5);\n ɵngcc0.ɵɵtemplate(5, NotificationComponent_div_1_ng_template_5_Template, 1, 1, \"ng-template\", null, 6, ɵngcc0.ɵɵtemplateRefExtractor);\n ɵngcc0.ɵɵtemplate(7, NotificationComponent_div_1_div_7_Template, 1, 1, \"div\", 7);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const _r4 = ɵngcc0.ɵɵreference(3);\n const _r7 = ɵngcc0.ɵɵreference(6);\n const ctx_r0 = ɵngcc0.ɵɵnextContext();\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx_r0.titleIsTemplate)(\"ngIfElse\", _r4);\n ɵngcc0.ɵɵadvance(3);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx_r0.contentIsTemplate)(\"ngIfElse\", _r7);\n ɵngcc0.ɵɵadvance(3);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx_r0.item.icon !== \"bare\");\n} }\nfunction NotificationComponent_div_2_div_1_ng_container_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementContainer(0);\n} }\nfunction NotificationComponent_div_2_div_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 17);\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_2_div_1_ng_container_1_Template, 1, 0, \"ng-container\", 9);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const ctx_r12 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngTemplateOutlet\", ctx_r12.item.html)(\"ngTemplateOutletContext\", ctx_r12.item.context);\n} }\nfunction NotificationComponent_div_2_ng_template_2_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"div\", 12);\n} if (rf & 2) {\n const ctx_r14 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵproperty(\"innerHTML\", ctx_r14.safeInputHtml, ɵngcc0.ɵɵsanitizeHtml);\n} }\nfunction NotificationComponent_div_2_div_4_Template(rf, ctx) { if (rf & 1) {\n const _r18 = ɵngcc0.ɵɵgetCurrentView();\n ɵngcc0.ɵɵelementStart(0, \"div\", 18);\n ɵngcc0.ɵɵlistener(\"click\", function NotificationComponent_div_2_div_4_Template_div_click_0_listener($event) { ɵngcc0.ɵɵrestoreView(_r18); const ctx_r17 = ɵngcc0.ɵɵnextContext(2); return ɵngcc0.ɵɵresetView(ctx_r17.onClickIcon($event)); });\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const ctx_r15 = ɵngcc0.ɵɵnextContext(2);\n ɵngcc0.ɵɵclassProp(\"icon-hover\", ctx_r15.clickIconToClose);\n ɵngcc0.ɵɵproperty(\"innerHTML\", ctx_r15.safeSvg, ɵngcc0.ɵɵsanitizeHtml);\n} }\nfunction NotificationComponent_div_2_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\");\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_2_div_1_Template, 2, 2, \"div\", 14);\n ɵngcc0.ɵɵtemplate(2, NotificationComponent_div_2_ng_template_2_Template, 1, 1, \"ng-template\", null, 15, ɵngcc0.ɵɵtemplateRefExtractor);\n ɵngcc0.ɵɵtemplate(4, NotificationComponent_div_2_div_4_Template, 1, 3, \"div\", 16);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const _r13 = ɵngcc0.ɵɵreference(3);\n const ctx_r1 = ɵngcc0.ɵɵnextContext();\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx_r1.htmlIsTemplate)(\"ngIfElse\", _r13);\n ɵngcc0.ɵɵadvance(3);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx_r1.item.icon);\n} }\nconst _c0 = function (a0) { return { \"width\": a0 }; };\nfunction NotificationComponent_div_3_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 19);\n ɵngcc0.ɵɵelement(1, \"span\", 20);\n ɵngcc0.ɵɵelementEnd();\n} if (rf & 2) {\n const ctx_r2 = ɵngcc0.ɵɵnextContext();\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngStyle\", ɵngcc0.ɵɵpureFunction1(1, _c0, ctx_r2.progressWidth + \"%\"));\n} }\nconst _c1 = function (a0, a1, a2, a3, a4, a5, a6, a7) { return { \"alert\": a0, \"error\": a1, \"warn\": a2, \"success\": a3, \"info\": a4, \"bare\": a5, \"rtl-mode\": a6, \"has-icon\": a7 }; };\nfunction SimpleNotificationsComponent_simple_notification_1_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelement(0, \"simple-notification\", 2);\n} if (rf & 2) {\n const a_r1 = ctx.$implicit;\n const i_r2 = ctx.index;\n const ctx_r0 = ɵngcc0.ɵɵnextContext();\n ɵngcc0.ɵɵproperty(\"item\", a_r1)(\"timeOut\", ctx_r0.timeOut)(\"clickToClose\", ctx_r0.clickToClose)(\"clickIconToClose\", ctx_r0.clickIconToClose)(\"maxLength\", ctx_r0.maxLength)(\"showProgressBar\", ctx_r0.showProgressBar)(\"pauseOnHover\", ctx_r0.pauseOnHover)(\"theClass\", ctx_r0.theClass)(\"rtl\", ctx_r0.rtl)(\"animate\", ctx_r0.animate)(\"position\", i_r2);\n} }\nconst DEFAULT_ICONS = {\r\n alert: `\n \n \n \n \n `,\r\n error: `\n \n \n \n \n `,\r\n info: `\n \n \n \n \n `,\r\n success: `\n \n \n \n \n `,\r\n warn: `\n \n \n \n \n \n `\r\n};\n\nvar NotificationType;\r\n(function (NotificationType) {\r\n NotificationType[\"Success\"] = \"success\";\r\n NotificationType[\"Error\"] = \"error\";\r\n NotificationType[\"Alert\"] = \"alert\";\r\n NotificationType[\"Info\"] = \"info\";\r\n NotificationType[\"Warn\"] = \"warn\";\r\n NotificationType[\"Bare\"] = \"bare\";\r\n})(NotificationType || (NotificationType = {}));\n\nvar __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n};\r\nvar __param = (this && this.__param) || function (paramIndex, decorator) {\r\n return function (target, key) { decorator(target, key, paramIndex); }\r\n};\r\nlet NotificationsService = class NotificationsService {\r\n constructor(globalOptions) {\r\n this.globalOptions = globalOptions;\r\n this.emitter = new Subject();\r\n this.icons = DEFAULT_ICONS;\r\n }\r\n set(notification, to) {\r\n notification.id = notification.override && notification.override.id ?\r\n notification.override.id :\r\n Math.random().toString(36).substring(3);\r\n notification.click = new EventEmitter();\r\n notification.clickIcon = new EventEmitter();\r\n notification.timeoutEnd = new EventEmitter();\r\n this.emitter.next({ command: 'set', notification, add: to });\r\n return notification;\r\n }\r\n success(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Success, icon: this.icons.success, override, context }, true);\r\n }\r\n error(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Error, icon: this.icons.error, override, context }, true);\r\n }\r\n alert(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Alert, icon: this.icons.alert, override, context }, true);\r\n }\r\n info(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Info, icon: this.icons.info, override, context }, true);\r\n }\r\n warn(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Warn, icon: this.icons.warn, override, context }, true);\r\n }\r\n bare(title = '', content = '', override, context) {\r\n return this.set({ title, content: content || '', type: NotificationType.Bare, icon: 'bare', override, context }, true);\r\n }\r\n // With type method\r\n create(title = '', content = '', type = NotificationType.Success, override, context) {\r\n return this.set({ title, content, type, icon: this.icons[type], override, context }, true);\r\n }\r\n // HTML Notification method\r\n html(html, type = NotificationType.Success, override, icon = 'bare', context) {\r\n return this.set({ html, type, icon: this.icons[icon], override, context }, true);\r\n }\r\n // Remove all notifications method\r\n remove(id) {\r\n if (id) {\r\n this.emitter.next({ command: 'clean', id });\r\n }\r\n else {\r\n this.emitter.next({ command: 'cleanAll' });\r\n }\r\n }\r\n};\nNotificationsService.ɵfac = function NotificationsService_Factory(t) { return new (t || NotificationsService)(ɵngcc0.ɵɵinject('options')); };\nNotificationsService.ɵprov = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjectable({ token: NotificationsService, factory: function (t) { return NotificationsService.ɵfac(t); } });\r\nNotificationsService.ctorParameters = () => [\r\n { type: undefined, decorators: [{ type: Inject, args: ['options',] }] }\r\n];\r\nNotificationsService = __decorate([ __param(0, Inject('options'))\r\n], NotificationsService);\n\nvar __decorate$1 = (this && this.__decorate) || function (decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n};\r\nlet NotificationComponent = class NotificationComponent {\r\n constructor(notificationService, domSanitizer, cd, zone) {\r\n this.notificationService = notificationService;\r\n this.domSanitizer = domSanitizer;\r\n this.cd = cd;\r\n this.zone = zone;\r\n this.titleIsTemplate = false;\r\n this.contentIsTemplate = false;\r\n this.htmlIsTemplate = false;\r\n this.progressWidth = 0;\r\n this.stopTime = false;\r\n this.framesPerSecond = 40;\r\n this.instance = () => {\r\n const now = new Date().getTime();\r\n if (this.endTime < now) {\r\n this.remove();\r\n this.item.timeoutEnd.emit();\r\n }\r\n else if (!this.stopTime) {\r\n if (this.showProgressBar) {\r\n // We add this.sleepTime just to have 100% before close\r\n this.progressWidth = Math.min((now - this.startTime + this.sleepTime) * 100 / this.timeOut, 100);\r\n }\r\n this.timer = setTimeout(this.instance, this.sleepTime);\r\n }\r\n this.zone.run(() => {\r\n if (!this.cd.destroyed) {\r\n this.cd.detectChanges();\r\n }\r\n });\r\n };\r\n }\r\n ngOnInit() {\r\n if (this.item.override) {\r\n this.attachOverrides();\r\n }\r\n if (this.animate) {\r\n this.item.state = this.animate;\r\n }\r\n if (this.timeOut !== 0) {\r\n this.startTimeOut();\r\n }\r\n this.contentType(this.item.title, 'title');\r\n this.contentType(this.item.content, 'content');\r\n this.contentType(this.item.html, 'html');\r\n this.safeSvg = this.domSanitizer.bypassSecurityTrustHtml(this.icon || this.item.icon);\r\n this.safeInputHtml = this.domSanitizer.bypassSecurityTrustHtml(this.item.html);\r\n }\r\n ngOnDestroy() {\r\n clearTimeout(this.timer);\r\n this.cd.detach();\r\n }\r\n startTimeOut() {\r\n this.sleepTime = 1000 / this.framesPerSecond /* ms */;\r\n this.startTime = new Date().getTime();\r\n this.endTime = this.startTime + this.timeOut;\r\n this.zone.runOutsideAngular(() => this.timer = setTimeout(this.instance, this.sleepTime));\r\n }\r\n onEnter() {\r\n if (this.pauseOnHover) {\r\n this.stopTime = true;\r\n this.pauseStart = new Date().getTime();\r\n }\r\n }\r\n onLeave() {\r\n if (this.pauseOnHover) {\r\n this.stopTime = false;\r\n this.startTime += (new Date().getTime() - this.pauseStart);\r\n this.endTime += (new Date().getTime() - this.pauseStart);\r\n this.zone.runOutsideAngular(() => setTimeout(this.instance, this.sleepTime));\r\n }\r\n }\r\n onClick(event) {\r\n this.item.click.emit(event);\r\n if (this.clickToClose) {\r\n this.remove();\r\n }\r\n }\r\n onClickIcon(event) {\r\n this.item.clickIcon.emit(event);\r\n if (this.clickIconToClose) {\r\n this.remove();\r\n }\r\n }\r\n // Attach all the overrides\r\n attachOverrides() {\r\n Object.keys(this.item.override).forEach(a => {\r\n if (this.hasOwnProperty(a)) {\r\n this[a] = this.item.override[a];\r\n }\r\n });\r\n }\r\n remove() {\r\n if (this.animate) {\r\n this.item.state = this.animate + 'Out';\r\n setTimeout(() => {\r\n this.notificationService.set(this.item, false);\r\n }, 310);\r\n }\r\n else {\r\n this.notificationService.set(this.item, false);\r\n }\r\n }\r\n contentType(item, key) {\r\n if (item instanceof TemplateRef) {\r\n this[key] = item;\r\n }\r\n else {\r\n this[key] = this.domSanitizer.bypassSecurityTrustHtml(item);\r\n }\r\n this[key + 'IsTemplate'] = item instanceof TemplateRef;\r\n }\r\n};\nNotificationComponent.ɵfac = function NotificationComponent_Factory(t) { return new (t || NotificationComponent)(ɵngcc0.ɵɵdirectiveInject(NotificationsService), ɵngcc0.ɵɵdirectiveInject(ɵngcc1.DomSanitizer), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.NgZone)); };\nNotificationComponent.ɵcmp = /*@__PURE__*/ ɵngcc0.ɵɵdefineComponent({ type: NotificationComponent, selectors: [[\"simple-notification\"]], inputs: { timeOut: \"timeOut\", showProgressBar: \"showProgressBar\", pauseOnHover: \"pauseOnHover\", clickToClose: \"clickToClose\", clickIconToClose: \"clickIconToClose\", maxLength: \"maxLength\", theClass: \"theClass\", rtl: \"rtl\", animate: \"animate\", position: \"position\", item: \"item\" }, decls: 4, vars: 16, consts: [[1, \"simple-notification\", 3, \"ngClass\", \"click\", \"mouseenter\", \"mouseleave\"], [4, \"ngIf\"], [\"class\", \"sn-progress-loader\", 4, \"ngIf\"], [\"class\", \"sn-title\", 4, \"ngIf\", \"ngIfElse\"], [\"regularTitle\", \"\"], [\"class\", \"sn-content\", 4, \"ngIf\", \"ngIfElse\"], [\"regularContent\", \"\"], [\"class\", \"icon\", 3, \"innerHTML\", 4, \"ngIf\"], [1, \"sn-title\"], [4, \"ngTemplateOutlet\", \"ngTemplateOutletContext\"], [1, \"sn-title\", 3, \"innerHTML\"], [1, \"sn-content\"], [1, \"sn-content\", 3, \"innerHTML\"], [1, \"icon\", 3, \"innerHTML\"], [\"class\", \"sn-html\", 4, \"ngIf\", \"ngIfElse\"], [\"regularHtml\", \"\"], [\"class\", \"icon\", 3, \"icon-hover\", \"innerHTML\", \"click\", 4, \"ngIf\"], [1, \"sn-html\"], [1, \"icon\", 3, \"innerHTML\", \"click\"], [1, \"sn-progress-loader\"], [3, \"ngStyle\"]], template: function NotificationComponent_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 0);\n ɵngcc0.ɵɵlistener(\"click\", function NotificationComponent_Template_div_click_0_listener($event) { return ctx.onClick($event); })(\"mouseenter\", function NotificationComponent_Template_div_mouseenter_0_listener() { return ctx.onEnter(); })(\"mouseleave\", function NotificationComponent_Template_div_mouseleave_0_listener() { return ctx.onLeave(); });\n ɵngcc0.ɵɵtemplate(1, NotificationComponent_div_1_Template, 8, 5, \"div\", 1);\n ɵngcc0.ɵɵtemplate(2, NotificationComponent_div_2_Template, 5, 3, \"div\", 1);\n ɵngcc0.ɵɵtemplate(3, NotificationComponent_div_3_Template, 2, 3, \"div\", 2);\n ɵngcc0.ɵɵelementEnd();\n } if (rf & 2) {\n ɵngcc0.ɵɵclassMap(ctx.theClass);\n ɵngcc0.ɵɵproperty(\"@enterLeave\", ctx.item.state)(\"ngClass\", ɵngcc0.ɵɵpureFunction8(7, _c1, ctx.item.type === \"alert\", ctx.item.type === \"error\", ctx.item.type === \"warn\", ctx.item.type === \"success\", ctx.item.type === \"info\", ctx.item.type === \"bare\", ctx.rtl, ctx.item.icon !== \"bare\"));\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngIf\", !ctx.item.html);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx.item.html);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngIf\", ctx.showProgressBar);\n } }, dependencies: [ɵngcc2.NgClass, ɵngcc2.NgIf, ɵngcc2.NgTemplateOutlet, ɵngcc2.NgStyle], styles: [\".simple-notification{width:100%;padding:10px 20px;box-sizing:border-box;position:relative;float:left;margin-bottom:10px;color:#fff;cursor:pointer;transition:.5s;min-height:70px}.simple-notification .sn-content,.simple-notification .sn-html,.simple-notification .sn-title{margin:0}.simple-notification .sn-title{line-height:30px;font-size:20px}.simple-notification .sn-content{font-size:16px;line-height:20px}.simple-notification.has-icon .sn-content,.simple-notification.has-icon .sn-html,.simple-notification.has-icon .sn-title{padding:0 50px 0 0}.simple-notification .icon{position:absolute;box-sizing:border-box;top:0;right:0;width:70px;height:70px;padding:10px}.simple-notification .icon.icon-hover:hover{opacity:.5}.simple-notification .icon svg{fill:#fff;width:100%;height:100%}.simple-notification .icon svg g{fill:#fff}.simple-notification.rtl-mode.has-icon .sn-content,.simple-notification.rtl-mode.has-icon .sn-html,.simple-notification.rtl-mode.has-icon .sn-title{padding:0 0 0 50px}.simple-notification.rtl-mode{direction:rtl}.simple-notification.rtl-mode .sn-content{padding:0 0 0 50px}.simple-notification.rtl-mode svg{left:0;right:auto}.simple-notification.error{background:#f44336}.simple-notification.success{background:#8bc34a}.simple-notification.alert{background:#ffdb5b}.simple-notification.info{background:#03a9f4}.simple-notification.warn{background:#ffdb5b}.simple-notification .sn-progress-loader{position:absolute;top:0;left:0;width:100%;height:5px}.simple-notification .sn-progress-loader span{float:left;height:100%}.simple-notification.success .sn-progress-loader span{background:#689f38}.simple-notification.error .sn-progress-loader span{background:#d32f2f}.simple-notification.alert .sn-progress-loader span{background:#edc242}.simple-notification.info .sn-progress-loader span{background:#0288d1}.simple-notification.warn .sn-progress-loader span{background:#edc242}.simple-notification.bare .sn-progress-loader span{background:#ccc}.simple-notification.warn div .sn-content,.simple-notification.warn div .sn-html,.simple-notification.warn div .sn-title{color:#444}\"], encapsulation: 2, data: { animation: [\n trigger('enterLeave', [\n // Fade\n state('fade', style({ opacity: 1 })),\n transition('* => fade', [\n style({ opacity: 0 }),\n animate('400ms ease-in-out')\n ]),\n state('fadeOut', style({ opacity: 0 })),\n transition('fade => fadeOut', [\n style({ opacity: 1 }),\n animate('300ms ease-in-out')\n ]),\n // Enter from top\n state('fromTop', style({ opacity: 1, transform: 'translateY(0)' })),\n transition('* => fromTop', [\n style({ opacity: 0, transform: 'translateY(-5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromTopOut', style({ opacity: 0, transform: 'translateY(5%)' })),\n transition('fromTop => fromTopOut', [\n style({ opacity: 1, transform: 'translateY(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from right\n state('fromRight', style({ opacity: 1, transform: 'translateX(0)' })),\n transition('* => fromRight', [\n style({ opacity: 0, transform: 'translateX(5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromRightOut', style({ opacity: 0, transform: 'translateX(-5%)' })),\n transition('fromRight => fromRightOut', [\n style({ opacity: 1, transform: 'translateX(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from bottom\n state('fromBottom', style({ opacity: 1, transform: 'translateY(0)' })),\n transition('* => fromBottom', [\n style({ opacity: 0, transform: 'translateY(5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromBottomOut', style({ opacity: 0, transform: 'translateY(-5%)' })),\n transition('fromBottom => fromBottomOut', [\n style({ opacity: 1, transform: 'translateY(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from left\n state('fromLeft', style({ opacity: 1, transform: 'translateX(0)' })),\n transition('* => fromLeft', [\n style({ opacity: 0, transform: 'translateX(-5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromLeftOut', style({ opacity: 0, transform: 'translateX(5%)' })),\n transition('fromLeft => fromLeftOut', [\n style({ opacity: 1, transform: 'translateX(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Rotate\n state('scale', style({ opacity: 1, transform: 'scale(1)' })),\n transition('* => scale', [\n style({ opacity: 0, transform: 'scale(0)' }),\n animate('400ms ease-in-out')\n ]),\n state('scaleOut', style({ opacity: 0, transform: 'scale(0)' })),\n transition('scale => scaleOut', [\n style({ opacity: 1, transform: 'scale(1)' }),\n animate('400ms ease-in-out')\n ]),\n // Scale\n state('rotate', style({ opacity: 1, transform: 'rotate(0deg)' })),\n transition('* => rotate', [\n style({ opacity: 0, transform: 'rotate(5deg)' }),\n animate('400ms ease-in-out')\n ]),\n state('rotateOut', style({ opacity: 0, transform: 'rotate(-5deg)' })),\n transition('rotate => rotateOut', [\n style({ opacity: 1, transform: 'rotate(0deg)' }),\n animate('400ms ease-in-out')\n ])\n ])\n ] }, changeDetection: 0 });\r\nNotificationComponent.ctorParameters = () => [\r\n { type: NotificationsService },\r\n { type: DomSanitizer },\r\n { type: ChangeDetectorRef },\r\n { type: NgZone }\r\n];\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"timeOut\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"showProgressBar\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"pauseOnHover\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"clickToClose\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"clickIconToClose\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"maxLength\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"theClass\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"rtl\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"animate\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"position\", void 0);\r\n__decorate$1([\r\n Input()\r\n], NotificationComponent.prototype, \"item\", void 0);\n\nvar NotificationAnimationType;\r\n(function (NotificationAnimationType) {\r\n NotificationAnimationType[\"Fade\"] = \"fade\";\r\n NotificationAnimationType[\"FromTop\"] = \"fromTop\";\r\n NotificationAnimationType[\"FromRight\"] = \"fromRight\";\r\n NotificationAnimationType[\"FromBottom\"] = \"fromBottom\";\r\n NotificationAnimationType[\"FromLeft\"] = \"fromLeft\";\r\n NotificationAnimationType[\"Scale\"] = \"scale\";\r\n NotificationAnimationType[\"Rotate\"] = \"rotate\";\r\n})(NotificationAnimationType || (NotificationAnimationType = {}));\n\nvar __decorate$2 = (this && this.__decorate) || function (decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n};\r\nlet SimpleNotificationsComponent = class SimpleNotificationsComponent {\r\n constructor(service, cd) {\r\n this.service = service;\r\n this.cd = cd;\r\n this.create = new EventEmitter();\r\n this.destroy = new EventEmitter();\r\n this.notifications = [];\r\n this.position = ['bottom', 'right'];\r\n // Received values\r\n this.lastOnBottom = true;\r\n this.maxStack = 8;\r\n this.preventLastDuplicates = false;\r\n this.preventDuplicates = false;\r\n // Sent values\r\n this.timeOut = 0;\r\n this.maxLength = 0;\r\n this.clickToClose = true;\r\n this.clickIconToClose = false;\r\n this.showProgressBar = true;\r\n this.pauseOnHover = true;\r\n this.theClass = '';\r\n this.rtl = false;\r\n this.animate = NotificationAnimationType.FromRight;\r\n this.usingComponentOptions = false;\r\n }\r\n set options(opt) {\r\n this.usingComponentOptions = true;\r\n this.attachChanges(opt);\r\n }\r\n ngOnInit() {\r\n /**\r\n * Only attach global options if config\r\n * options were never sent through input\r\n */\r\n if (!this.usingComponentOptions) {\r\n this.attachChanges(this.service.globalOptions);\r\n }\r\n this.listener = this.service.emitter\r\n .subscribe(item => {\r\n switch (item.command) {\r\n case 'cleanAll':\r\n this.notifications = [];\r\n break;\r\n case 'clean':\r\n this.cleanSingle(item.id);\r\n break;\r\n case 'set':\r\n if (item.add) {\r\n this.add(item.notification);\r\n }\r\n else {\r\n this.defaultBehavior(item);\r\n }\r\n break;\r\n default:\r\n this.defaultBehavior(item);\r\n break;\r\n }\r\n if (!this.cd.destroyed) {\r\n this.cd.detectChanges();\r\n }\r\n });\r\n }\r\n ngOnDestroy() {\r\n if (this.listener) {\r\n this.listener.unsubscribe();\r\n }\r\n this.cd.detach();\r\n }\r\n // Default behavior on event\r\n defaultBehavior(value) {\r\n this.notifications.splice(this.notifications.indexOf(value.notification), 1);\r\n this.destroy.emit(this.buildEmit(value.notification, false));\r\n }\r\n // Add the new notification to the notification array\r\n add(item) {\r\n item.createdOn = new Date();\r\n const toBlock = this.preventLastDuplicates || this.preventDuplicates ? this.block(item) : false;\r\n // Save this as the last created notification\r\n this.lastNotificationCreated = item;\r\n // Override icon if set\r\n if (item.override && item.override.icons && item.override.icons[item.type]) {\r\n item.icon = item.override.icons[item.type];\r\n }\r\n if (!toBlock) {\r\n // Check if the notification should be added at the start or the end of the array\r\n if (this.lastOnBottom) {\r\n if (this.notifications.length >= this.maxStack) {\r\n this.notifications.splice(0, 1);\r\n }\r\n this.notifications.push(item);\r\n }\r\n else {\r\n if (this.notifications.length >= this.maxStack) {\r\n this.notifications.splice(this.notifications.length - 1, 1);\r\n }\r\n this.notifications.splice(0, 0, item);\r\n }\r\n this.create.emit(this.buildEmit(item, true));\r\n }\r\n }\r\n // Check if notifications should be prevented\r\n block(item) {\r\n const toCheck = item.html ? this.checkHtml : this.checkStandard;\r\n if (this.preventDuplicates && this.notifications.length > 0) {\r\n for (const notification of this.notifications) {\r\n if (toCheck(notification, item)) {\r\n return true;\r\n }\r\n }\r\n }\r\n if (this.preventLastDuplicates) {\r\n let comp;\r\n if (this.preventLastDuplicates === 'visible' && this.notifications.length > 0) {\r\n if (this.lastOnBottom) {\r\n comp = this.notifications[this.notifications.length - 1];\r\n }\r\n else {\r\n comp = this.notifications[0];\r\n }\r\n }\r\n else if (this.preventLastDuplicates === 'all' && this.lastNotificationCreated) {\r\n comp = this.lastNotificationCreated;\r\n }\r\n else {\r\n return false;\r\n }\r\n return toCheck(comp, item);\r\n }\r\n return false;\r\n }\r\n checkStandard(checker, item) {\r\n return checker.type === item.type && checker.title === item.title && checker.content === item.content;\r\n }\r\n checkHtml(checker, item) {\r\n return checker.html ?\r\n checker.type === item.type && checker.title === item.title && checker.content === item.content && checker.html === item.html :\r\n false;\r\n }\r\n // Attach all the changes received in the options object\r\n attachChanges(options) {\r\n for (const key in options) {\r\n if (this.hasOwnProperty(key)) {\r\n this[key] = options[key];\r\n }\r\n else if (key === 'icons') {\r\n this.service.icons = options[key];\r\n }\r\n }\r\n }\r\n buildEmit(notification, to) {\r\n const toEmit = {\r\n createdOn: notification.createdOn,\r\n type: notification.type,\r\n icon: notification.icon,\r\n id: notification.id\r\n };\r\n if (notification.html) {\r\n toEmit.html = notification.html;\r\n }\r\n else {\r\n toEmit.title = notification.title;\r\n toEmit.content = notification.content;\r\n }\r\n if (!to) {\r\n toEmit.destroyedOn = new Date();\r\n }\r\n return toEmit;\r\n }\r\n cleanSingle(id) {\r\n let indexOfDelete = 0;\r\n let doDelete = false;\r\n let noti;\r\n this.notifications.forEach((notification, idx) => {\r\n if (notification.id === id) {\r\n indexOfDelete = idx;\r\n noti = notification;\r\n doDelete = true;\r\n }\r\n });\r\n if (doDelete) {\r\n this.notifications.splice(indexOfDelete, 1);\r\n this.destroy.emit(this.buildEmit(noti, false));\r\n }\r\n }\r\n};\nSimpleNotificationsComponent.ɵfac = function SimpleNotificationsComponent_Factory(t) { return new (t || SimpleNotificationsComponent)(ɵngcc0.ɵɵdirectiveInject(NotificationsService), ɵngcc0.ɵɵdirectiveInject(ɵngcc0.ChangeDetectorRef)); };\nSimpleNotificationsComponent.ɵcmp = /*@__PURE__*/ ɵngcc0.ɵɵdefineComponent({ type: SimpleNotificationsComponent, selectors: [[\"simple-notifications\"]], inputs: { options: \"options\" }, outputs: { create: \"create\", destroy: \"destroy\" }, decls: 2, vars: 2, consts: [[1, \"simple-notification-wrapper\", 3, \"ngClass\"], [3, \"item\", \"timeOut\", \"clickToClose\", \"clickIconToClose\", \"maxLength\", \"showProgressBar\", \"pauseOnHover\", \"theClass\", \"rtl\", \"animate\", \"position\", 4, \"ngFor\", \"ngForOf\"], [3, \"item\", \"timeOut\", \"clickToClose\", \"clickIconToClose\", \"maxLength\", \"showProgressBar\", \"pauseOnHover\", \"theClass\", \"rtl\", \"animate\", \"position\"]], template: function SimpleNotificationsComponent_Template(rf, ctx) { if (rf & 1) {\n ɵngcc0.ɵɵelementStart(0, \"div\", 0);\n ɵngcc0.ɵɵtemplate(1, SimpleNotificationsComponent_simple_notification_1_Template, 1, 11, \"simple-notification\", 1);\n ɵngcc0.ɵɵelementEnd();\n } if (rf & 2) {\n ɵngcc0.ɵɵproperty(\"ngClass\", ctx.position);\n ɵngcc0.ɵɵadvance(1);\n ɵngcc0.ɵɵproperty(\"ngForOf\", ctx.notifications);\n } }, dependencies: [ɵngcc2.NgClass, ɵngcc2.NgForOf, NotificationComponent], styles: [\".simple-notification-wrapper{position:fixed;width:300px;z-index:1000}.simple-notification-wrapper.left{left:20px}.simple-notification-wrapper.top{top:20px}.simple-notification-wrapper.right{right:20px}.simple-notification-wrapper.bottom{bottom:20px}.simple-notification-wrapper.center{left:50%;transform:translateX(-50%)}.simple-notification-wrapper.middle{top:50%;transform:translateY(-50%)}.simple-notification-wrapper.middle.center{transform:translate(-50%,-50%)}@media (max-width:340px){.simple-notification-wrapper{width:auto;left:20px;right:20px}}\"], encapsulation: 2, changeDetection: 0 });\r\nSimpleNotificationsComponent.ctorParameters = () => [\r\n { type: NotificationsService },\r\n { type: ChangeDetectorRef }\r\n];\r\n__decorate$2([\r\n Input()\r\n], SimpleNotificationsComponent.prototype, \"options\", null);\r\n__decorate$2([\r\n Output()\r\n], SimpleNotificationsComponent.prototype, \"create\", void 0);\r\n__decorate$2([\r\n Output()\r\n], SimpleNotificationsComponent.prototype, \"destroy\", void 0);\n\nconst DEFAULT_OPTIONS = {\r\n position: ['bottom', 'right'],\r\n timeOut: 0,\r\n showProgressBar: true,\r\n pauseOnHover: true,\r\n lastOnBottom: true,\r\n clickToClose: true,\r\n clickIconToClose: false,\r\n maxLength: 0,\r\n maxStack: 8,\r\n preventDuplicates: false,\r\n preventLastDuplicates: false,\r\n theClass: '',\r\n rtl: false,\r\n animate: NotificationAnimationType.FromRight,\r\n icons: DEFAULT_ICONS\r\n};\n\nvar __decorate$3 = (this && this.__decorate) || function (decorators, target, key, desc) {\r\n var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;\r\n if (typeof Reflect === \"object\" && typeof Reflect.decorate === \"function\") r = Reflect.decorate(decorators, target, key, desc);\r\n else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;\r\n return c > 3 && r && Object.defineProperty(target, key, r), r;\r\n};\r\nvar SimpleNotificationsModule_1;\r\nconst OPTIONS = new InjectionToken('options');\r\nfunction optionsFactory(options) {\r\n return Object.assign(Object.assign({}, DEFAULT_OPTIONS), options);\r\n}\r\nlet SimpleNotificationsModule = SimpleNotificationsModule_1 = class SimpleNotificationsModule {\r\n static forRoot(options = {}) {\r\n return {\r\n ngModule: SimpleNotificationsModule_1,\r\n providers: [\r\n NotificationsService,\r\n {\r\n provide: OPTIONS,\r\n useValue: options\r\n },\r\n {\r\n provide: 'options',\r\n useFactory: optionsFactory,\r\n deps: [OPTIONS]\r\n }\r\n ]\r\n };\r\n }\r\n};\nSimpleNotificationsModule.ɵfac = function SimpleNotificationsModule_Factory(t) { return new (t || SimpleNotificationsModule)(); };\nSimpleNotificationsModule.ɵmod = /*@__PURE__*/ ɵngcc0.ɵɵdefineNgModule({ type: SimpleNotificationsModule });\nSimpleNotificationsModule.ɵinj = /*@__PURE__*/ ɵngcc0.ɵɵdefineInjector({ imports: [CommonModule] });\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(NotificationsService, [{\n type: Injectable\n }], function () { return [{ type: undefined, decorators: [{\n type: Inject,\n args: ['options']\n }] }]; }, null); })();\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(NotificationComponent, [{\n type: Component,\n args: [{ selector: 'simple-notification', encapsulation: ViewEncapsulation.None, animations: [\n trigger('enterLeave', [\n // Fade\n state('fade', style({ opacity: 1 })),\n transition('* => fade', [\n style({ opacity: 0 }),\n animate('400ms ease-in-out')\n ]),\n state('fadeOut', style({ opacity: 0 })),\n transition('fade => fadeOut', [\n style({ opacity: 1 }),\n animate('300ms ease-in-out')\n ]),\n // Enter from top\n state('fromTop', style({ opacity: 1, transform: 'translateY(0)' })),\n transition('* => fromTop', [\n style({ opacity: 0, transform: 'translateY(-5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromTopOut', style({ opacity: 0, transform: 'translateY(5%)' })),\n transition('fromTop => fromTopOut', [\n style({ opacity: 1, transform: 'translateY(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from right\n state('fromRight', style({ opacity: 1, transform: 'translateX(0)' })),\n transition('* => fromRight', [\n style({ opacity: 0, transform: 'translateX(5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromRightOut', style({ opacity: 0, transform: 'translateX(-5%)' })),\n transition('fromRight => fromRightOut', [\n style({ opacity: 1, transform: 'translateX(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from bottom\n state('fromBottom', style({ opacity: 1, transform: 'translateY(0)' })),\n transition('* => fromBottom', [\n style({ opacity: 0, transform: 'translateY(5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromBottomOut', style({ opacity: 0, transform: 'translateY(-5%)' })),\n transition('fromBottom => fromBottomOut', [\n style({ opacity: 1, transform: 'translateY(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Enter from left\n state('fromLeft', style({ opacity: 1, transform: 'translateX(0)' })),\n transition('* => fromLeft', [\n style({ opacity: 0, transform: 'translateX(-5%)' }),\n animate('400ms ease-in-out')\n ]),\n state('fromLeftOut', style({ opacity: 0, transform: 'translateX(5%)' })),\n transition('fromLeft => fromLeftOut', [\n style({ opacity: 1, transform: 'translateX(0)' }),\n animate('300ms ease-in-out')\n ]),\n // Rotate\n state('scale', style({ opacity: 1, transform: 'scale(1)' })),\n transition('* => scale', [\n style({ opacity: 0, transform: 'scale(0)' }),\n animate('400ms ease-in-out')\n ]),\n state('scaleOut', style({ opacity: 0, transform: 'scale(0)' })),\n transition('scale => scaleOut', [\n style({ opacity: 1, transform: 'scale(1)' }),\n animate('400ms ease-in-out')\n ]),\n // Scale\n state('rotate', style({ opacity: 1, transform: 'rotate(0deg)' })),\n transition('* => rotate', [\n style({ opacity: 0, transform: 'rotate(5deg)' }),\n animate('400ms ease-in-out')\n ]),\n state('rotateOut', style({ opacity: 0, transform: 'rotate(-5deg)' })),\n transition('rotate => rotateOut', [\n style({ opacity: 1, transform: 'rotate(0deg)' }),\n animate('400ms ease-in-out')\n ])\n ])\n ], template: \"
    \\r\\n\\r\\n
    \\r\\n\\r\\n
    \\r\\n \\r\\n
    \\r\\n\\r\\n \\r\\n
    \\r\\n
    \\r\\n\\r\\n
    \\r\\n \\r\\n
    \\r\\n\\r\\n \\r\\n
    \\r\\n
    \\r\\n\\r\\n
    \\r\\n
    \\r\\n
    \\r\\n
    \\r\\n \\r\\n
    \\r\\n\\r\\n \\r\\n
    \\r\\n
    \\r\\n\\r\\n
    \\r\\n
    \\r\\n\\r\\n
    \\r\\n \\r\\n
    \\r\\n\\r\\n
    \\r\\n\", changeDetection: ChangeDetectionStrategy.OnPush, styles: [\".simple-notification{width:100%;padding:10px 20px;box-sizing:border-box;position:relative;float:left;margin-bottom:10px;color:#fff;cursor:pointer;transition:.5s;min-height:70px}.simple-notification .sn-content,.simple-notification .sn-html,.simple-notification .sn-title{margin:0}.simple-notification .sn-title{line-height:30px;font-size:20px}.simple-notification .sn-content{font-size:16px;line-height:20px}.simple-notification.has-icon .sn-content,.simple-notification.has-icon .sn-html,.simple-notification.has-icon .sn-title{padding:0 50px 0 0}.simple-notification .icon{position:absolute;box-sizing:border-box;top:0;right:0;width:70px;height:70px;padding:10px}.simple-notification .icon.icon-hover:hover{opacity:.5}.simple-notification .icon svg{fill:#fff;width:100%;height:100%}.simple-notification .icon svg g{fill:#fff}.simple-notification.rtl-mode.has-icon .sn-content,.simple-notification.rtl-mode.has-icon .sn-html,.simple-notification.rtl-mode.has-icon .sn-title{padding:0 0 0 50px}.simple-notification.rtl-mode{direction:rtl}.simple-notification.rtl-mode .sn-content{padding:0 0 0 50px}.simple-notification.rtl-mode svg{left:0;right:auto}.simple-notification.error{background:#f44336}.simple-notification.success{background:#8bc34a}.simple-notification.alert{background:#ffdb5b}.simple-notification.info{background:#03a9f4}.simple-notification.warn{background:#ffdb5b}.simple-notification .sn-progress-loader{position:absolute;top:0;left:0;width:100%;height:5px}.simple-notification .sn-progress-loader span{float:left;height:100%}.simple-notification.success .sn-progress-loader span{background:#689f38}.simple-notification.error .sn-progress-loader span{background:#d32f2f}.simple-notification.alert .sn-progress-loader span{background:#edc242}.simple-notification.info .sn-progress-loader span{background:#0288d1}.simple-notification.warn .sn-progress-loader span{background:#edc242}.simple-notification.bare .sn-progress-loader span{background:#ccc}.simple-notification.warn div .sn-content,.simple-notification.warn div .sn-html,.simple-notification.warn div .sn-title{color:#444}\"] }]\n }], function () { return [{ type: NotificationsService }, { type: ɵngcc1.DomSanitizer }, { type: ɵngcc0.ChangeDetectorRef }, { type: ɵngcc0.NgZone }]; }, { timeOut: [{\n type: Input\n }], showProgressBar: [{\n type: Input\n }], pauseOnHover: [{\n type: Input\n }], clickToClose: [{\n type: Input\n }], clickIconToClose: [{\n type: Input\n }], maxLength: [{\n type: Input\n }], theClass: [{\n type: Input\n }], rtl: [{\n type: Input\n }], animate: [{\n type: Input\n }], position: [{\n type: Input\n }], item: [{\n type: Input\n }] }); })();\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(SimpleNotificationsComponent, [{\n type: Component,\n args: [{ selector: 'simple-notifications', encapsulation: ViewEncapsulation.None, template: \"
    \\r\\n \\r\\n \\r\\n
    \", changeDetection: ChangeDetectionStrategy.OnPush, styles: [\".simple-notification-wrapper{position:fixed;width:300px;z-index:1000}.simple-notification-wrapper.left{left:20px}.simple-notification-wrapper.top{top:20px}.simple-notification-wrapper.right{right:20px}.simple-notification-wrapper.bottom{bottom:20px}.simple-notification-wrapper.center{left:50%;transform:translateX(-50%)}.simple-notification-wrapper.middle{top:50%;transform:translateY(-50%)}.simple-notification-wrapper.middle.center{transform:translate(-50%,-50%)}@media (max-width:340px){.simple-notification-wrapper{width:auto;left:20px;right:20px}}\"] }]\n }], function () { return [{ type: NotificationsService }, { type: ɵngcc0.ChangeDetectorRef }]; }, { create: [{\n type: Output\n }], destroy: [{\n type: Output\n }], options: [{\n type: Input\n }] }); })();\n(function () { (typeof ngDevMode === \"undefined\" || ngDevMode) && ɵngcc0.ɵsetClassMetadata(SimpleNotificationsModule, [{\n type: NgModule,\n args: [{\n imports: [\n CommonModule\n ],\n declarations: [\n SimpleNotificationsComponent,\n NotificationComponent\n ],\n exports: [SimpleNotificationsComponent]\n }]\n }], null, null); })();\n(function () { (typeof ngJitMode === \"undefined\" || ngJitMode) && ɵngcc0.ɵɵsetNgModuleScope(SimpleNotificationsModule, { declarations: function () { return [SimpleNotificationsComponent, NotificationComponent]; }, imports: function () { return [CommonModule]; }, exports: function () { return [SimpleNotificationsComponent]; } }); })();\n\n/**\r\n * Generated bundle index. Do not edit.\r\n */\n\nexport { NotificationAnimationType, NotificationComponent, NotificationType, NotificationsService, OPTIONS, SimpleNotificationsComponent, SimpleNotificationsModule, optionsFactory };\n\n","/*!\n * Cropper.js v1.6.2\n * https://fengyuanchen.github.io/cropperjs\n *\n * Copyright 2015-present Chen Fengyuan\n * Released under the MIT license\n *\n * Date: 2024-04-21T07:43:05.335Z\n */\n\n(function (global, factory) {\n typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n typeof define === 'function' && define.amd ? define(factory) :\n (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.Cropper = factory());\n})(this, (function () { 'use strict';\n\n function ownKeys(e, r) {\n var t = Object.keys(e);\n if (Object.getOwnPropertySymbols) {\n var o = Object.getOwnPropertySymbols(e);\n r && (o = o.filter(function (r) {\n return Object.getOwnPropertyDescriptor(e, r).enumerable;\n })), t.push.apply(t, o);\n }\n return t;\n }\n function _objectSpread2(e) {\n for (var r = 1; r < arguments.length; r++) {\n var t = null != arguments[r] ? arguments[r] : {};\n r % 2 ? ownKeys(Object(t), !0).forEach(function (r) {\n _defineProperty(e, r, t[r]);\n }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) {\n Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r));\n });\n }\n return e;\n }\n function _toPrimitive(t, r) {\n if (\"object\" != typeof t || !t) return t;\n var e = t[Symbol.toPrimitive];\n if (void 0 !== e) {\n var i = e.call(t, r || \"default\");\n if (\"object\" != typeof i) return i;\n throw new TypeError(\"@@toPrimitive must return a primitive value.\");\n }\n return (\"string\" === r ? String : Number)(t);\n }\n function _toPropertyKey(t) {\n var i = _toPrimitive(t, \"string\");\n return \"symbol\" == typeof i ? i : i + \"\";\n }\n function _typeof(o) {\n \"@babel/helpers - typeof\";\n\n return _typeof = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (o) {\n return typeof o;\n } : function (o) {\n return o && \"function\" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? \"symbol\" : typeof o;\n }, _typeof(o);\n }\n function _classCallCheck(instance, Constructor) {\n if (!(instance instanceof Constructor)) {\n throw new TypeError(\"Cannot call a class as a function\");\n }\n }\n function _defineProperties(target, props) {\n for (var i = 0; i < props.length; i++) {\n var descriptor = props[i];\n descriptor.enumerable = descriptor.enumerable || false;\n descriptor.configurable = true;\n if (\"value\" in descriptor) descriptor.writable = true;\n Object.defineProperty(target, _toPropertyKey(descriptor.key), descriptor);\n }\n }\n function _createClass(Constructor, protoProps, staticProps) {\n if (protoProps) _defineProperties(Constructor.prototype, protoProps);\n if (staticProps) _defineProperties(Constructor, staticProps);\n Object.defineProperty(Constructor, \"prototype\", {\n writable: false\n });\n return Constructor;\n }\n function _defineProperty(obj, key, value) {\n key = _toPropertyKey(key);\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 return obj;\n }\n function _toConsumableArray(arr) {\n return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread();\n }\n function _arrayWithoutHoles(arr) {\n if (Array.isArray(arr)) return _arrayLikeToArray(arr);\n }\n function _iterableToArray(iter) {\n if (typeof Symbol !== \"undefined\" && iter[Symbol.iterator] != null || iter[\"@@iterator\"] != null) return Array.from(iter);\n }\n function _unsupportedIterableToArray(o, minLen) {\n if (!o) return;\n if (typeof o === \"string\") return _arrayLikeToArray(o, minLen);\n var n = Object.prototype.toString.call(o).slice(8, -1);\n if (n === \"Object\" && o.constructor) n = o.constructor.name;\n if (n === \"Map\" || n === \"Set\") return Array.from(o);\n if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);\n }\n function _arrayLikeToArray(arr, len) {\n if (len == null || len > arr.length) len = arr.length;\n for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];\n return arr2;\n }\n function _nonIterableSpread() {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n\n var IS_BROWSER = typeof window !== 'undefined' && typeof window.document !== 'undefined';\n var WINDOW = IS_BROWSER ? window : {};\n var IS_TOUCH_DEVICE = IS_BROWSER && WINDOW.document.documentElement ? 'ontouchstart' in WINDOW.document.documentElement : false;\n var HAS_POINTER_EVENT = IS_BROWSER ? 'PointerEvent' in WINDOW : false;\n var NAMESPACE = 'cropper';\n\n // Actions\n var ACTION_ALL = 'all';\n var ACTION_CROP = 'crop';\n var ACTION_MOVE = 'move';\n var ACTION_ZOOM = 'zoom';\n var ACTION_EAST = 'e';\n var ACTION_WEST = 'w';\n var ACTION_SOUTH = 's';\n var ACTION_NORTH = 'n';\n var ACTION_NORTH_EAST = 'ne';\n var ACTION_NORTH_WEST = 'nw';\n var ACTION_SOUTH_EAST = 'se';\n var ACTION_SOUTH_WEST = 'sw';\n\n // Classes\n var CLASS_CROP = \"\".concat(NAMESPACE, \"-crop\");\n var CLASS_DISABLED = \"\".concat(NAMESPACE, \"-disabled\");\n var CLASS_HIDDEN = \"\".concat(NAMESPACE, \"-hidden\");\n var CLASS_HIDE = \"\".concat(NAMESPACE, \"-hide\");\n var CLASS_INVISIBLE = \"\".concat(NAMESPACE, \"-invisible\");\n var CLASS_MODAL = \"\".concat(NAMESPACE, \"-modal\");\n var CLASS_MOVE = \"\".concat(NAMESPACE, \"-move\");\n\n // Data keys\n var DATA_ACTION = \"\".concat(NAMESPACE, \"Action\");\n var DATA_PREVIEW = \"\".concat(NAMESPACE, \"Preview\");\n\n // Drag modes\n var DRAG_MODE_CROP = 'crop';\n var DRAG_MODE_MOVE = 'move';\n var DRAG_MODE_NONE = 'none';\n\n // Events\n var EVENT_CROP = 'crop';\n var EVENT_CROP_END = 'cropend';\n var EVENT_CROP_MOVE = 'cropmove';\n var EVENT_CROP_START = 'cropstart';\n var EVENT_DBLCLICK = 'dblclick';\n var EVENT_TOUCH_START = IS_TOUCH_DEVICE ? 'touchstart' : 'mousedown';\n var EVENT_TOUCH_MOVE = IS_TOUCH_DEVICE ? 'touchmove' : 'mousemove';\n var EVENT_TOUCH_END = IS_TOUCH_DEVICE ? 'touchend touchcancel' : 'mouseup';\n var EVENT_POINTER_DOWN = HAS_POINTER_EVENT ? 'pointerdown' : EVENT_TOUCH_START;\n var EVENT_POINTER_MOVE = HAS_POINTER_EVENT ? 'pointermove' : EVENT_TOUCH_MOVE;\n var EVENT_POINTER_UP = HAS_POINTER_EVENT ? 'pointerup pointercancel' : EVENT_TOUCH_END;\n var EVENT_READY = 'ready';\n var EVENT_RESIZE = 'resize';\n var EVENT_WHEEL = 'wheel';\n var EVENT_ZOOM = 'zoom';\n\n // Mime types\n var MIME_TYPE_JPEG = 'image/jpeg';\n\n // RegExps\n var REGEXP_ACTIONS = /^e|w|s|n|se|sw|ne|nw|all|crop|move|zoom$/;\n var REGEXP_DATA_URL = /^data:/;\n var REGEXP_DATA_URL_JPEG = /^data:image\\/jpeg;base64,/;\n var REGEXP_TAG_NAME = /^img|canvas$/i;\n\n // Misc\n // Inspired by the default width and height of a canvas element.\n var MIN_CONTAINER_WIDTH = 200;\n var MIN_CONTAINER_HEIGHT = 100;\n\n var DEFAULTS = {\n // Define the view mode of the cropper\n viewMode: 0,\n // 0, 1, 2, 3\n\n // Define the dragging mode of the cropper\n dragMode: DRAG_MODE_CROP,\n // 'crop', 'move' or 'none'\n\n // Define the initial aspect ratio of the crop box\n initialAspectRatio: NaN,\n // Define the aspect ratio of the crop box\n aspectRatio: NaN,\n // An object with the previous cropping result data\n data: null,\n // A selector for adding extra containers to preview\n preview: '',\n // Re-render the cropper when resize the window\n responsive: true,\n // Restore the cropped area after resize the window\n restore: true,\n // Check if the current image is a cross-origin image\n checkCrossOrigin: true,\n // Check the current image's Exif Orientation information\n checkOrientation: true,\n // Show the black modal\n modal: true,\n // Show the dashed lines for guiding\n guides: true,\n // Show the center indicator for guiding\n center: true,\n // Show the white modal to highlight the crop box\n highlight: true,\n // Show the grid background\n background: true,\n // Enable to crop the image automatically when initialize\n autoCrop: true,\n // Define the percentage of automatic cropping area when initializes\n autoCropArea: 0.8,\n // Enable to move the image\n movable: true,\n // Enable to rotate the image\n rotatable: true,\n // Enable to scale the image\n scalable: true,\n // Enable to zoom the image\n zoomable: true,\n // Enable to zoom the image by dragging touch\n zoomOnTouch: true,\n // Enable to zoom the image by wheeling mouse\n zoomOnWheel: true,\n // Define zoom ratio when zoom the image by wheeling mouse\n wheelZoomRatio: 0.1,\n // Enable to move the crop box\n cropBoxMovable: true,\n // Enable to resize the crop box\n cropBoxResizable: true,\n // Toggle drag mode between \"crop\" and \"move\" when click twice on the cropper\n toggleDragModeOnDblclick: true,\n // Size limitation\n minCanvasWidth: 0,\n minCanvasHeight: 0,\n minCropBoxWidth: 0,\n minCropBoxHeight: 0,\n minContainerWidth: MIN_CONTAINER_WIDTH,\n minContainerHeight: MIN_CONTAINER_HEIGHT,\n // Shortcuts of events\n ready: null,\n cropstart: null,\n cropmove: null,\n cropend: null,\n crop: null,\n zoom: null\n };\n\n var TEMPLATE = '
    ' + '
    ' + '
    ' + '
    ' + '
    ' + '
    ' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '' + '
    ' + '
    ';\n\n /**\n * Check if the given value is not a number.\n */\n var isNaN = Number.isNaN || WINDOW.isNaN;\n\n /**\n * Check if the given value is a number.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a number, else `false`.\n */\n function isNumber(value) {\n return typeof value === 'number' && !isNaN(value);\n }\n\n /**\n * Check if the given value is a positive number.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a positive number, else `false`.\n */\n var isPositiveNumber = function isPositiveNumber(value) {\n return value > 0 && value < Infinity;\n };\n\n /**\n * Check if the given value is undefined.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is undefined, else `false`.\n */\n function isUndefined(value) {\n return typeof value === 'undefined';\n }\n\n /**\n * Check if the given value is an object.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is an object, else `false`.\n */\n function isObject(value) {\n return _typeof(value) === 'object' && value !== null;\n }\n var hasOwnProperty = Object.prototype.hasOwnProperty;\n\n /**\n * Check if the given value is a plain object.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a plain object, else `false`.\n */\n function isPlainObject(value) {\n if (!isObject(value)) {\n return false;\n }\n try {\n var _constructor = value.constructor;\n var prototype = _constructor.prototype;\n return _constructor && prototype && hasOwnProperty.call(prototype, 'isPrototypeOf');\n } catch (error) {\n return false;\n }\n }\n\n /**\n * Check if the given value is a function.\n * @param {*} value - The value to check.\n * @returns {boolean} Returns `true` if the given value is a function, else `false`.\n */\n function isFunction(value) {\n return typeof value === 'function';\n }\n var slice = Array.prototype.slice;\n\n /**\n * Convert array-like or iterable object to an array.\n * @param {*} value - The value to convert.\n * @returns {Array} Returns a new array.\n */\n function toArray(value) {\n return Array.from ? Array.from(value) : slice.call(value);\n }\n\n /**\n * Iterate the given data.\n * @param {*} data - The data to iterate.\n * @param {Function} callback - The process function for each element.\n * @returns {*} The original data.\n */\n function forEach(data, callback) {\n if (data && isFunction(callback)) {\n if (Array.isArray(data) || isNumber(data.length) /* array-like */) {\n toArray(data).forEach(function (value, key) {\n callback.call(data, value, key, data);\n });\n } else if (isObject(data)) {\n Object.keys(data).forEach(function (key) {\n callback.call(data, data[key], key, data);\n });\n }\n }\n return data;\n }\n\n /**\n * Extend the given object.\n * @param {*} target - The target object to extend.\n * @param {*} args - The rest objects for merging to the target object.\n * @returns {Object} The extended object.\n */\n var assign = Object.assign || function assign(target) {\n for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {\n args[_key - 1] = arguments[_key];\n }\n if (isObject(target) && args.length > 0) {\n args.forEach(function (arg) {\n if (isObject(arg)) {\n Object.keys(arg).forEach(function (key) {\n target[key] = arg[key];\n });\n }\n });\n }\n return target;\n };\n var REGEXP_DECIMALS = /\\.\\d*(?:0|9){12}\\d*$/;\n\n /**\n * Normalize decimal number.\n * Check out {@link https://0.30000000000000004.com/}\n * @param {number} value - The value to normalize.\n * @param {number} [times=100000000000] - The times for normalizing.\n * @returns {number} Returns the normalized number.\n */\n function normalizeDecimalNumber(value) {\n var times = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 100000000000;\n return REGEXP_DECIMALS.test(value) ? Math.round(value * times) / times : value;\n }\n var REGEXP_SUFFIX = /^width|height|left|top|marginLeft|marginTop$/;\n\n /**\n * Apply styles to the given element.\n * @param {Element} element - The target element.\n * @param {Object} styles - The styles for applying.\n */\n function setStyle(element, styles) {\n var style = element.style;\n forEach(styles, function (value, property) {\n if (REGEXP_SUFFIX.test(property) && isNumber(value)) {\n value = \"\".concat(value, \"px\");\n }\n style[property] = value;\n });\n }\n\n /**\n * Check if the given element has a special class.\n * @param {Element} element - The element to check.\n * @param {string} value - The class to search.\n * @returns {boolean} Returns `true` if the special class was found.\n */\n function hasClass(element, value) {\n return element.classList ? element.classList.contains(value) : element.className.indexOf(value) > -1;\n }\n\n /**\n * Add classes to the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be added.\n */\n function addClass(element, value) {\n if (!value) {\n return;\n }\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n addClass(elem, value);\n });\n return;\n }\n if (element.classList) {\n element.classList.add(value);\n return;\n }\n var className = element.className.trim();\n if (!className) {\n element.className = value;\n } else if (className.indexOf(value) < 0) {\n element.className = \"\".concat(className, \" \").concat(value);\n }\n }\n\n /**\n * Remove classes from the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be removed.\n */\n function removeClass(element, value) {\n if (!value) {\n return;\n }\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n removeClass(elem, value);\n });\n return;\n }\n if (element.classList) {\n element.classList.remove(value);\n return;\n }\n if (element.className.indexOf(value) >= 0) {\n element.className = element.className.replace(value, '');\n }\n }\n\n /**\n * Add or remove classes from the given element.\n * @param {Element} element - The target element.\n * @param {string} value - The classes to be toggled.\n * @param {boolean} added - Add only.\n */\n function toggleClass(element, value, added) {\n if (!value) {\n return;\n }\n if (isNumber(element.length)) {\n forEach(element, function (elem) {\n toggleClass(elem, value, added);\n });\n return;\n }\n\n // IE10-11 doesn't support the second parameter of `classList.toggle`\n if (added) {\n addClass(element, value);\n } else {\n removeClass(element, value);\n }\n }\n var REGEXP_CAMEL_CASE = /([a-z\\d])([A-Z])/g;\n\n /**\n * Transform the given string from camelCase to kebab-case\n * @param {string} value - The value to transform.\n * @returns {string} The transformed value.\n */\n function toParamCase(value) {\n return value.replace(REGEXP_CAMEL_CASE, '$1-$2').toLowerCase();\n }\n\n /**\n * Get data from the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to get.\n * @returns {string} The data value.\n */\n function getData(element, name) {\n if (isObject(element[name])) {\n return element[name];\n }\n if (element.dataset) {\n return element.dataset[name];\n }\n return element.getAttribute(\"data-\".concat(toParamCase(name)));\n }\n\n /**\n * Set data to the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to set.\n * @param {string} data - The data value.\n */\n function setData(element, name, data) {\n if (isObject(data)) {\n element[name] = data;\n } else if (element.dataset) {\n element.dataset[name] = data;\n } else {\n element.setAttribute(\"data-\".concat(toParamCase(name)), data);\n }\n }\n\n /**\n * Remove data from the given element.\n * @param {Element} element - The target element.\n * @param {string} name - The data key to remove.\n */\n function removeData(element, name) {\n if (isObject(element[name])) {\n try {\n delete element[name];\n } catch (error) {\n element[name] = undefined;\n }\n } else if (element.dataset) {\n // #128 Safari not allows to delete dataset property\n try {\n delete element.dataset[name];\n } catch (error) {\n element.dataset[name] = undefined;\n }\n } else {\n element.removeAttribute(\"data-\".concat(toParamCase(name)));\n }\n }\n var REGEXP_SPACES = /\\s\\s*/;\n var onceSupported = function () {\n var supported = false;\n if (IS_BROWSER) {\n var once = false;\n var listener = function listener() {};\n var options = Object.defineProperty({}, 'once', {\n get: function get() {\n supported = true;\n return once;\n },\n /**\n * This setter can fix a `TypeError` in strict mode\n * {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Errors/Getter_only}\n * @param {boolean} value - The value to set\n */\n set: function set(value) {\n once = value;\n }\n });\n WINDOW.addEventListener('test', listener, options);\n WINDOW.removeEventListener('test', listener, options);\n }\n return supported;\n }();\n\n /**\n * Remove event listener from the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Function} listener - The event listener.\n * @param {Object} options - The event options.\n */\n function removeListener(element, type, listener) {\n var options = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {};\n var handler = listener;\n type.trim().split(REGEXP_SPACES).forEach(function (event) {\n if (!onceSupported) {\n var listeners = element.listeners;\n if (listeners && listeners[event] && listeners[event][listener]) {\n handler = listeners[event][listener];\n delete listeners[event][listener];\n if (Object.keys(listeners[event]).length === 0) {\n delete listeners[event];\n }\n if (Object.keys(listeners).length === 0) {\n delete element.listeners;\n }\n }\n }\n element.removeEventListener(event, handler, options);\n });\n }\n\n /**\n * Add event listener to the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Function} listener - The event listener.\n * @param {Object} options - The event options.\n */\n function addListener(element, type, listener) {\n var options = arguments.length > 3 && arguments[3] !== undefined ? arguments[3] : {};\n var _handler = listener;\n type.trim().split(REGEXP_SPACES).forEach(function (event) {\n if (options.once && !onceSupported) {\n var _element$listeners = element.listeners,\n listeners = _element$listeners === void 0 ? {} : _element$listeners;\n _handler = function handler() {\n delete listeners[event][listener];\n element.removeEventListener(event, _handler, options);\n for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {\n args[_key2] = arguments[_key2];\n }\n listener.apply(element, args);\n };\n if (!listeners[event]) {\n listeners[event] = {};\n }\n if (listeners[event][listener]) {\n element.removeEventListener(event, listeners[event][listener], options);\n }\n listeners[event][listener] = _handler;\n element.listeners = listeners;\n }\n element.addEventListener(event, _handler, options);\n });\n }\n\n /**\n * Dispatch event on the target element.\n * @param {Element} element - The event target.\n * @param {string} type - The event type(s).\n * @param {Object} data - The additional event data.\n * @returns {boolean} Indicate if the event is default prevented or not.\n */\n function dispatchEvent(element, type, data) {\n var event;\n\n // Event and CustomEvent on IE9-11 are global objects, not constructors\n if (isFunction(Event) && isFunction(CustomEvent)) {\n event = new CustomEvent(type, {\n detail: data,\n bubbles: true,\n cancelable: true\n });\n } else {\n event = document.createEvent('CustomEvent');\n event.initCustomEvent(type, true, true, data);\n }\n return element.dispatchEvent(event);\n }\n\n /**\n * Get the offset base on the document.\n * @param {Element} element - The target element.\n * @returns {Object} The offset data.\n */\n function getOffset(element) {\n var box = element.getBoundingClientRect();\n return {\n left: box.left + (window.pageXOffset - document.documentElement.clientLeft),\n top: box.top + (window.pageYOffset - document.documentElement.clientTop)\n };\n }\n var location = WINDOW.location;\n var REGEXP_ORIGINS = /^(\\w+:)\\/\\/([^:/?#]*):?(\\d*)/i;\n\n /**\n * Check if the given URL is a cross origin URL.\n * @param {string} url - The target URL.\n * @returns {boolean} Returns `true` if the given URL is a cross origin URL, else `false`.\n */\n function isCrossOriginURL(url) {\n var parts = url.match(REGEXP_ORIGINS);\n return parts !== null && (parts[1] !== location.protocol || parts[2] !== location.hostname || parts[3] !== location.port);\n }\n\n /**\n * Add timestamp to the given URL.\n * @param {string} url - The target URL.\n * @returns {string} The result URL.\n */\n function addTimestamp(url) {\n var timestamp = \"timestamp=\".concat(new Date().getTime());\n return url + (url.indexOf('?') === -1 ? '?' : '&') + timestamp;\n }\n\n /**\n * Get transforms base on the given object.\n * @param {Object} obj - The target object.\n * @returns {string} A string contains transform values.\n */\n function getTransforms(_ref) {\n var rotate = _ref.rotate,\n scaleX = _ref.scaleX,\n scaleY = _ref.scaleY,\n translateX = _ref.translateX,\n translateY = _ref.translateY;\n var values = [];\n if (isNumber(translateX) && translateX !== 0) {\n values.push(\"translateX(\".concat(translateX, \"px)\"));\n }\n if (isNumber(translateY) && translateY !== 0) {\n values.push(\"translateY(\".concat(translateY, \"px)\"));\n }\n\n // Rotate should come first before scale to match orientation transform\n if (isNumber(rotate) && rotate !== 0) {\n values.push(\"rotate(\".concat(rotate, \"deg)\"));\n }\n if (isNumber(scaleX) && scaleX !== 1) {\n values.push(\"scaleX(\".concat(scaleX, \")\"));\n }\n if (isNumber(scaleY) && scaleY !== 1) {\n values.push(\"scaleY(\".concat(scaleY, \")\"));\n }\n var transform = values.length ? values.join(' ') : 'none';\n return {\n WebkitTransform: transform,\n msTransform: transform,\n transform: transform\n };\n }\n\n /**\n * Get the max ratio of a group of pointers.\n * @param {string} pointers - The target pointers.\n * @returns {number} The result ratio.\n */\n function getMaxZoomRatio(pointers) {\n var pointers2 = _objectSpread2({}, pointers);\n var maxRatio = 0;\n forEach(pointers, function (pointer, pointerId) {\n delete pointers2[pointerId];\n forEach(pointers2, function (pointer2) {\n var x1 = Math.abs(pointer.startX - pointer2.startX);\n var y1 = Math.abs(pointer.startY - pointer2.startY);\n var x2 = Math.abs(pointer.endX - pointer2.endX);\n var y2 = Math.abs(pointer.endY - pointer2.endY);\n var z1 = Math.sqrt(x1 * x1 + y1 * y1);\n var z2 = Math.sqrt(x2 * x2 + y2 * y2);\n var ratio = (z2 - z1) / z1;\n if (Math.abs(ratio) > Math.abs(maxRatio)) {\n maxRatio = ratio;\n }\n });\n });\n return maxRatio;\n }\n\n /**\n * Get a pointer from an event object.\n * @param {Object} event - The target event object.\n * @param {boolean} endOnly - Indicates if only returns the end point coordinate or not.\n * @returns {Object} The result pointer contains start and/or end point coordinates.\n */\n function getPointer(_ref2, endOnly) {\n var pageX = _ref2.pageX,\n pageY = _ref2.pageY;\n var end = {\n endX: pageX,\n endY: pageY\n };\n return endOnly ? end : _objectSpread2({\n startX: pageX,\n startY: pageY\n }, end);\n }\n\n /**\n * Get the center point coordinate of a group of pointers.\n * @param {Object} pointers - The target pointers.\n * @returns {Object} The center point coordinate.\n */\n function getPointersCenter(pointers) {\n var pageX = 0;\n var pageY = 0;\n var count = 0;\n forEach(pointers, function (_ref3) {\n var startX = _ref3.startX,\n startY = _ref3.startY;\n pageX += startX;\n pageY += startY;\n count += 1;\n });\n pageX /= count;\n pageY /= count;\n return {\n pageX: pageX,\n pageY: pageY\n };\n }\n\n /**\n * Get the max sizes in a rectangle under the given aspect ratio.\n * @param {Object} data - The original sizes.\n * @param {string} [type='contain'] - The adjust type.\n * @returns {Object} The result sizes.\n */\n function getAdjustedSizes(_ref4) {\n var aspectRatio = _ref4.aspectRatio,\n height = _ref4.height,\n width = _ref4.width;\n var type = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'contain';\n var isValidWidth = isPositiveNumber(width);\n var isValidHeight = isPositiveNumber(height);\n if (isValidWidth && isValidHeight) {\n var adjustedWidth = height * aspectRatio;\n if (type === 'contain' && adjustedWidth > width || type === 'cover' && adjustedWidth < width) {\n height = width / aspectRatio;\n } else {\n width = height * aspectRatio;\n }\n } else if (isValidWidth) {\n height = width / aspectRatio;\n } else if (isValidHeight) {\n width = height * aspectRatio;\n }\n return {\n width: width,\n height: height\n };\n }\n\n /**\n * Get the new sizes of a rectangle after rotated.\n * @param {Object} data - The original sizes.\n * @returns {Object} The result sizes.\n */\n function getRotatedSizes(_ref5) {\n var width = _ref5.width,\n height = _ref5.height,\n degree = _ref5.degree;\n degree = Math.abs(degree) % 180;\n if (degree === 90) {\n return {\n width: height,\n height: width\n };\n }\n var arc = degree % 90 * Math.PI / 180;\n var sinArc = Math.sin(arc);\n var cosArc = Math.cos(arc);\n var newWidth = width * cosArc + height * sinArc;\n var newHeight = width * sinArc + height * cosArc;\n return degree > 90 ? {\n width: newHeight,\n height: newWidth\n } : {\n width: newWidth,\n height: newHeight\n };\n }\n\n /**\n * Get a canvas which drew the given image.\n * @param {HTMLImageElement} image - The image for drawing.\n * @param {Object} imageData - The image data.\n * @param {Object} canvasData - The canvas data.\n * @param {Object} options - The options.\n * @returns {HTMLCanvasElement} The result canvas.\n */\n function getSourceCanvas(image, _ref6, _ref7, _ref8) {\n var imageAspectRatio = _ref6.aspectRatio,\n imageNaturalWidth = _ref6.naturalWidth,\n imageNaturalHeight = _ref6.naturalHeight,\n _ref6$rotate = _ref6.rotate,\n rotate = _ref6$rotate === void 0 ? 0 : _ref6$rotate,\n _ref6$scaleX = _ref6.scaleX,\n scaleX = _ref6$scaleX === void 0 ? 1 : _ref6$scaleX,\n _ref6$scaleY = _ref6.scaleY,\n scaleY = _ref6$scaleY === void 0 ? 1 : _ref6$scaleY;\n var aspectRatio = _ref7.aspectRatio,\n naturalWidth = _ref7.naturalWidth,\n naturalHeight = _ref7.naturalHeight;\n var _ref8$fillColor = _ref8.fillColor,\n fillColor = _ref8$fillColor === void 0 ? 'transparent' : _ref8$fillColor,\n _ref8$imageSmoothingE = _ref8.imageSmoothingEnabled,\n imageSmoothingEnabled = _ref8$imageSmoothingE === void 0 ? true : _ref8$imageSmoothingE,\n _ref8$imageSmoothingQ = _ref8.imageSmoothingQuality,\n imageSmoothingQuality = _ref8$imageSmoothingQ === void 0 ? 'low' : _ref8$imageSmoothingQ,\n _ref8$maxWidth = _ref8.maxWidth,\n maxWidth = _ref8$maxWidth === void 0 ? Infinity : _ref8$maxWidth,\n _ref8$maxHeight = _ref8.maxHeight,\n maxHeight = _ref8$maxHeight === void 0 ? Infinity : _ref8$maxHeight,\n _ref8$minWidth = _ref8.minWidth,\n minWidth = _ref8$minWidth === void 0 ? 0 : _ref8$minWidth,\n _ref8$minHeight = _ref8.minHeight,\n minHeight = _ref8$minHeight === void 0 ? 0 : _ref8$minHeight;\n var canvas = document.createElement('canvas');\n var context = canvas.getContext('2d');\n var maxSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: maxWidth,\n height: maxHeight\n });\n var minSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: minWidth,\n height: minHeight\n }, 'cover');\n var width = Math.min(maxSizes.width, Math.max(minSizes.width, naturalWidth));\n var height = Math.min(maxSizes.height, Math.max(minSizes.height, naturalHeight));\n\n // Note: should always use image's natural sizes for drawing as\n // imageData.naturalWidth === canvasData.naturalHeight when rotate % 180 === 90\n var destMaxSizes = getAdjustedSizes({\n aspectRatio: imageAspectRatio,\n width: maxWidth,\n height: maxHeight\n });\n var destMinSizes = getAdjustedSizes({\n aspectRatio: imageAspectRatio,\n width: minWidth,\n height: minHeight\n }, 'cover');\n var destWidth = Math.min(destMaxSizes.width, Math.max(destMinSizes.width, imageNaturalWidth));\n var destHeight = Math.min(destMaxSizes.height, Math.max(destMinSizes.height, imageNaturalHeight));\n var params = [-destWidth / 2, -destHeight / 2, destWidth, destHeight];\n canvas.width = normalizeDecimalNumber(width);\n canvas.height = normalizeDecimalNumber(height);\n context.fillStyle = fillColor;\n context.fillRect(0, 0, width, height);\n context.save();\n context.translate(width / 2, height / 2);\n context.rotate(rotate * Math.PI / 180);\n context.scale(scaleX, scaleY);\n context.imageSmoothingEnabled = imageSmoothingEnabled;\n context.imageSmoothingQuality = imageSmoothingQuality;\n context.drawImage.apply(context, [image].concat(_toConsumableArray(params.map(function (param) {\n return Math.floor(normalizeDecimalNumber(param));\n }))));\n context.restore();\n return canvas;\n }\n var fromCharCode = String.fromCharCode;\n\n /**\n * Get string from char code in data view.\n * @param {DataView} dataView - The data view for read.\n * @param {number} start - The start index.\n * @param {number} length - The read length.\n * @returns {string} The read result.\n */\n function getStringFromCharCode(dataView, start, length) {\n var str = '';\n length += start;\n for (var i = start; i < length; i += 1) {\n str += fromCharCode(dataView.getUint8(i));\n }\n return str;\n }\n var REGEXP_DATA_URL_HEAD = /^data:.*,/;\n\n /**\n * Transform Data URL to array buffer.\n * @param {string} dataURL - The Data URL to transform.\n * @returns {ArrayBuffer} The result array buffer.\n */\n function dataURLToArrayBuffer(dataURL) {\n var base64 = dataURL.replace(REGEXP_DATA_URL_HEAD, '');\n var binary = atob(base64);\n var arrayBuffer = new ArrayBuffer(binary.length);\n var uint8 = new Uint8Array(arrayBuffer);\n forEach(uint8, function (value, i) {\n uint8[i] = binary.charCodeAt(i);\n });\n return arrayBuffer;\n }\n\n /**\n * Transform array buffer to Data URL.\n * @param {ArrayBuffer} arrayBuffer - The array buffer to transform.\n * @param {string} mimeType - The mime type of the Data URL.\n * @returns {string} The result Data URL.\n */\n function arrayBufferToDataURL(arrayBuffer, mimeType) {\n var chunks = [];\n\n // Chunk Typed Array for better performance (#435)\n var chunkSize = 8192;\n var uint8 = new Uint8Array(arrayBuffer);\n while (uint8.length > 0) {\n // XXX: Babel's `toConsumableArray` helper will throw error in IE or Safari 9\n // eslint-disable-next-line prefer-spread\n chunks.push(fromCharCode.apply(null, toArray(uint8.subarray(0, chunkSize))));\n uint8 = uint8.subarray(chunkSize);\n }\n return \"data:\".concat(mimeType, \";base64,\").concat(btoa(chunks.join('')));\n }\n\n /**\n * Get orientation value from given array buffer.\n * @param {ArrayBuffer} arrayBuffer - The array buffer to read.\n * @returns {number} The read orientation value.\n */\n function resetAndGetOrientation(arrayBuffer) {\n var dataView = new DataView(arrayBuffer);\n var orientation;\n\n // Ignores range error when the image does not have correct Exif information\n try {\n var littleEndian;\n var app1Start;\n var ifdStart;\n\n // Only handle JPEG image (start by 0xFFD8)\n if (dataView.getUint8(0) === 0xFF && dataView.getUint8(1) === 0xD8) {\n var length = dataView.byteLength;\n var offset = 2;\n while (offset + 1 < length) {\n if (dataView.getUint8(offset) === 0xFF && dataView.getUint8(offset + 1) === 0xE1) {\n app1Start = offset;\n break;\n }\n offset += 1;\n }\n }\n if (app1Start) {\n var exifIDCode = app1Start + 4;\n var tiffOffset = app1Start + 10;\n if (getStringFromCharCode(dataView, exifIDCode, 4) === 'Exif') {\n var endianness = dataView.getUint16(tiffOffset);\n littleEndian = endianness === 0x4949;\n if (littleEndian || endianness === 0x4D4D /* bigEndian */) {\n if (dataView.getUint16(tiffOffset + 2, littleEndian) === 0x002A) {\n var firstIFDOffset = dataView.getUint32(tiffOffset + 4, littleEndian);\n if (firstIFDOffset >= 0x00000008) {\n ifdStart = tiffOffset + firstIFDOffset;\n }\n }\n }\n }\n }\n if (ifdStart) {\n var _length = dataView.getUint16(ifdStart, littleEndian);\n var _offset;\n var i;\n for (i = 0; i < _length; i += 1) {\n _offset = ifdStart + i * 12 + 2;\n if (dataView.getUint16(_offset, littleEndian) === 0x0112 /* Orientation */) {\n // 8 is the offset of the current tag's value\n _offset += 8;\n\n // Get the original orientation value\n orientation = dataView.getUint16(_offset, littleEndian);\n\n // Override the orientation with its default value\n dataView.setUint16(_offset, 1, littleEndian);\n break;\n }\n }\n }\n } catch (error) {\n orientation = 1;\n }\n return orientation;\n }\n\n /**\n * Parse Exif Orientation value.\n * @param {number} orientation - The orientation to parse.\n * @returns {Object} The parsed result.\n */\n function parseOrientation(orientation) {\n var rotate = 0;\n var scaleX = 1;\n var scaleY = 1;\n switch (orientation) {\n // Flip horizontal\n case 2:\n scaleX = -1;\n break;\n\n // Rotate left 180°\n case 3:\n rotate = -180;\n break;\n\n // Flip vertical\n case 4:\n scaleY = -1;\n break;\n\n // Flip vertical and rotate right 90°\n case 5:\n rotate = 90;\n scaleY = -1;\n break;\n\n // Rotate right 90°\n case 6:\n rotate = 90;\n break;\n\n // Flip horizontal and rotate right 90°\n case 7:\n rotate = 90;\n scaleX = -1;\n break;\n\n // Rotate left 90°\n case 8:\n rotate = -90;\n break;\n }\n return {\n rotate: rotate,\n scaleX: scaleX,\n scaleY: scaleY\n };\n }\n\n var render = {\n render: function render() {\n this.initContainer();\n this.initCanvas();\n this.initCropBox();\n this.renderCanvas();\n if (this.cropped) {\n this.renderCropBox();\n }\n },\n initContainer: function initContainer() {\n var element = this.element,\n options = this.options,\n container = this.container,\n cropper = this.cropper;\n var minWidth = Number(options.minContainerWidth);\n var minHeight = Number(options.minContainerHeight);\n addClass(cropper, CLASS_HIDDEN);\n removeClass(element, CLASS_HIDDEN);\n var containerData = {\n width: Math.max(container.offsetWidth, minWidth >= 0 ? minWidth : MIN_CONTAINER_WIDTH),\n height: Math.max(container.offsetHeight, minHeight >= 0 ? minHeight : MIN_CONTAINER_HEIGHT)\n };\n this.containerData = containerData;\n setStyle(cropper, {\n width: containerData.width,\n height: containerData.height\n });\n addClass(element, CLASS_HIDDEN);\n removeClass(cropper, CLASS_HIDDEN);\n },\n // Canvas (image wrapper)\n initCanvas: function initCanvas() {\n var containerData = this.containerData,\n imageData = this.imageData;\n var viewMode = this.options.viewMode;\n var rotated = Math.abs(imageData.rotate) % 180 === 90;\n var naturalWidth = rotated ? imageData.naturalHeight : imageData.naturalWidth;\n var naturalHeight = rotated ? imageData.naturalWidth : imageData.naturalHeight;\n var aspectRatio = naturalWidth / naturalHeight;\n var canvasWidth = containerData.width;\n var canvasHeight = containerData.height;\n if (containerData.height * aspectRatio > containerData.width) {\n if (viewMode === 3) {\n canvasWidth = containerData.height * aspectRatio;\n } else {\n canvasHeight = containerData.width / aspectRatio;\n }\n } else if (viewMode === 3) {\n canvasHeight = containerData.width / aspectRatio;\n } else {\n canvasWidth = containerData.height * aspectRatio;\n }\n var canvasData = {\n aspectRatio: aspectRatio,\n naturalWidth: naturalWidth,\n naturalHeight: naturalHeight,\n width: canvasWidth,\n height: canvasHeight\n };\n this.canvasData = canvasData;\n this.limited = viewMode === 1 || viewMode === 2;\n this.limitCanvas(true, true);\n canvasData.width = Math.min(Math.max(canvasData.width, canvasData.minWidth), canvasData.maxWidth);\n canvasData.height = Math.min(Math.max(canvasData.height, canvasData.minHeight), canvasData.maxHeight);\n canvasData.left = (containerData.width - canvasData.width) / 2;\n canvasData.top = (containerData.height - canvasData.height) / 2;\n canvasData.oldLeft = canvasData.left;\n canvasData.oldTop = canvasData.top;\n this.initialCanvasData = assign({}, canvasData);\n },\n limitCanvas: function limitCanvas(sizeLimited, positionLimited) {\n var options = this.options,\n containerData = this.containerData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var viewMode = options.viewMode;\n var aspectRatio = canvasData.aspectRatio;\n var cropped = this.cropped && cropBoxData;\n if (sizeLimited) {\n var minCanvasWidth = Number(options.minCanvasWidth) || 0;\n var minCanvasHeight = Number(options.minCanvasHeight) || 0;\n if (viewMode > 1) {\n minCanvasWidth = Math.max(minCanvasWidth, containerData.width);\n minCanvasHeight = Math.max(minCanvasHeight, containerData.height);\n if (viewMode === 3) {\n if (minCanvasHeight * aspectRatio > minCanvasWidth) {\n minCanvasWidth = minCanvasHeight * aspectRatio;\n } else {\n minCanvasHeight = minCanvasWidth / aspectRatio;\n }\n }\n } else if (viewMode > 0) {\n if (minCanvasWidth) {\n minCanvasWidth = Math.max(minCanvasWidth, cropped ? cropBoxData.width : 0);\n } else if (minCanvasHeight) {\n minCanvasHeight = Math.max(minCanvasHeight, cropped ? cropBoxData.height : 0);\n } else if (cropped) {\n minCanvasWidth = cropBoxData.width;\n minCanvasHeight = cropBoxData.height;\n if (minCanvasHeight * aspectRatio > minCanvasWidth) {\n minCanvasWidth = minCanvasHeight * aspectRatio;\n } else {\n minCanvasHeight = minCanvasWidth / aspectRatio;\n }\n }\n }\n var _getAdjustedSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: minCanvasWidth,\n height: minCanvasHeight\n });\n minCanvasWidth = _getAdjustedSizes.width;\n minCanvasHeight = _getAdjustedSizes.height;\n canvasData.minWidth = minCanvasWidth;\n canvasData.minHeight = minCanvasHeight;\n canvasData.maxWidth = Infinity;\n canvasData.maxHeight = Infinity;\n }\n if (positionLimited) {\n if (viewMode > (cropped ? 0 : 1)) {\n var newCanvasLeft = containerData.width - canvasData.width;\n var newCanvasTop = containerData.height - canvasData.height;\n canvasData.minLeft = Math.min(0, newCanvasLeft);\n canvasData.minTop = Math.min(0, newCanvasTop);\n canvasData.maxLeft = Math.max(0, newCanvasLeft);\n canvasData.maxTop = Math.max(0, newCanvasTop);\n if (cropped && this.limited) {\n canvasData.minLeft = Math.min(cropBoxData.left, cropBoxData.left + (cropBoxData.width - canvasData.width));\n canvasData.minTop = Math.min(cropBoxData.top, cropBoxData.top + (cropBoxData.height - canvasData.height));\n canvasData.maxLeft = cropBoxData.left;\n canvasData.maxTop = cropBoxData.top;\n if (viewMode === 2) {\n if (canvasData.width >= containerData.width) {\n canvasData.minLeft = Math.min(0, newCanvasLeft);\n canvasData.maxLeft = Math.max(0, newCanvasLeft);\n }\n if (canvasData.height >= containerData.height) {\n canvasData.minTop = Math.min(0, newCanvasTop);\n canvasData.maxTop = Math.max(0, newCanvasTop);\n }\n }\n }\n } else {\n canvasData.minLeft = -canvasData.width;\n canvasData.minTop = -canvasData.height;\n canvasData.maxLeft = containerData.width;\n canvasData.maxTop = containerData.height;\n }\n }\n },\n renderCanvas: function renderCanvas(changed, transformed) {\n var canvasData = this.canvasData,\n imageData = this.imageData;\n if (transformed) {\n var _getRotatedSizes = getRotatedSizes({\n width: imageData.naturalWidth * Math.abs(imageData.scaleX || 1),\n height: imageData.naturalHeight * Math.abs(imageData.scaleY || 1),\n degree: imageData.rotate || 0\n }),\n naturalWidth = _getRotatedSizes.width,\n naturalHeight = _getRotatedSizes.height;\n var width = canvasData.width * (naturalWidth / canvasData.naturalWidth);\n var height = canvasData.height * (naturalHeight / canvasData.naturalHeight);\n canvasData.left -= (width - canvasData.width) / 2;\n canvasData.top -= (height - canvasData.height) / 2;\n canvasData.width = width;\n canvasData.height = height;\n canvasData.aspectRatio = naturalWidth / naturalHeight;\n canvasData.naturalWidth = naturalWidth;\n canvasData.naturalHeight = naturalHeight;\n this.limitCanvas(true, false);\n }\n if (canvasData.width > canvasData.maxWidth || canvasData.width < canvasData.minWidth) {\n canvasData.left = canvasData.oldLeft;\n }\n if (canvasData.height > canvasData.maxHeight || canvasData.height < canvasData.minHeight) {\n canvasData.top = canvasData.oldTop;\n }\n canvasData.width = Math.min(Math.max(canvasData.width, canvasData.minWidth), canvasData.maxWidth);\n canvasData.height = Math.min(Math.max(canvasData.height, canvasData.minHeight), canvasData.maxHeight);\n this.limitCanvas(false, true);\n canvasData.left = Math.min(Math.max(canvasData.left, canvasData.minLeft), canvasData.maxLeft);\n canvasData.top = Math.min(Math.max(canvasData.top, canvasData.minTop), canvasData.maxTop);\n canvasData.oldLeft = canvasData.left;\n canvasData.oldTop = canvasData.top;\n setStyle(this.canvas, assign({\n width: canvasData.width,\n height: canvasData.height\n }, getTransforms({\n translateX: canvasData.left,\n translateY: canvasData.top\n })));\n this.renderImage(changed);\n if (this.cropped && this.limited) {\n this.limitCropBox(true, true);\n }\n },\n renderImage: function renderImage(changed) {\n var canvasData = this.canvasData,\n imageData = this.imageData;\n var width = imageData.naturalWidth * (canvasData.width / canvasData.naturalWidth);\n var height = imageData.naturalHeight * (canvasData.height / canvasData.naturalHeight);\n assign(imageData, {\n width: width,\n height: height,\n left: (canvasData.width - width) / 2,\n top: (canvasData.height - height) / 2\n });\n setStyle(this.image, assign({\n width: imageData.width,\n height: imageData.height\n }, getTransforms(assign({\n translateX: imageData.left,\n translateY: imageData.top\n }, imageData))));\n if (changed) {\n this.output();\n }\n },\n initCropBox: function initCropBox() {\n var options = this.options,\n canvasData = this.canvasData;\n var aspectRatio = options.aspectRatio || options.initialAspectRatio;\n var autoCropArea = Number(options.autoCropArea) || 0.8;\n var cropBoxData = {\n width: canvasData.width,\n height: canvasData.height\n };\n if (aspectRatio) {\n if (canvasData.height * aspectRatio > canvasData.width) {\n cropBoxData.height = cropBoxData.width / aspectRatio;\n } else {\n cropBoxData.width = cropBoxData.height * aspectRatio;\n }\n }\n this.cropBoxData = cropBoxData;\n this.limitCropBox(true, true);\n\n // Initialize auto crop area\n cropBoxData.width = Math.min(Math.max(cropBoxData.width, cropBoxData.minWidth), cropBoxData.maxWidth);\n cropBoxData.height = Math.min(Math.max(cropBoxData.height, cropBoxData.minHeight), cropBoxData.maxHeight);\n\n // The width/height of auto crop area must large than \"minWidth/Height\"\n cropBoxData.width = Math.max(cropBoxData.minWidth, cropBoxData.width * autoCropArea);\n cropBoxData.height = Math.max(cropBoxData.minHeight, cropBoxData.height * autoCropArea);\n cropBoxData.left = canvasData.left + (canvasData.width - cropBoxData.width) / 2;\n cropBoxData.top = canvasData.top + (canvasData.height - cropBoxData.height) / 2;\n cropBoxData.oldLeft = cropBoxData.left;\n cropBoxData.oldTop = cropBoxData.top;\n this.initialCropBoxData = assign({}, cropBoxData);\n },\n limitCropBox: function limitCropBox(sizeLimited, positionLimited) {\n var options = this.options,\n containerData = this.containerData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData,\n limited = this.limited;\n var aspectRatio = options.aspectRatio;\n if (sizeLimited) {\n var minCropBoxWidth = Number(options.minCropBoxWidth) || 0;\n var minCropBoxHeight = Number(options.minCropBoxHeight) || 0;\n var maxCropBoxWidth = limited ? Math.min(containerData.width, canvasData.width, canvasData.width + canvasData.left, containerData.width - canvasData.left) : containerData.width;\n var maxCropBoxHeight = limited ? Math.min(containerData.height, canvasData.height, canvasData.height + canvasData.top, containerData.height - canvasData.top) : containerData.height;\n\n // The min/maxCropBoxWidth/Height must be less than container's width/height\n minCropBoxWidth = Math.min(minCropBoxWidth, containerData.width);\n minCropBoxHeight = Math.min(minCropBoxHeight, containerData.height);\n if (aspectRatio) {\n if (minCropBoxWidth && minCropBoxHeight) {\n if (minCropBoxHeight * aspectRatio > minCropBoxWidth) {\n minCropBoxHeight = minCropBoxWidth / aspectRatio;\n } else {\n minCropBoxWidth = minCropBoxHeight * aspectRatio;\n }\n } else if (minCropBoxWidth) {\n minCropBoxHeight = minCropBoxWidth / aspectRatio;\n } else if (minCropBoxHeight) {\n minCropBoxWidth = minCropBoxHeight * aspectRatio;\n }\n if (maxCropBoxHeight * aspectRatio > maxCropBoxWidth) {\n maxCropBoxHeight = maxCropBoxWidth / aspectRatio;\n } else {\n maxCropBoxWidth = maxCropBoxHeight * aspectRatio;\n }\n }\n\n // The minWidth/Height must be less than maxWidth/Height\n cropBoxData.minWidth = Math.min(minCropBoxWidth, maxCropBoxWidth);\n cropBoxData.minHeight = Math.min(minCropBoxHeight, maxCropBoxHeight);\n cropBoxData.maxWidth = maxCropBoxWidth;\n cropBoxData.maxHeight = maxCropBoxHeight;\n }\n if (positionLimited) {\n if (limited) {\n cropBoxData.minLeft = Math.max(0, canvasData.left);\n cropBoxData.minTop = Math.max(0, canvasData.top);\n cropBoxData.maxLeft = Math.min(containerData.width, canvasData.left + canvasData.width) - cropBoxData.width;\n cropBoxData.maxTop = Math.min(containerData.height, canvasData.top + canvasData.height) - cropBoxData.height;\n } else {\n cropBoxData.minLeft = 0;\n cropBoxData.minTop = 0;\n cropBoxData.maxLeft = containerData.width - cropBoxData.width;\n cropBoxData.maxTop = containerData.height - cropBoxData.height;\n }\n }\n },\n renderCropBox: function renderCropBox() {\n var options = this.options,\n containerData = this.containerData,\n cropBoxData = this.cropBoxData;\n if (cropBoxData.width > cropBoxData.maxWidth || cropBoxData.width < cropBoxData.minWidth) {\n cropBoxData.left = cropBoxData.oldLeft;\n }\n if (cropBoxData.height > cropBoxData.maxHeight || cropBoxData.height < cropBoxData.minHeight) {\n cropBoxData.top = cropBoxData.oldTop;\n }\n cropBoxData.width = Math.min(Math.max(cropBoxData.width, cropBoxData.minWidth), cropBoxData.maxWidth);\n cropBoxData.height = Math.min(Math.max(cropBoxData.height, cropBoxData.minHeight), cropBoxData.maxHeight);\n this.limitCropBox(false, true);\n cropBoxData.left = Math.min(Math.max(cropBoxData.left, cropBoxData.minLeft), cropBoxData.maxLeft);\n cropBoxData.top = Math.min(Math.max(cropBoxData.top, cropBoxData.minTop), cropBoxData.maxTop);\n cropBoxData.oldLeft = cropBoxData.left;\n cropBoxData.oldTop = cropBoxData.top;\n if (options.movable && options.cropBoxMovable) {\n // Turn to move the canvas when the crop box is equal to the container\n setData(this.face, DATA_ACTION, cropBoxData.width >= containerData.width && cropBoxData.height >= containerData.height ? ACTION_MOVE : ACTION_ALL);\n }\n setStyle(this.cropBox, assign({\n width: cropBoxData.width,\n height: cropBoxData.height\n }, getTransforms({\n translateX: cropBoxData.left,\n translateY: cropBoxData.top\n })));\n if (this.cropped && this.limited) {\n this.limitCanvas(true, true);\n }\n if (!this.disabled) {\n this.output();\n }\n },\n output: function output() {\n this.preview();\n dispatchEvent(this.element, EVENT_CROP, this.getData());\n }\n };\n\n var preview = {\n initPreview: function initPreview() {\n var element = this.element,\n crossOrigin = this.crossOrigin;\n var preview = this.options.preview;\n var url = crossOrigin ? this.crossOriginUrl : this.url;\n var alt = element.alt || 'The image to preview';\n var image = document.createElement('img');\n if (crossOrigin) {\n image.crossOrigin = crossOrigin;\n }\n image.src = url;\n image.alt = alt;\n this.viewBox.appendChild(image);\n this.viewBoxImage = image;\n if (!preview) {\n return;\n }\n var previews = preview;\n if (typeof preview === 'string') {\n previews = element.ownerDocument.querySelectorAll(preview);\n } else if (preview.querySelector) {\n previews = [preview];\n }\n this.previews = previews;\n forEach(previews, function (el) {\n var img = document.createElement('img');\n\n // Save the original size for recover\n setData(el, DATA_PREVIEW, {\n width: el.offsetWidth,\n height: el.offsetHeight,\n html: el.innerHTML\n });\n if (crossOrigin) {\n img.crossOrigin = crossOrigin;\n }\n img.src = url;\n img.alt = alt;\n\n /**\n * Override img element styles\n * Add `display:block` to avoid margin top issue\n * Add `height:auto` to override `height` attribute on IE8\n * (Occur only when margin-top <= -height)\n */\n img.style.cssText = 'display:block;' + 'width:100%;' + 'height:auto;' + 'min-width:0!important;' + 'min-height:0!important;' + 'max-width:none!important;' + 'max-height:none!important;' + 'image-orientation:0deg!important;\"';\n el.innerHTML = '';\n el.appendChild(img);\n });\n },\n resetPreview: function resetPreview() {\n forEach(this.previews, function (element) {\n var data = getData(element, DATA_PREVIEW);\n setStyle(element, {\n width: data.width,\n height: data.height\n });\n element.innerHTML = data.html;\n removeData(element, DATA_PREVIEW);\n });\n },\n preview: function preview() {\n var imageData = this.imageData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var cropBoxWidth = cropBoxData.width,\n cropBoxHeight = cropBoxData.height;\n var width = imageData.width,\n height = imageData.height;\n var left = cropBoxData.left - canvasData.left - imageData.left;\n var top = cropBoxData.top - canvasData.top - imageData.top;\n if (!this.cropped || this.disabled) {\n return;\n }\n setStyle(this.viewBoxImage, assign({\n width: width,\n height: height\n }, getTransforms(assign({\n translateX: -left,\n translateY: -top\n }, imageData))));\n forEach(this.previews, function (element) {\n var data = getData(element, DATA_PREVIEW);\n var originalWidth = data.width;\n var originalHeight = data.height;\n var newWidth = originalWidth;\n var newHeight = originalHeight;\n var ratio = 1;\n if (cropBoxWidth) {\n ratio = originalWidth / cropBoxWidth;\n newHeight = cropBoxHeight * ratio;\n }\n if (cropBoxHeight && newHeight > originalHeight) {\n ratio = originalHeight / cropBoxHeight;\n newWidth = cropBoxWidth * ratio;\n newHeight = originalHeight;\n }\n setStyle(element, {\n width: newWidth,\n height: newHeight\n });\n setStyle(element.getElementsByTagName('img')[0], assign({\n width: width * ratio,\n height: height * ratio\n }, getTransforms(assign({\n translateX: -left * ratio,\n translateY: -top * ratio\n }, imageData))));\n });\n }\n };\n\n var events = {\n bind: function bind() {\n var element = this.element,\n options = this.options,\n cropper = this.cropper;\n if (isFunction(options.cropstart)) {\n addListener(element, EVENT_CROP_START, options.cropstart);\n }\n if (isFunction(options.cropmove)) {\n addListener(element, EVENT_CROP_MOVE, options.cropmove);\n }\n if (isFunction(options.cropend)) {\n addListener(element, EVENT_CROP_END, options.cropend);\n }\n if (isFunction(options.crop)) {\n addListener(element, EVENT_CROP, options.crop);\n }\n if (isFunction(options.zoom)) {\n addListener(element, EVENT_ZOOM, options.zoom);\n }\n addListener(cropper, EVENT_POINTER_DOWN, this.onCropStart = this.cropStart.bind(this));\n if (options.zoomable && options.zoomOnWheel) {\n addListener(cropper, EVENT_WHEEL, this.onWheel = this.wheel.bind(this), {\n passive: false,\n capture: true\n });\n }\n if (options.toggleDragModeOnDblclick) {\n addListener(cropper, EVENT_DBLCLICK, this.onDblclick = this.dblclick.bind(this));\n }\n addListener(element.ownerDocument, EVENT_POINTER_MOVE, this.onCropMove = this.cropMove.bind(this));\n addListener(element.ownerDocument, EVENT_POINTER_UP, this.onCropEnd = this.cropEnd.bind(this));\n if (options.responsive) {\n addListener(window, EVENT_RESIZE, this.onResize = this.resize.bind(this));\n }\n },\n unbind: function unbind() {\n var element = this.element,\n options = this.options,\n cropper = this.cropper;\n if (isFunction(options.cropstart)) {\n removeListener(element, EVENT_CROP_START, options.cropstart);\n }\n if (isFunction(options.cropmove)) {\n removeListener(element, EVENT_CROP_MOVE, options.cropmove);\n }\n if (isFunction(options.cropend)) {\n removeListener(element, EVENT_CROP_END, options.cropend);\n }\n if (isFunction(options.crop)) {\n removeListener(element, EVENT_CROP, options.crop);\n }\n if (isFunction(options.zoom)) {\n removeListener(element, EVENT_ZOOM, options.zoom);\n }\n removeListener(cropper, EVENT_POINTER_DOWN, this.onCropStart);\n if (options.zoomable && options.zoomOnWheel) {\n removeListener(cropper, EVENT_WHEEL, this.onWheel, {\n passive: false,\n capture: true\n });\n }\n if (options.toggleDragModeOnDblclick) {\n removeListener(cropper, EVENT_DBLCLICK, this.onDblclick);\n }\n removeListener(element.ownerDocument, EVENT_POINTER_MOVE, this.onCropMove);\n removeListener(element.ownerDocument, EVENT_POINTER_UP, this.onCropEnd);\n if (options.responsive) {\n removeListener(window, EVENT_RESIZE, this.onResize);\n }\n }\n };\n\n var handlers = {\n resize: function resize() {\n if (this.disabled) {\n return;\n }\n var options = this.options,\n container = this.container,\n containerData = this.containerData;\n var ratioX = container.offsetWidth / containerData.width;\n var ratioY = container.offsetHeight / containerData.height;\n var ratio = Math.abs(ratioX - 1) > Math.abs(ratioY - 1) ? ratioX : ratioY;\n\n // Resize when width changed or height changed\n if (ratio !== 1) {\n var canvasData;\n var cropBoxData;\n if (options.restore) {\n canvasData = this.getCanvasData();\n cropBoxData = this.getCropBoxData();\n }\n this.render();\n if (options.restore) {\n this.setCanvasData(forEach(canvasData, function (n, i) {\n canvasData[i] = n * ratio;\n }));\n this.setCropBoxData(forEach(cropBoxData, function (n, i) {\n cropBoxData[i] = n * ratio;\n }));\n }\n }\n },\n dblclick: function dblclick() {\n if (this.disabled || this.options.dragMode === DRAG_MODE_NONE) {\n return;\n }\n this.setDragMode(hasClass(this.dragBox, CLASS_CROP) ? DRAG_MODE_MOVE : DRAG_MODE_CROP);\n },\n wheel: function wheel(event) {\n var _this = this;\n var ratio = Number(this.options.wheelZoomRatio) || 0.1;\n var delta = 1;\n if (this.disabled) {\n return;\n }\n event.preventDefault();\n\n // Limit wheel speed to prevent zoom too fast (#21)\n if (this.wheeling) {\n return;\n }\n this.wheeling = true;\n setTimeout(function () {\n _this.wheeling = false;\n }, 50);\n if (event.deltaY) {\n delta = event.deltaY > 0 ? 1 : -1;\n } else if (event.wheelDelta) {\n delta = -event.wheelDelta / 120;\n } else if (event.detail) {\n delta = event.detail > 0 ? 1 : -1;\n }\n this.zoom(-delta * ratio, event);\n },\n cropStart: function cropStart(event) {\n var buttons = event.buttons,\n button = event.button;\n if (this.disabled\n\n // Handle mouse event and pointer event and ignore touch event\n || (event.type === 'mousedown' || event.type === 'pointerdown' && event.pointerType === 'mouse') && (\n // No primary button (Usually the left button)\n isNumber(buttons) && buttons !== 1 || isNumber(button) && button !== 0\n\n // Open context menu\n || event.ctrlKey)) {\n return;\n }\n var options = this.options,\n pointers = this.pointers;\n var action;\n if (event.changedTouches) {\n // Handle touch event\n forEach(event.changedTouches, function (touch) {\n pointers[touch.identifier] = getPointer(touch);\n });\n } else {\n // Handle mouse event and pointer event\n pointers[event.pointerId || 0] = getPointer(event);\n }\n if (Object.keys(pointers).length > 1 && options.zoomable && options.zoomOnTouch) {\n action = ACTION_ZOOM;\n } else {\n action = getData(event.target, DATA_ACTION);\n }\n if (!REGEXP_ACTIONS.test(action)) {\n return;\n }\n if (dispatchEvent(this.element, EVENT_CROP_START, {\n originalEvent: event,\n action: action\n }) === false) {\n return;\n }\n\n // This line is required for preventing page zooming in iOS browsers\n event.preventDefault();\n this.action = action;\n this.cropping = false;\n if (action === ACTION_CROP) {\n this.cropping = true;\n addClass(this.dragBox, CLASS_MODAL);\n }\n },\n cropMove: function cropMove(event) {\n var action = this.action;\n if (this.disabled || !action) {\n return;\n }\n var pointers = this.pointers;\n event.preventDefault();\n if (dispatchEvent(this.element, EVENT_CROP_MOVE, {\n originalEvent: event,\n action: action\n }) === false) {\n return;\n }\n if (event.changedTouches) {\n forEach(event.changedTouches, function (touch) {\n // The first parameter should not be undefined (#432)\n assign(pointers[touch.identifier] || {}, getPointer(touch, true));\n });\n } else {\n assign(pointers[event.pointerId || 0] || {}, getPointer(event, true));\n }\n this.change(event);\n },\n cropEnd: function cropEnd(event) {\n if (this.disabled) {\n return;\n }\n var action = this.action,\n pointers = this.pointers;\n if (event.changedTouches) {\n forEach(event.changedTouches, function (touch) {\n delete pointers[touch.identifier];\n });\n } else {\n delete pointers[event.pointerId || 0];\n }\n if (!action) {\n return;\n }\n event.preventDefault();\n if (!Object.keys(pointers).length) {\n this.action = '';\n }\n if (this.cropping) {\n this.cropping = false;\n toggleClass(this.dragBox, CLASS_MODAL, this.cropped && this.options.modal);\n }\n dispatchEvent(this.element, EVENT_CROP_END, {\n originalEvent: event,\n action: action\n });\n }\n };\n\n var change = {\n change: function change(event) {\n var options = this.options,\n canvasData = this.canvasData,\n containerData = this.containerData,\n cropBoxData = this.cropBoxData,\n pointers = this.pointers;\n var action = this.action;\n var aspectRatio = options.aspectRatio;\n var left = cropBoxData.left,\n top = cropBoxData.top,\n width = cropBoxData.width,\n height = cropBoxData.height;\n var right = left + width;\n var bottom = top + height;\n var minLeft = 0;\n var minTop = 0;\n var maxWidth = containerData.width;\n var maxHeight = containerData.height;\n var renderable = true;\n var offset;\n\n // Locking aspect ratio in \"free mode\" by holding shift key\n if (!aspectRatio && event.shiftKey) {\n aspectRatio = width && height ? width / height : 1;\n }\n if (this.limited) {\n minLeft = cropBoxData.minLeft;\n minTop = cropBoxData.minTop;\n maxWidth = minLeft + Math.min(containerData.width, canvasData.width, canvasData.left + canvasData.width);\n maxHeight = minTop + Math.min(containerData.height, canvasData.height, canvasData.top + canvasData.height);\n }\n var pointer = pointers[Object.keys(pointers)[0]];\n var range = {\n x: pointer.endX - pointer.startX,\n y: pointer.endY - pointer.startY\n };\n var check = function check(side) {\n switch (side) {\n case ACTION_EAST:\n if (right + range.x > maxWidth) {\n range.x = maxWidth - right;\n }\n break;\n case ACTION_WEST:\n if (left + range.x < minLeft) {\n range.x = minLeft - left;\n }\n break;\n case ACTION_NORTH:\n if (top + range.y < minTop) {\n range.y = minTop - top;\n }\n break;\n case ACTION_SOUTH:\n if (bottom + range.y > maxHeight) {\n range.y = maxHeight - bottom;\n }\n break;\n }\n };\n switch (action) {\n // Move crop box\n case ACTION_ALL:\n left += range.x;\n top += range.y;\n break;\n\n // Resize crop box\n case ACTION_EAST:\n if (range.x >= 0 && (right >= maxWidth || aspectRatio && (top <= minTop || bottom >= maxHeight))) {\n renderable = false;\n break;\n }\n check(ACTION_EAST);\n width += range.x;\n if (width < 0) {\n action = ACTION_WEST;\n width = -width;\n left -= width;\n }\n if (aspectRatio) {\n height = width / aspectRatio;\n top += (cropBoxData.height - height) / 2;\n }\n break;\n case ACTION_NORTH:\n if (range.y <= 0 && (top <= minTop || aspectRatio && (left <= minLeft || right >= maxWidth))) {\n renderable = false;\n break;\n }\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n if (height < 0) {\n action = ACTION_SOUTH;\n height = -height;\n top -= height;\n }\n if (aspectRatio) {\n width = height * aspectRatio;\n left += (cropBoxData.width - width) / 2;\n }\n break;\n case ACTION_WEST:\n if (range.x <= 0 && (left <= minLeft || aspectRatio && (top <= minTop || bottom >= maxHeight))) {\n renderable = false;\n break;\n }\n check(ACTION_WEST);\n width -= range.x;\n left += range.x;\n if (width < 0) {\n action = ACTION_EAST;\n width = -width;\n left -= width;\n }\n if (aspectRatio) {\n height = width / aspectRatio;\n top += (cropBoxData.height - height) / 2;\n }\n break;\n case ACTION_SOUTH:\n if (range.y >= 0 && (bottom >= maxHeight || aspectRatio && (left <= minLeft || right >= maxWidth))) {\n renderable = false;\n break;\n }\n check(ACTION_SOUTH);\n height += range.y;\n if (height < 0) {\n action = ACTION_NORTH;\n height = -height;\n top -= height;\n }\n if (aspectRatio) {\n width = height * aspectRatio;\n left += (cropBoxData.width - width) / 2;\n }\n break;\n case ACTION_NORTH_EAST:\n if (aspectRatio) {\n if (range.y <= 0 && (top <= minTop || right >= maxWidth)) {\n renderable = false;\n break;\n }\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n width = height * aspectRatio;\n } else {\n check(ACTION_NORTH);\n check(ACTION_EAST);\n if (range.x >= 0) {\n if (right < maxWidth) {\n width += range.x;\n } else if (range.y <= 0 && top <= minTop) {\n renderable = false;\n }\n } else {\n width += range.x;\n }\n if (range.y <= 0) {\n if (top > minTop) {\n height -= range.y;\n top += range.y;\n }\n } else {\n height -= range.y;\n top += range.y;\n }\n }\n if (width < 0 && height < 0) {\n action = ACTION_SOUTH_WEST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_NORTH_WEST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_SOUTH_EAST;\n height = -height;\n top -= height;\n }\n break;\n case ACTION_NORTH_WEST:\n if (aspectRatio) {\n if (range.y <= 0 && (top <= minTop || left <= minLeft)) {\n renderable = false;\n break;\n }\n check(ACTION_NORTH);\n height -= range.y;\n top += range.y;\n width = height * aspectRatio;\n left += cropBoxData.width - width;\n } else {\n check(ACTION_NORTH);\n check(ACTION_WEST);\n if (range.x <= 0) {\n if (left > minLeft) {\n width -= range.x;\n left += range.x;\n } else if (range.y <= 0 && top <= minTop) {\n renderable = false;\n }\n } else {\n width -= range.x;\n left += range.x;\n }\n if (range.y <= 0) {\n if (top > minTop) {\n height -= range.y;\n top += range.y;\n }\n } else {\n height -= range.y;\n top += range.y;\n }\n }\n if (width < 0 && height < 0) {\n action = ACTION_SOUTH_EAST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_NORTH_EAST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_SOUTH_WEST;\n height = -height;\n top -= height;\n }\n break;\n case ACTION_SOUTH_WEST:\n if (aspectRatio) {\n if (range.x <= 0 && (left <= minLeft || bottom >= maxHeight)) {\n renderable = false;\n break;\n }\n check(ACTION_WEST);\n width -= range.x;\n left += range.x;\n height = width / aspectRatio;\n } else {\n check(ACTION_SOUTH);\n check(ACTION_WEST);\n if (range.x <= 0) {\n if (left > minLeft) {\n width -= range.x;\n left += range.x;\n } else if (range.y >= 0 && bottom >= maxHeight) {\n renderable = false;\n }\n } else {\n width -= range.x;\n left += range.x;\n }\n if (range.y >= 0) {\n if (bottom < maxHeight) {\n height += range.y;\n }\n } else {\n height += range.y;\n }\n }\n if (width < 0 && height < 0) {\n action = ACTION_NORTH_EAST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_SOUTH_EAST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_NORTH_WEST;\n height = -height;\n top -= height;\n }\n break;\n case ACTION_SOUTH_EAST:\n if (aspectRatio) {\n if (range.x >= 0 && (right >= maxWidth || bottom >= maxHeight)) {\n renderable = false;\n break;\n }\n check(ACTION_EAST);\n width += range.x;\n height = width / aspectRatio;\n } else {\n check(ACTION_SOUTH);\n check(ACTION_EAST);\n if (range.x >= 0) {\n if (right < maxWidth) {\n width += range.x;\n } else if (range.y >= 0 && bottom >= maxHeight) {\n renderable = false;\n }\n } else {\n width += range.x;\n }\n if (range.y >= 0) {\n if (bottom < maxHeight) {\n height += range.y;\n }\n } else {\n height += range.y;\n }\n }\n if (width < 0 && height < 0) {\n action = ACTION_NORTH_WEST;\n height = -height;\n width = -width;\n top -= height;\n left -= width;\n } else if (width < 0) {\n action = ACTION_SOUTH_WEST;\n width = -width;\n left -= width;\n } else if (height < 0) {\n action = ACTION_NORTH_EAST;\n height = -height;\n top -= height;\n }\n break;\n\n // Move canvas\n case ACTION_MOVE:\n this.move(range.x, range.y);\n renderable = false;\n break;\n\n // Zoom canvas\n case ACTION_ZOOM:\n this.zoom(getMaxZoomRatio(pointers), event);\n renderable = false;\n break;\n\n // Create crop box\n case ACTION_CROP:\n if (!range.x || !range.y) {\n renderable = false;\n break;\n }\n offset = getOffset(this.cropper);\n left = pointer.startX - offset.left;\n top = pointer.startY - offset.top;\n width = cropBoxData.minWidth;\n height = cropBoxData.minHeight;\n if (range.x > 0) {\n action = range.y > 0 ? ACTION_SOUTH_EAST : ACTION_NORTH_EAST;\n } else if (range.x < 0) {\n left -= width;\n action = range.y > 0 ? ACTION_SOUTH_WEST : ACTION_NORTH_WEST;\n }\n if (range.y < 0) {\n top -= height;\n }\n\n // Show the crop box if is hidden\n if (!this.cropped) {\n removeClass(this.cropBox, CLASS_HIDDEN);\n this.cropped = true;\n if (this.limited) {\n this.limitCropBox(true, true);\n }\n }\n break;\n }\n if (renderable) {\n cropBoxData.width = width;\n cropBoxData.height = height;\n cropBoxData.left = left;\n cropBoxData.top = top;\n this.action = action;\n this.renderCropBox();\n }\n\n // Override\n forEach(pointers, function (p) {\n p.startX = p.endX;\n p.startY = p.endY;\n });\n }\n };\n\n var methods = {\n // Show the crop box manually\n crop: function crop() {\n if (this.ready && !this.cropped && !this.disabled) {\n this.cropped = true;\n this.limitCropBox(true, true);\n if (this.options.modal) {\n addClass(this.dragBox, CLASS_MODAL);\n }\n removeClass(this.cropBox, CLASS_HIDDEN);\n this.setCropBoxData(this.initialCropBoxData);\n }\n return this;\n },\n // Reset the image and crop box to their initial states\n reset: function reset() {\n if (this.ready && !this.disabled) {\n this.imageData = assign({}, this.initialImageData);\n this.canvasData = assign({}, this.initialCanvasData);\n this.cropBoxData = assign({}, this.initialCropBoxData);\n this.renderCanvas();\n if (this.cropped) {\n this.renderCropBox();\n }\n }\n return this;\n },\n // Clear the crop box\n clear: function clear() {\n if (this.cropped && !this.disabled) {\n assign(this.cropBoxData, {\n left: 0,\n top: 0,\n width: 0,\n height: 0\n });\n this.cropped = false;\n this.renderCropBox();\n this.limitCanvas(true, true);\n\n // Render canvas after crop box rendered\n this.renderCanvas();\n removeClass(this.dragBox, CLASS_MODAL);\n addClass(this.cropBox, CLASS_HIDDEN);\n }\n return this;\n },\n /**\n * Replace the image's src and rebuild the cropper\n * @param {string} url - The new URL.\n * @param {boolean} [hasSameSize] - Indicate if the new image has the same size as the old one.\n * @returns {Cropper} this\n */\n replace: function replace(url) {\n var hasSameSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : false;\n if (!this.disabled && url) {\n if (this.isImg) {\n this.element.src = url;\n }\n if (hasSameSize) {\n this.url = url;\n this.image.src = url;\n if (this.ready) {\n this.viewBoxImage.src = url;\n forEach(this.previews, function (element) {\n element.getElementsByTagName('img')[0].src = url;\n });\n }\n } else {\n if (this.isImg) {\n this.replaced = true;\n }\n this.options.data = null;\n this.uncreate();\n this.load(url);\n }\n }\n return this;\n },\n // Enable (unfreeze) the cropper\n enable: function enable() {\n if (this.ready && this.disabled) {\n this.disabled = false;\n removeClass(this.cropper, CLASS_DISABLED);\n }\n return this;\n },\n // Disable (freeze) the cropper\n disable: function disable() {\n if (this.ready && !this.disabled) {\n this.disabled = true;\n addClass(this.cropper, CLASS_DISABLED);\n }\n return this;\n },\n /**\n * Destroy the cropper and remove the instance from the image\n * @returns {Cropper} this\n */\n destroy: function destroy() {\n var element = this.element;\n if (!element[NAMESPACE]) {\n return this;\n }\n element[NAMESPACE] = undefined;\n if (this.isImg && this.replaced) {\n element.src = this.originalUrl;\n }\n this.uncreate();\n return this;\n },\n /**\n * Move the canvas with relative offsets\n * @param {number} offsetX - The relative offset distance on the x-axis.\n * @param {number} [offsetY=offsetX] - The relative offset distance on the y-axis.\n * @returns {Cropper} this\n */\n move: function move(offsetX) {\n var offsetY = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : offsetX;\n var _this$canvasData = this.canvasData,\n left = _this$canvasData.left,\n top = _this$canvasData.top;\n return this.moveTo(isUndefined(offsetX) ? offsetX : left + Number(offsetX), isUndefined(offsetY) ? offsetY : top + Number(offsetY));\n },\n /**\n * Move the canvas to an absolute point\n * @param {number} x - The x-axis coordinate.\n * @param {number} [y=x] - The y-axis coordinate.\n * @returns {Cropper} this\n */\n moveTo: function moveTo(x) {\n var y = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : x;\n var canvasData = this.canvasData;\n var changed = false;\n x = Number(x);\n y = Number(y);\n if (this.ready && !this.disabled && this.options.movable) {\n if (isNumber(x)) {\n canvasData.left = x;\n changed = true;\n }\n if (isNumber(y)) {\n canvasData.top = y;\n changed = true;\n }\n if (changed) {\n this.renderCanvas(true);\n }\n }\n return this;\n },\n /**\n * Zoom the canvas with a relative ratio\n * @param {number} ratio - The target ratio.\n * @param {Event} _originalEvent - The original event if any.\n * @returns {Cropper} this\n */\n zoom: function zoom(ratio, _originalEvent) {\n var canvasData = this.canvasData;\n ratio = Number(ratio);\n if (ratio < 0) {\n ratio = 1 / (1 - ratio);\n } else {\n ratio = 1 + ratio;\n }\n return this.zoomTo(canvasData.width * ratio / canvasData.naturalWidth, null, _originalEvent);\n },\n /**\n * Zoom the canvas to an absolute ratio\n * @param {number} ratio - The target ratio.\n * @param {Object} pivot - The zoom pivot point coordinate.\n * @param {Event} _originalEvent - The original event if any.\n * @returns {Cropper} this\n */\n zoomTo: function zoomTo(ratio, pivot, _originalEvent) {\n var options = this.options,\n canvasData = this.canvasData;\n var width = canvasData.width,\n height = canvasData.height,\n naturalWidth = canvasData.naturalWidth,\n naturalHeight = canvasData.naturalHeight;\n ratio = Number(ratio);\n if (ratio >= 0 && this.ready && !this.disabled && options.zoomable) {\n var newWidth = naturalWidth * ratio;\n var newHeight = naturalHeight * ratio;\n if (dispatchEvent(this.element, EVENT_ZOOM, {\n ratio: ratio,\n oldRatio: width / naturalWidth,\n originalEvent: _originalEvent\n }) === false) {\n return this;\n }\n if (_originalEvent) {\n var pointers = this.pointers;\n var offset = getOffset(this.cropper);\n var center = pointers && Object.keys(pointers).length ? getPointersCenter(pointers) : {\n pageX: _originalEvent.pageX,\n pageY: _originalEvent.pageY\n };\n\n // Zoom from the triggering point of the event\n canvasData.left -= (newWidth - width) * ((center.pageX - offset.left - canvasData.left) / width);\n canvasData.top -= (newHeight - height) * ((center.pageY - offset.top - canvasData.top) / height);\n } else if (isPlainObject(pivot) && isNumber(pivot.x) && isNumber(pivot.y)) {\n canvasData.left -= (newWidth - width) * ((pivot.x - canvasData.left) / width);\n canvasData.top -= (newHeight - height) * ((pivot.y - canvasData.top) / height);\n } else {\n // Zoom from the center of the canvas\n canvasData.left -= (newWidth - width) / 2;\n canvasData.top -= (newHeight - height) / 2;\n }\n canvasData.width = newWidth;\n canvasData.height = newHeight;\n this.renderCanvas(true);\n }\n return this;\n },\n /**\n * Rotate the canvas with a relative degree\n * @param {number} degree - The rotate degree.\n * @returns {Cropper} this\n */\n rotate: function rotate(degree) {\n return this.rotateTo((this.imageData.rotate || 0) + Number(degree));\n },\n /**\n * Rotate the canvas to an absolute degree\n * @param {number} degree - The rotate degree.\n * @returns {Cropper} this\n */\n rotateTo: function rotateTo(degree) {\n degree = Number(degree);\n if (isNumber(degree) && this.ready && !this.disabled && this.options.rotatable) {\n this.imageData.rotate = degree % 360;\n this.renderCanvas(true, true);\n }\n return this;\n },\n /**\n * Scale the image on the x-axis.\n * @param {number} scaleX - The scale ratio on the x-axis.\n * @returns {Cropper} this\n */\n scaleX: function scaleX(_scaleX) {\n var scaleY = this.imageData.scaleY;\n return this.scale(_scaleX, isNumber(scaleY) ? scaleY : 1);\n },\n /**\n * Scale the image on the y-axis.\n * @param {number} scaleY - The scale ratio on the y-axis.\n * @returns {Cropper} this\n */\n scaleY: function scaleY(_scaleY) {\n var scaleX = this.imageData.scaleX;\n return this.scale(isNumber(scaleX) ? scaleX : 1, _scaleY);\n },\n /**\n * Scale the image\n * @param {number} scaleX - The scale ratio on the x-axis.\n * @param {number} [scaleY=scaleX] - The scale ratio on the y-axis.\n * @returns {Cropper} this\n */\n scale: function scale(scaleX) {\n var scaleY = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : scaleX;\n var imageData = this.imageData;\n var transformed = false;\n scaleX = Number(scaleX);\n scaleY = Number(scaleY);\n if (this.ready && !this.disabled && this.options.scalable) {\n if (isNumber(scaleX)) {\n imageData.scaleX = scaleX;\n transformed = true;\n }\n if (isNumber(scaleY)) {\n imageData.scaleY = scaleY;\n transformed = true;\n }\n if (transformed) {\n this.renderCanvas(true, true);\n }\n }\n return this;\n },\n /**\n * Get the cropped area position and size data (base on the original image)\n * @param {boolean} [rounded=false] - Indicate if round the data values or not.\n * @returns {Object} The result cropped data.\n */\n getData: function getData() {\n var rounded = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : false;\n var options = this.options,\n imageData = this.imageData,\n canvasData = this.canvasData,\n cropBoxData = this.cropBoxData;\n var data;\n if (this.ready && this.cropped) {\n data = {\n x: cropBoxData.left - canvasData.left,\n y: cropBoxData.top - canvasData.top,\n width: cropBoxData.width,\n height: cropBoxData.height\n };\n var ratio = imageData.width / imageData.naturalWidth;\n forEach(data, function (n, i) {\n data[i] = n / ratio;\n });\n if (rounded) {\n // In case rounding off leads to extra 1px in right or bottom border\n // we should round the top-left corner and the dimension (#343).\n var bottom = Math.round(data.y + data.height);\n var right = Math.round(data.x + data.width);\n data.x = Math.round(data.x);\n data.y = Math.round(data.y);\n data.width = right - data.x;\n data.height = bottom - data.y;\n }\n } else {\n data = {\n x: 0,\n y: 0,\n width: 0,\n height: 0\n };\n }\n if (options.rotatable) {\n data.rotate = imageData.rotate || 0;\n }\n if (options.scalable) {\n data.scaleX = imageData.scaleX || 1;\n data.scaleY = imageData.scaleY || 1;\n }\n return data;\n },\n /**\n * Set the cropped area position and size with new data\n * @param {Object} data - The new data.\n * @returns {Cropper} this\n */\n setData: function setData(data) {\n var options = this.options,\n imageData = this.imageData,\n canvasData = this.canvasData;\n var cropBoxData = {};\n if (this.ready && !this.disabled && isPlainObject(data)) {\n var transformed = false;\n if (options.rotatable) {\n if (isNumber(data.rotate) && data.rotate !== imageData.rotate) {\n imageData.rotate = data.rotate;\n transformed = true;\n }\n }\n if (options.scalable) {\n if (isNumber(data.scaleX) && data.scaleX !== imageData.scaleX) {\n imageData.scaleX = data.scaleX;\n transformed = true;\n }\n if (isNumber(data.scaleY) && data.scaleY !== imageData.scaleY) {\n imageData.scaleY = data.scaleY;\n transformed = true;\n }\n }\n if (transformed) {\n this.renderCanvas(true, true);\n }\n var ratio = imageData.width / imageData.naturalWidth;\n if (isNumber(data.x)) {\n cropBoxData.left = data.x * ratio + canvasData.left;\n }\n if (isNumber(data.y)) {\n cropBoxData.top = data.y * ratio + canvasData.top;\n }\n if (isNumber(data.width)) {\n cropBoxData.width = data.width * ratio;\n }\n if (isNumber(data.height)) {\n cropBoxData.height = data.height * ratio;\n }\n this.setCropBoxData(cropBoxData);\n }\n return this;\n },\n /**\n * Get the container size data.\n * @returns {Object} The result container data.\n */\n getContainerData: function getContainerData() {\n return this.ready ? assign({}, this.containerData) : {};\n },\n /**\n * Get the image position and size data.\n * @returns {Object} The result image data.\n */\n getImageData: function getImageData() {\n return this.sized ? assign({}, this.imageData) : {};\n },\n /**\n * Get the canvas position and size data.\n * @returns {Object} The result canvas data.\n */\n getCanvasData: function getCanvasData() {\n var canvasData = this.canvasData;\n var data = {};\n if (this.ready) {\n forEach(['left', 'top', 'width', 'height', 'naturalWidth', 'naturalHeight'], function (n) {\n data[n] = canvasData[n];\n });\n }\n return data;\n },\n /**\n * Set the canvas position and size with new data.\n * @param {Object} data - The new canvas data.\n * @returns {Cropper} this\n */\n setCanvasData: function setCanvasData(data) {\n var canvasData = this.canvasData;\n var aspectRatio = canvasData.aspectRatio;\n if (this.ready && !this.disabled && isPlainObject(data)) {\n if (isNumber(data.left)) {\n canvasData.left = data.left;\n }\n if (isNumber(data.top)) {\n canvasData.top = data.top;\n }\n if (isNumber(data.width)) {\n canvasData.width = data.width;\n canvasData.height = data.width / aspectRatio;\n } else if (isNumber(data.height)) {\n canvasData.height = data.height;\n canvasData.width = data.height * aspectRatio;\n }\n this.renderCanvas(true);\n }\n return this;\n },\n /**\n * Get the crop box position and size data.\n * @returns {Object} The result crop box data.\n */\n getCropBoxData: function getCropBoxData() {\n var cropBoxData = this.cropBoxData;\n var data;\n if (this.ready && this.cropped) {\n data = {\n left: cropBoxData.left,\n top: cropBoxData.top,\n width: cropBoxData.width,\n height: cropBoxData.height\n };\n }\n return data || {};\n },\n /**\n * Set the crop box position and size with new data.\n * @param {Object} data - The new crop box data.\n * @returns {Cropper} this\n */\n setCropBoxData: function setCropBoxData(data) {\n var cropBoxData = this.cropBoxData;\n var aspectRatio = this.options.aspectRatio;\n var widthChanged;\n var heightChanged;\n if (this.ready && this.cropped && !this.disabled && isPlainObject(data)) {\n if (isNumber(data.left)) {\n cropBoxData.left = data.left;\n }\n if (isNumber(data.top)) {\n cropBoxData.top = data.top;\n }\n if (isNumber(data.width) && data.width !== cropBoxData.width) {\n widthChanged = true;\n cropBoxData.width = data.width;\n }\n if (isNumber(data.height) && data.height !== cropBoxData.height) {\n heightChanged = true;\n cropBoxData.height = data.height;\n }\n if (aspectRatio) {\n if (widthChanged) {\n cropBoxData.height = cropBoxData.width / aspectRatio;\n } else if (heightChanged) {\n cropBoxData.width = cropBoxData.height * aspectRatio;\n }\n }\n this.renderCropBox();\n }\n return this;\n },\n /**\n * Get a canvas drawn the cropped image.\n * @param {Object} [options={}] - The config options.\n * @returns {HTMLCanvasElement} - The result canvas.\n */\n getCroppedCanvas: function getCroppedCanvas() {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n if (!this.ready || !window.HTMLCanvasElement) {\n return null;\n }\n var canvasData = this.canvasData;\n var source = getSourceCanvas(this.image, this.imageData, canvasData, options);\n\n // Returns the source canvas if it is not cropped.\n if (!this.cropped) {\n return source;\n }\n var _this$getData = this.getData(options.rounded),\n initialX = _this$getData.x,\n initialY = _this$getData.y,\n initialWidth = _this$getData.width,\n initialHeight = _this$getData.height;\n var ratio = source.width / Math.floor(canvasData.naturalWidth);\n if (ratio !== 1) {\n initialX *= ratio;\n initialY *= ratio;\n initialWidth *= ratio;\n initialHeight *= ratio;\n }\n var aspectRatio = initialWidth / initialHeight;\n var maxSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.maxWidth || Infinity,\n height: options.maxHeight || Infinity\n });\n var minSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.minWidth || 0,\n height: options.minHeight || 0\n }, 'cover');\n var _getAdjustedSizes = getAdjustedSizes({\n aspectRatio: aspectRatio,\n width: options.width || (ratio !== 1 ? source.width : initialWidth),\n height: options.height || (ratio !== 1 ? source.height : initialHeight)\n }),\n width = _getAdjustedSizes.width,\n height = _getAdjustedSizes.height;\n width = Math.min(maxSizes.width, Math.max(minSizes.width, width));\n height = Math.min(maxSizes.height, Math.max(minSizes.height, height));\n var canvas = document.createElement('canvas');\n var context = canvas.getContext('2d');\n canvas.width = normalizeDecimalNumber(width);\n canvas.height = normalizeDecimalNumber(height);\n context.fillStyle = options.fillColor || 'transparent';\n context.fillRect(0, 0, width, height);\n var _options$imageSmoothi = options.imageSmoothingEnabled,\n imageSmoothingEnabled = _options$imageSmoothi === void 0 ? true : _options$imageSmoothi,\n imageSmoothingQuality = options.imageSmoothingQuality;\n context.imageSmoothingEnabled = imageSmoothingEnabled;\n if (imageSmoothingQuality) {\n context.imageSmoothingQuality = imageSmoothingQuality;\n }\n\n // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D.drawImage\n var sourceWidth = source.width;\n var sourceHeight = source.height;\n\n // Source canvas parameters\n var srcX = initialX;\n var srcY = initialY;\n var srcWidth;\n var srcHeight;\n\n // Destination canvas parameters\n var dstX;\n var dstY;\n var dstWidth;\n var dstHeight;\n if (srcX <= -initialWidth || srcX > sourceWidth) {\n srcX = 0;\n srcWidth = 0;\n dstX = 0;\n dstWidth = 0;\n } else if (srcX <= 0) {\n dstX = -srcX;\n srcX = 0;\n srcWidth = Math.min(sourceWidth, initialWidth + srcX);\n dstWidth = srcWidth;\n } else if (srcX <= sourceWidth) {\n dstX = 0;\n srcWidth = Math.min(initialWidth, sourceWidth - srcX);\n dstWidth = srcWidth;\n }\n if (srcWidth <= 0 || srcY <= -initialHeight || srcY > sourceHeight) {\n srcY = 0;\n srcHeight = 0;\n dstY = 0;\n dstHeight = 0;\n } else if (srcY <= 0) {\n dstY = -srcY;\n srcY = 0;\n srcHeight = Math.min(sourceHeight, initialHeight + srcY);\n dstHeight = srcHeight;\n } else if (srcY <= sourceHeight) {\n dstY = 0;\n srcHeight = Math.min(initialHeight, sourceHeight - srcY);\n dstHeight = srcHeight;\n }\n var params = [srcX, srcY, srcWidth, srcHeight];\n\n // Avoid \"IndexSizeError\"\n if (dstWidth > 0 && dstHeight > 0) {\n var scale = width / initialWidth;\n params.push(dstX * scale, dstY * scale, dstWidth * scale, dstHeight * scale);\n }\n\n // All the numerical parameters should be integer for `drawImage`\n // https://github.com/fengyuanchen/cropper/issues/476\n context.drawImage.apply(context, [source].concat(_toConsumableArray(params.map(function (param) {\n return Math.floor(normalizeDecimalNumber(param));\n }))));\n return canvas;\n },\n /**\n * Change the aspect ratio of the crop box.\n * @param {number} aspectRatio - The new aspect ratio.\n * @returns {Cropper} this\n */\n setAspectRatio: function setAspectRatio(aspectRatio) {\n var options = this.options;\n if (!this.disabled && !isUndefined(aspectRatio)) {\n // 0 -> NaN\n options.aspectRatio = Math.max(0, aspectRatio) || NaN;\n if (this.ready) {\n this.initCropBox();\n if (this.cropped) {\n this.renderCropBox();\n }\n }\n }\n return this;\n },\n /**\n * Change the drag mode.\n * @param {string} mode - The new drag mode.\n * @returns {Cropper} this\n */\n setDragMode: function setDragMode(mode) {\n var options = this.options,\n dragBox = this.dragBox,\n face = this.face;\n if (this.ready && !this.disabled) {\n var croppable = mode === DRAG_MODE_CROP;\n var movable = options.movable && mode === DRAG_MODE_MOVE;\n mode = croppable || movable ? mode : DRAG_MODE_NONE;\n options.dragMode = mode;\n setData(dragBox, DATA_ACTION, mode);\n toggleClass(dragBox, CLASS_CROP, croppable);\n toggleClass(dragBox, CLASS_MOVE, movable);\n if (!options.cropBoxMovable) {\n // Sync drag mode to crop box when it is not movable\n setData(face, DATA_ACTION, mode);\n toggleClass(face, CLASS_CROP, croppable);\n toggleClass(face, CLASS_MOVE, movable);\n }\n }\n return this;\n }\n };\n\n var AnotherCropper = WINDOW.Cropper;\n var Cropper = /*#__PURE__*/function () {\n /**\n * Create a new Cropper.\n * @param {Element} element - The target element for cropping.\n * @param {Object} [options={}] - The configuration options.\n */\n function Cropper(element) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};\n _classCallCheck(this, Cropper);\n if (!element || !REGEXP_TAG_NAME.test(element.tagName)) {\n throw new Error('The first argument is required and must be an or element.');\n }\n this.element = element;\n this.options = assign({}, DEFAULTS, isPlainObject(options) && options);\n this.cropped = false;\n this.disabled = false;\n this.pointers = {};\n this.ready = false;\n this.reloading = false;\n this.replaced = false;\n this.sized = false;\n this.sizing = false;\n this.init();\n }\n return _createClass(Cropper, [{\n key: \"init\",\n value: function init() {\n var element = this.element;\n var tagName = element.tagName.toLowerCase();\n var url;\n if (element[NAMESPACE]) {\n return;\n }\n element[NAMESPACE] = this;\n if (tagName === 'img') {\n this.isImg = true;\n\n // e.g.: \"img/picture.jpg\"\n url = element.getAttribute('src') || '';\n this.originalUrl = url;\n\n // Stop when it's a blank image\n if (!url) {\n return;\n }\n\n // e.g.: \"https://example.com/img/picture.jpg\"\n url = element.src;\n } else if (tagName === 'canvas' && window.HTMLCanvasElement) {\n url = element.toDataURL();\n }\n this.load(url);\n }\n }, {\n key: \"load\",\n value: function load(url) {\n var _this = this;\n if (!url) {\n return;\n }\n this.url = url;\n this.imageData = {};\n var element = this.element,\n options = this.options;\n if (!options.rotatable && !options.scalable) {\n options.checkOrientation = false;\n }\n\n // Only IE10+ supports Typed Arrays\n if (!options.checkOrientation || !window.ArrayBuffer) {\n this.clone();\n return;\n }\n\n // Detect the mime type of the image directly if it is a Data URL\n if (REGEXP_DATA_URL.test(url)) {\n // Read ArrayBuffer from Data URL of JPEG images directly for better performance\n if (REGEXP_DATA_URL_JPEG.test(url)) {\n this.read(dataURLToArrayBuffer(url));\n } else {\n // Only a JPEG image may contains Exif Orientation information,\n // the rest types of Data URLs are not necessary to check orientation at all.\n this.clone();\n }\n return;\n }\n\n // 1. Detect the mime type of the image by a XMLHttpRequest.\n // 2. Load the image as ArrayBuffer for reading orientation if its a JPEG image.\n var xhr = new XMLHttpRequest();\n var clone = this.clone.bind(this);\n this.reloading = true;\n this.xhr = xhr;\n\n // 1. Cross origin requests are only supported for protocol schemes:\n // http, https, data, chrome, chrome-extension.\n // 2. Access to XMLHttpRequest from a Data URL will be blocked by CORS policy\n // in some browsers as IE11 and Safari.\n xhr.onabort = clone;\n xhr.onerror = clone;\n xhr.ontimeout = clone;\n xhr.onprogress = function () {\n // Abort the request directly if it not a JPEG image for better performance\n if (xhr.getResponseHeader('content-type') !== MIME_TYPE_JPEG) {\n xhr.abort();\n }\n };\n xhr.onload = function () {\n _this.read(xhr.response);\n };\n xhr.onloadend = function () {\n _this.reloading = false;\n _this.xhr = null;\n };\n\n // Bust cache when there is a \"crossOrigin\" property to avoid browser cache error\n if (options.checkCrossOrigin && isCrossOriginURL(url) && element.crossOrigin) {\n url = addTimestamp(url);\n }\n\n // The third parameter is required for avoiding side-effect (#682)\n xhr.open('GET', url, true);\n xhr.responseType = 'arraybuffer';\n xhr.withCredentials = element.crossOrigin === 'use-credentials';\n xhr.send();\n }\n }, {\n key: \"read\",\n value: function read(arrayBuffer) {\n var options = this.options,\n imageData = this.imageData;\n\n // Reset the orientation value to its default value 1\n // as some iOS browsers will render image with its orientation\n var orientation = resetAndGetOrientation(arrayBuffer);\n var rotate = 0;\n var scaleX = 1;\n var scaleY = 1;\n if (orientation > 1) {\n // Generate a new URL which has the default orientation value\n this.url = arrayBufferToDataURL(arrayBuffer, MIME_TYPE_JPEG);\n var _parseOrientation = parseOrientation(orientation);\n rotate = _parseOrientation.rotate;\n scaleX = _parseOrientation.scaleX;\n scaleY = _parseOrientation.scaleY;\n }\n if (options.rotatable) {\n imageData.rotate = rotate;\n }\n if (options.scalable) {\n imageData.scaleX = scaleX;\n imageData.scaleY = scaleY;\n }\n this.clone();\n }\n }, {\n key: \"clone\",\n value: function clone() {\n var element = this.element,\n url = this.url;\n var crossOrigin = element.crossOrigin;\n var crossOriginUrl = url;\n if (this.options.checkCrossOrigin && isCrossOriginURL(url)) {\n if (!crossOrigin) {\n crossOrigin = 'anonymous';\n }\n\n // Bust cache when there is not a \"crossOrigin\" property (#519)\n crossOriginUrl = addTimestamp(url);\n }\n this.crossOrigin = crossOrigin;\n this.crossOriginUrl = crossOriginUrl;\n var image = document.createElement('img');\n if (crossOrigin) {\n image.crossOrigin = crossOrigin;\n }\n image.src = crossOriginUrl || url;\n image.alt = element.alt || 'The image to crop';\n this.image = image;\n image.onload = this.start.bind(this);\n image.onerror = this.stop.bind(this);\n addClass(image, CLASS_HIDE);\n element.parentNode.insertBefore(image, element.nextSibling);\n }\n }, {\n key: \"start\",\n value: function start() {\n var _this2 = this;\n var image = this.image;\n image.onload = null;\n image.onerror = null;\n this.sizing = true;\n\n // Match all browsers that use WebKit as the layout engine in iOS devices,\n // such as Safari for iOS, Chrome for iOS, and in-app browsers.\n var isIOSWebKit = WINDOW.navigator && /(?:iPad|iPhone|iPod).*?AppleWebKit/i.test(WINDOW.navigator.userAgent);\n var done = function done(naturalWidth, naturalHeight) {\n assign(_this2.imageData, {\n naturalWidth: naturalWidth,\n naturalHeight: naturalHeight,\n aspectRatio: naturalWidth / naturalHeight\n });\n _this2.initialImageData = assign({}, _this2.imageData);\n _this2.sizing = false;\n _this2.sized = true;\n _this2.build();\n };\n\n // Most modern browsers (excepts iOS WebKit)\n if (image.naturalWidth && !isIOSWebKit) {\n done(image.naturalWidth, image.naturalHeight);\n return;\n }\n var sizingImage = document.createElement('img');\n var body = document.body || document.documentElement;\n this.sizingImage = sizingImage;\n sizingImage.onload = function () {\n done(sizingImage.width, sizingImage.height);\n if (!isIOSWebKit) {\n body.removeChild(sizingImage);\n }\n };\n sizingImage.src = image.src;\n\n // iOS WebKit will convert the image automatically\n // with its orientation once append it into DOM (#279)\n if (!isIOSWebKit) {\n sizingImage.style.cssText = 'left:0;' + 'max-height:none!important;' + 'max-width:none!important;' + 'min-height:0!important;' + 'min-width:0!important;' + 'opacity:0;' + 'position:absolute;' + 'top:0;' + 'z-index:-1;';\n body.appendChild(sizingImage);\n }\n }\n }, {\n key: \"stop\",\n value: function stop() {\n var image = this.image;\n image.onload = null;\n image.onerror = null;\n image.parentNode.removeChild(image);\n this.image = null;\n }\n }, {\n key: \"build\",\n value: function build() {\n if (!this.sized || this.ready) {\n return;\n }\n var element = this.element,\n options = this.options,\n image = this.image;\n\n // Create cropper elements\n var container = element.parentNode;\n var template = document.createElement('div');\n template.innerHTML = TEMPLATE;\n var cropper = template.querySelector(\".\".concat(NAMESPACE, \"-container\"));\n var canvas = cropper.querySelector(\".\".concat(NAMESPACE, \"-canvas\"));\n var dragBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-drag-box\"));\n var cropBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-crop-box\"));\n var face = cropBox.querySelector(\".\".concat(NAMESPACE, \"-face\"));\n this.container = container;\n this.cropper = cropper;\n this.canvas = canvas;\n this.dragBox = dragBox;\n this.cropBox = cropBox;\n this.viewBox = cropper.querySelector(\".\".concat(NAMESPACE, \"-view-box\"));\n this.face = face;\n canvas.appendChild(image);\n\n // Hide the original image\n addClass(element, CLASS_HIDDEN);\n\n // Inserts the cropper after to the current image\n container.insertBefore(cropper, element.nextSibling);\n\n // Show the hidden image\n removeClass(image, CLASS_HIDE);\n this.initPreview();\n this.bind();\n options.initialAspectRatio = Math.max(0, options.initialAspectRatio) || NaN;\n options.aspectRatio = Math.max(0, options.aspectRatio) || NaN;\n options.viewMode = Math.max(0, Math.min(3, Math.round(options.viewMode))) || 0;\n addClass(cropBox, CLASS_HIDDEN);\n if (!options.guides) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-dashed\")), CLASS_HIDDEN);\n }\n if (!options.center) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-center\")), CLASS_HIDDEN);\n }\n if (options.background) {\n addClass(cropper, \"\".concat(NAMESPACE, \"-bg\"));\n }\n if (!options.highlight) {\n addClass(face, CLASS_INVISIBLE);\n }\n if (options.cropBoxMovable) {\n addClass(face, CLASS_MOVE);\n setData(face, DATA_ACTION, ACTION_ALL);\n }\n if (!options.cropBoxResizable) {\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-line\")), CLASS_HIDDEN);\n addClass(cropBox.getElementsByClassName(\"\".concat(NAMESPACE, \"-point\")), CLASS_HIDDEN);\n }\n this.render();\n this.ready = true;\n this.setDragMode(options.dragMode);\n if (options.autoCrop) {\n this.crop();\n }\n this.setData(options.data);\n if (isFunction(options.ready)) {\n addListener(element, EVENT_READY, options.ready, {\n once: true\n });\n }\n dispatchEvent(element, EVENT_READY);\n }\n }, {\n key: \"unbuild\",\n value: function unbuild() {\n if (!this.ready) {\n return;\n }\n this.ready = false;\n this.unbind();\n this.resetPreview();\n var parentNode = this.cropper.parentNode;\n if (parentNode) {\n parentNode.removeChild(this.cropper);\n }\n removeClass(this.element, CLASS_HIDDEN);\n }\n }, {\n key: \"uncreate\",\n value: function uncreate() {\n if (this.ready) {\n this.unbuild();\n this.ready = false;\n this.cropped = false;\n } else if (this.sizing) {\n this.sizingImage.onload = null;\n this.sizing = false;\n this.sized = false;\n } else if (this.reloading) {\n this.xhr.onabort = null;\n this.xhr.abort();\n } else if (this.image) {\n this.stop();\n }\n }\n\n /**\n * Get the no conflict cropper class.\n * @returns {Cropper} The cropper class.\n */\n }], [{\n key: \"noConflict\",\n value: function noConflict() {\n window.Cropper = AnotherCropper;\n return Cropper;\n }\n\n /**\n * Change the default options.\n * @param {Object} options - The new default options.\n */\n }, {\n key: \"setDefaults\",\n value: function setDefaults(options) {\n assign(DEFAULTS, isPlainObject(options) && options);\n }\n }]);\n }();\n assign(Cropper.prototype, render, preview, events, handlers, change, methods);\n\n return Cropper;\n\n}));\n","export default function addLeadingZeros(number, targetLength) {\n var sign = number < 0 ? '-' : '';\n var output = Math.abs(number).toString();\n while (output.length < targetLength) {\n output = '0' + output;\n }\n return sign + output;\n}","var formatDistanceLocale = {\n lessThanXSeconds: {\n one: 'less than a second',\n other: 'less than {{count}} seconds'\n },\n xSeconds: {\n one: '1 second',\n other: '{{count}} seconds'\n },\n halfAMinute: 'half a minute',\n lessThanXMinutes: {\n one: 'less than a minute',\n other: 'less than {{count}} minutes'\n },\n xMinutes: {\n one: '1 minute',\n other: '{{count}} minutes'\n },\n aboutXHours: {\n one: 'about 1 hour',\n other: 'about {{count}} hours'\n },\n xHours: {\n one: '1 hour',\n other: '{{count}} hours'\n },\n xDays: {\n one: '1 day',\n other: '{{count}} days'\n },\n aboutXWeeks: {\n one: 'about 1 week',\n other: 'about {{count}} weeks'\n },\n xWeeks: {\n one: '1 week',\n other: '{{count}} weeks'\n },\n aboutXMonths: {\n one: 'about 1 month',\n other: 'about {{count}} months'\n },\n xMonths: {\n one: '1 month',\n other: '{{count}} months'\n },\n aboutXYears: {\n one: 'about 1 year',\n other: 'about {{count}} years'\n },\n xYears: {\n one: '1 year',\n other: '{{count}} years'\n },\n overXYears: {\n one: 'over 1 year',\n other: 'over {{count}} years'\n },\n almostXYears: {\n one: 'almost 1 year',\n other: 'almost {{count}} years'\n }\n};\nvar formatDistance = function formatDistance(token, count, options) {\n var result;\n var tokenValue = formatDistanceLocale[token];\n if (typeof tokenValue === 'string') {\n result = tokenValue;\n } else if (count === 1) {\n result = tokenValue.one;\n } else {\n result = tokenValue.other.replace('{{count}}', count.toString());\n }\n if (options !== null && options !== void 0 && options.addSuffix) {\n if (options.comparison && options.comparison > 0) {\n return 'in ' + result;\n } else {\n return result + ' ago';\n }\n }\n return result;\n};\nexport default formatDistance;","export default function buildFormatLongFn(args) {\n return function () {\n var options = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};\n // TODO: Remove String()\n var width = options.width ? String(options.width) : args.defaultWidth;\n var format = args.formats[width] || args.formats[args.defaultWidth];\n return format;\n };\n}","import buildFormatLongFn from \"../../../_lib/buildFormatLongFn/index.js\";\nvar dateFormats = {\n full: 'EEEE, MMMM do, y',\n long: 'MMMM do, y',\n medium: 'MMM d, y',\n short: 'MM/dd/yyyy'\n};\nvar timeFormats = {\n full: 'h:mm:ss a zzzz',\n long: 'h:mm:ss a z',\n medium: 'h:mm:ss a',\n short: 'h:mm a'\n};\nvar dateTimeFormats = {\n full: \"{{date}} 'at' {{time}}\",\n long: \"{{date}} 'at' {{time}}\",\n medium: '{{date}}, {{time}}',\n short: '{{date}}, {{time}}'\n};\nvar formatLong = {\n date: buildFormatLongFn({\n formats: dateFormats,\n defaultWidth: 'full'\n }),\n time: buildFormatLongFn({\n formats: timeFormats,\n defaultWidth: 'full'\n }),\n dateTime: buildFormatLongFn({\n formats: dateTimeFormats,\n defaultWidth: 'full'\n })\n};\nexport default formatLong;","var formatRelativeLocale = {\n lastWeek: \"'last' eeee 'at' p\",\n yesterday: \"'yesterday at' p\",\n today: \"'today at' p\",\n tomorrow: \"'tomorrow at' p\",\n nextWeek: \"eeee 'at' p\",\n other: 'P'\n};\nvar formatRelative = function formatRelative(token, _date, _baseDate, _options) {\n return formatRelativeLocale[token];\n};\nexport default formatRelative;","export default function buildLocalizeFn(args) {\n return function (dirtyIndex, options) {\n var context = options !== null && options !== void 0 && options.context ? String(options.context) : 'standalone';\n var valuesArray;\n if (context === 'formatting' && args.formattingValues) {\n var defaultWidth = args.defaultFormattingWidth || args.defaultWidth;\n var width = options !== null && options !== void 0 && options.width ? String(options.width) : defaultWidth;\n valuesArray = args.formattingValues[width] || args.formattingValues[defaultWidth];\n } else {\n var _defaultWidth = args.defaultWidth;\n var _width = options !== null && options !== void 0 && options.width ? String(options.width) : args.defaultWidth;\n valuesArray = args.values[_width] || args.values[_defaultWidth];\n }\n var index = args.argumentCallback ? args.argumentCallback(dirtyIndex) : dirtyIndex;\n // @ts-ignore: For some reason TypeScript just don't want to match it, no matter how hard we try. I challenge you to try to remove it!\n return valuesArray[index];\n };\n}","export default function buildMatchFn(args) {\n return function (string) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};\n var width = options.width;\n var matchPattern = width && args.matchPatterns[width] || args.matchPatterns[args.defaultMatchWidth];\n var matchResult = string.match(matchPattern);\n if (!matchResult) {\n return null;\n }\n var matchedString = matchResult[0];\n var parsePatterns = width && args.parsePatterns[width] || args.parsePatterns[args.defaultParseWidth];\n var key = Array.isArray(parsePatterns) ? findIndex(parsePatterns, function (pattern) {\n return pattern.test(matchedString);\n }) : findKey(parsePatterns, function (pattern) {\n return pattern.test(matchedString);\n });\n var value;\n value = args.valueCallback ? args.valueCallback(key) : key;\n value = options.valueCallback ? options.valueCallback(value) : value;\n var rest = string.slice(matchedString.length);\n return {\n value: value,\n rest: rest\n };\n };\n}\nfunction findKey(object, predicate) {\n for (var key in object) {\n if (object.hasOwnProperty(key) && predicate(object[key])) {\n return key;\n }\n }\n return undefined;\n}\nfunction findIndex(array, predicate) {\n for (var key = 0; key < array.length; key++) {\n if (predicate(array[key])) {\n return key;\n }\n }\n return undefined;\n}","import defaultLocale from \"../../locale/en-US/index.js\";\nexport default defaultLocale;","import formatDistance from \"./_lib/formatDistance/index.js\";\nimport formatLong from \"./_lib/formatLong/index.js\";\nimport formatRelative from \"./_lib/formatRelative/index.js\";\nimport localize from \"./_lib/localize/index.js\";\nimport match from \"./_lib/match/index.js\";\n/**\n * @type {Locale}\n * @category Locales\n * @summary English locale (United States).\n * @language English\n * @iso-639-2 eng\n * @author Sasha Koss [@kossnocorp]{@link https://github.com/kossnocorp}\n * @author Lesha Koss [@leshakoss]{@link https://github.com/leshakoss}\n */\nvar locale = {\n code: 'en-US',\n formatDistance: formatDistance,\n formatLong: formatLong,\n formatRelative: formatRelative,\n localize: localize,\n match: match,\n options: {\n weekStartsOn: 0 /* Sunday */,\n firstWeekContainsDate: 1\n }\n};\nexport default locale;","import buildLocalizeFn from \"../../../_lib/buildLocalizeFn/index.js\";\nvar eraValues = {\n narrow: ['B', 'A'],\n abbreviated: ['BC', 'AD'],\n wide: ['Before Christ', 'Anno Domini']\n};\nvar quarterValues = {\n narrow: ['1', '2', '3', '4'],\n abbreviated: ['Q1', 'Q2', 'Q3', 'Q4'],\n wide: ['1st quarter', '2nd quarter', '3rd quarter', '4th quarter']\n};\n\n// Note: in English, the names of days of the week and months are capitalized.\n// If you are making a new locale based on this one, check if the same is true for the language you're working on.\n// Generally, formatted dates should look like they are in the middle of a sentence,\n// e.g. in Spanish language the weekdays and months should be in the lowercase.\nvar monthValues = {\n narrow: ['J', 'F', 'M', 'A', 'M', 'J', 'J', 'A', 'S', 'O', 'N', 'D'],\n abbreviated: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec'],\n wide: ['January', 'February', 'March', 'April', 'May', 'June', 'July', 'August', 'September', 'October', 'November', 'December']\n};\nvar dayValues = {\n narrow: ['S', 'M', 'T', 'W', 'T', 'F', 'S'],\n short: ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa'],\n abbreviated: ['Sun', 'Mon', 'Tue', 'Wed', 'Thu', 'Fri', 'Sat'],\n wide: ['Sunday', 'Monday', 'Tuesday', 'Wednesday', 'Thursday', 'Friday', 'Saturday']\n};\nvar dayPeriodValues = {\n narrow: {\n am: 'a',\n pm: 'p',\n midnight: 'mi',\n noon: 'n',\n morning: 'morning',\n afternoon: 'afternoon',\n evening: 'evening',\n night: 'night'\n },\n abbreviated: {\n am: 'AM',\n pm: 'PM',\n midnight: 'midnight',\n noon: 'noon',\n morning: 'morning',\n afternoon: 'afternoon',\n evening: 'evening',\n night: 'night'\n },\n wide: {\n am: 'a.m.',\n pm: 'p.m.',\n midnight: 'midnight',\n noon: 'noon',\n morning: 'morning',\n afternoon: 'afternoon',\n evening: 'evening',\n night: 'night'\n }\n};\nvar formattingDayPeriodValues = {\n narrow: {\n am: 'a',\n pm: 'p',\n midnight: 'mi',\n noon: 'n',\n morning: 'in the morning',\n afternoon: 'in the afternoon',\n evening: 'in the evening',\n night: 'at night'\n },\n abbreviated: {\n am: 'AM',\n pm: 'PM',\n midnight: 'midnight',\n noon: 'noon',\n morning: 'in the morning',\n afternoon: 'in the afternoon',\n evening: 'in the evening',\n night: 'at night'\n },\n wide: {\n am: 'a.m.',\n pm: 'p.m.',\n midnight: 'midnight',\n noon: 'noon',\n morning: 'in the morning',\n afternoon: 'in the afternoon',\n evening: 'in the evening',\n night: 'at night'\n }\n};\nvar ordinalNumber = function ordinalNumber(dirtyNumber, _options) {\n var number = Number(dirtyNumber);\n\n // If ordinal numbers depend on context, for example,\n // if they are different for different grammatical genders,\n // use `options.unit`.\n //\n // `unit` can be 'year', 'quarter', 'month', 'week', 'date', 'dayOfYear',\n // 'day', 'hour', 'minute', 'second'.\n\n var rem100 = number % 100;\n if (rem100 > 20 || rem100 < 10) {\n switch (rem100 % 10) {\n case 1:\n return number + 'st';\n case 2:\n return number + 'nd';\n case 3:\n return number + 'rd';\n }\n }\n return number + 'th';\n};\nvar localize = {\n ordinalNumber: ordinalNumber,\n era: buildLocalizeFn({\n values: eraValues,\n defaultWidth: 'wide'\n }),\n quarter: buildLocalizeFn({\n values: quarterValues,\n defaultWidth: 'wide',\n argumentCallback: function argumentCallback(quarter) {\n return quarter - 1;\n }\n }),\n month: buildLocalizeFn({\n values: monthValues,\n defaultWidth: 'wide'\n }),\n day: buildLocalizeFn({\n values: dayValues,\n defaultWidth: 'wide'\n }),\n dayPeriod: buildLocalizeFn({\n values: dayPeriodValues,\n defaultWidth: 'wide',\n formattingValues: formattingDayPeriodValues,\n defaultFormattingWidth: 'wide'\n })\n};\nexport default localize;","import buildMatchFn from \"../../../_lib/buildMatchFn/index.js\";\nimport buildMatchPatternFn from \"../../../_lib/buildMatchPatternFn/index.js\";\nvar matchOrdinalNumberPattern = /^(\\d+)(th|st|nd|rd)?/i;\nvar parseOrdinalNumberPattern = /\\d+/i;\nvar matchEraPatterns = {\n narrow: /^(b|a)/i,\n abbreviated: /^(b\\.?\\s?c\\.?|b\\.?\\s?c\\.?\\s?e\\.?|a\\.?\\s?d\\.?|c\\.?\\s?e\\.?)/i,\n wide: /^(before christ|before common era|anno domini|common era)/i\n};\nvar parseEraPatterns = {\n any: [/^b/i, /^(a|c)/i]\n};\nvar matchQuarterPatterns = {\n narrow: /^[1234]/i,\n abbreviated: /^q[1234]/i,\n wide: /^[1234](th|st|nd|rd)? quarter/i\n};\nvar parseQuarterPatterns = {\n any: [/1/i, /2/i, /3/i, /4/i]\n};\nvar matchMonthPatterns = {\n narrow: /^[jfmasond]/i,\n abbreviated: /^(jan|feb|mar|apr|may|jun|jul|aug|sep|oct|nov|dec)/i,\n wide: /^(january|february|march|april|may|june|july|august|september|october|november|december)/i\n};\nvar parseMonthPatterns = {\n narrow: [/^j/i, /^f/i, /^m/i, /^a/i, /^m/i, /^j/i, /^j/i, /^a/i, /^s/i, /^o/i, /^n/i, /^d/i],\n any: [/^ja/i, /^f/i, /^mar/i, /^ap/i, /^may/i, /^jun/i, /^jul/i, /^au/i, /^s/i, /^o/i, /^n/i, /^d/i]\n};\nvar matchDayPatterns = {\n narrow: /^[smtwf]/i,\n short: /^(su|mo|tu|we|th|fr|sa)/i,\n abbreviated: /^(sun|mon|tue|wed|thu|fri|sat)/i,\n wide: /^(sunday|monday|tuesday|wednesday|thursday|friday|saturday)/i\n};\nvar parseDayPatterns = {\n narrow: [/^s/i, /^m/i, /^t/i, /^w/i, /^t/i, /^f/i, /^s/i],\n any: [/^su/i, /^m/i, /^tu/i, /^w/i, /^th/i, /^f/i, /^sa/i]\n};\nvar matchDayPeriodPatterns = {\n narrow: /^(a|p|mi|n|(in the|at) (morning|afternoon|evening|night))/i,\n any: /^([ap]\\.?\\s?m\\.?|midnight|noon|(in the|at) (morning|afternoon|evening|night))/i\n};\nvar parseDayPeriodPatterns = {\n any: {\n am: /^a/i,\n pm: /^p/i,\n midnight: /^mi/i,\n noon: /^no/i,\n morning: /morning/i,\n afternoon: /afternoon/i,\n evening: /evening/i,\n night: /night/i\n }\n};\nvar match = {\n ordinalNumber: buildMatchPatternFn({\n matchPattern: matchOrdinalNumberPattern,\n parsePattern: parseOrdinalNumberPattern,\n valueCallback: function valueCallback(value) {\n return parseInt(value, 10);\n }\n }),\n era: buildMatchFn({\n matchPatterns: matchEraPatterns,\n defaultMatchWidth: 'wide',\n parsePatterns: parseEraPatterns,\n defaultParseWidth: 'any'\n }),\n quarter: buildMatchFn({\n matchPatterns: matchQuarterPatterns,\n defaultMatchWidth: 'wide',\n parsePatterns: parseQuarterPatterns,\n defaultParseWidth: 'any',\n valueCallback: function valueCallback(index) {\n return index + 1;\n }\n }),\n month: buildMatchFn({\n matchPatterns: matchMonthPatterns,\n defaultMatchWidth: 'wide',\n parsePatterns: parseMonthPatterns,\n defaultParseWidth: 'any'\n }),\n day: buildMatchFn({\n matchPatterns: matchDayPatterns,\n defaultMatchWidth: 'wide',\n parsePatterns: parseDayPatterns,\n defaultParseWidth: 'any'\n }),\n dayPeriod: buildMatchFn({\n matchPatterns: matchDayPeriodPatterns,\n defaultMatchWidth: 'any',\n parsePatterns: parseDayPeriodPatterns,\n defaultParseWidth: 'any'\n })\n};\nexport default match;","export default function buildMatchPatternFn(args) {\n return function (string) {\n var options = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {};\n var matchResult = string.match(args.matchPattern);\n if (!matchResult) return null;\n var matchedString = matchResult[0];\n var parseResult = string.match(args.parsePattern);\n if (!parseResult) return null;\n var value = args.valueCallback ? args.valueCallback(parseResult[0]) : parseResult[0];\n value = options.valueCallback ? options.valueCallback(value) : value;\n var rest = string.slice(matchedString.length);\n return {\n value: value,\n rest: rest\n };\n };\n}","var defaultOptions = {};\nexport function getDefaultOptions() {\n return defaultOptions;\n}\nexport function setDefaultOptions(newOptions) {\n defaultOptions = newOptions;\n}","var dateLongFormatter = function dateLongFormatter(pattern, formatLong) {\n switch (pattern) {\n case 'P':\n return formatLong.date({\n width: 'short'\n });\n case 'PP':\n return formatLong.date({\n width: 'medium'\n });\n case 'PPP':\n return formatLong.date({\n width: 'long'\n });\n case 'PPPP':\n default:\n return formatLong.date({\n width: 'full'\n });\n }\n};\nvar timeLongFormatter = function timeLongFormatter(pattern, formatLong) {\n switch (pattern) {\n case 'p':\n return formatLong.time({\n width: 'short'\n });\n case 'pp':\n return formatLong.time({\n width: 'medium'\n });\n case 'ppp':\n return formatLong.time({\n width: 'long'\n });\n case 'pppp':\n default:\n return formatLong.time({\n width: 'full'\n });\n }\n};\nvar dateTimeLongFormatter = function dateTimeLongFormatter(pattern, formatLong) {\n var matchResult = pattern.match(/(P+)(p+)?/) || [];\n var datePattern = matchResult[1];\n var timePattern = matchResult[2];\n if (!timePattern) {\n return dateLongFormatter(pattern, formatLong);\n }\n var dateTimeFormat;\n switch (datePattern) {\n case 'P':\n dateTimeFormat = formatLong.dateTime({\n width: 'short'\n });\n break;\n case 'PP':\n dateTimeFormat = formatLong.dateTime({\n width: 'medium'\n });\n break;\n case 'PPP':\n dateTimeFormat = formatLong.dateTime({\n width: 'long'\n });\n break;\n case 'PPPP':\n default:\n dateTimeFormat = formatLong.dateTime({\n width: 'full'\n });\n break;\n }\n return dateTimeFormat.replace('{{date}}', dateLongFormatter(datePattern, formatLong)).replace('{{time}}', timeLongFormatter(timePattern, formatLong));\n};\nvar longFormatters = {\n p: timeLongFormatter,\n P: dateTimeLongFormatter\n};\nexport default longFormatters;","/**\n * Google Chrome as of 67.0.3396.87 introduced timezones with offset that includes seconds.\n * They usually appear for dates that denote time before the timezones were introduced\n * (e.g. for 'Europe/Prague' timezone the offset is GMT+00:57:44 before 1 October 1891\n * and GMT+01:00:00 after that date)\n *\n * Date#getTimezoneOffset returns the offset in minutes and would return 57 for the example above,\n * which would lead to incorrect calculations.\n *\n * This function returns the timezone offset in milliseconds that takes seconds in account.\n */\nexport default function getTimezoneOffsetInMilliseconds(date) {\n var utcDate = new Date(Date.UTC(date.getFullYear(), date.getMonth(), date.getDate(), date.getHours(), date.getMinutes(), date.getSeconds(), date.getMilliseconds()));\n utcDate.setUTCFullYear(date.getFullYear());\n return date.getTime() - utcDate.getTime();\n}","import getUTCISOWeekYear from \"../getUTCISOWeekYear/index.js\";\nimport startOfUTCISOWeek from \"../startOfUTCISOWeek/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nexport default function startOfUTCISOWeekYear(dirtyDate) {\n requiredArgs(1, arguments);\n var year = getUTCISOWeekYear(dirtyDate);\n var fourthOfJanuary = new Date(0);\n fourthOfJanuary.setUTCFullYear(year, 0, 4);\n fourthOfJanuary.setUTCHours(0, 0, 0, 0);\n var date = startOfUTCISOWeek(fourthOfJanuary);\n return date;\n}","import toDate from \"../../toDate/index.js\";\nimport startOfUTCISOWeek from \"../startOfUTCISOWeek/index.js\";\nimport startOfUTCISOWeekYear from \"../startOfUTCISOWeekYear/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nvar MILLISECONDS_IN_WEEK = 604800000;\nexport default function getUTCISOWeek(dirtyDate) {\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var diff = startOfUTCISOWeek(date).getTime() - startOfUTCISOWeekYear(date).getTime();\n\n // Round the number of days to the nearest integer\n // because the number of milliseconds in a week is not constant\n // (e.g. it's different in the week of the daylight saving time clock shift)\n return Math.round(diff / MILLISECONDS_IN_WEEK) + 1;\n}","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport startOfUTCISOWeek from \"../startOfUTCISOWeek/index.js\";\nexport default function getUTCISOWeekYear(dirtyDate) {\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var year = date.getUTCFullYear();\n var fourthOfJanuaryOfNextYear = new Date(0);\n fourthOfJanuaryOfNextYear.setUTCFullYear(year + 1, 0, 4);\n fourthOfJanuaryOfNextYear.setUTCHours(0, 0, 0, 0);\n var startOfNextYear = startOfUTCISOWeek(fourthOfJanuaryOfNextYear);\n var fourthOfJanuaryOfThisYear = new Date(0);\n fourthOfJanuaryOfThisYear.setUTCFullYear(year, 0, 4);\n fourthOfJanuaryOfThisYear.setUTCHours(0, 0, 0, 0);\n var startOfThisYear = startOfUTCISOWeek(fourthOfJanuaryOfThisYear);\n if (date.getTime() >= startOfNextYear.getTime()) {\n return year + 1;\n } else if (date.getTime() >= startOfThisYear.getTime()) {\n return year;\n } else {\n return year - 1;\n }\n}","import getUTCWeekYear from \"../getUTCWeekYear/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport startOfUTCWeek from \"../startOfUTCWeek/index.js\";\nimport toInteger from \"../toInteger/index.js\";\nimport { getDefaultOptions } from \"../defaultOptions/index.js\";\nexport default function startOfUTCWeekYear(dirtyDate, options) {\n var _ref, _ref2, _ref3, _options$firstWeekCon, _options$locale, _options$locale$optio, _defaultOptions$local, _defaultOptions$local2;\n requiredArgs(1, arguments);\n var defaultOptions = getDefaultOptions();\n var firstWeekContainsDate = toInteger((_ref = (_ref2 = (_ref3 = (_options$firstWeekCon = options === null || options === void 0 ? void 0 : options.firstWeekContainsDate) !== null && _options$firstWeekCon !== void 0 ? _options$firstWeekCon : options === null || options === void 0 ? void 0 : (_options$locale = options.locale) === null || _options$locale === void 0 ? void 0 : (_options$locale$optio = _options$locale.options) === null || _options$locale$optio === void 0 ? void 0 : _options$locale$optio.firstWeekContainsDate) !== null && _ref3 !== void 0 ? _ref3 : defaultOptions.firstWeekContainsDate) !== null && _ref2 !== void 0 ? _ref2 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.firstWeekContainsDate) !== null && _ref !== void 0 ? _ref : 1);\n var year = getUTCWeekYear(dirtyDate, options);\n var firstWeek = new Date(0);\n firstWeek.setUTCFullYear(year, 0, firstWeekContainsDate);\n firstWeek.setUTCHours(0, 0, 0, 0);\n var date = startOfUTCWeek(firstWeek, options);\n return date;\n}","import toDate from \"../../toDate/index.js\";\nimport startOfUTCWeek from \"../startOfUTCWeek/index.js\";\nimport startOfUTCWeekYear from \"../startOfUTCWeekYear/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nvar MILLISECONDS_IN_WEEK = 604800000;\nexport default function getUTCWeek(dirtyDate, options) {\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var diff = startOfUTCWeek(date, options).getTime() - startOfUTCWeekYear(date, options).getTime();\n\n // Round the number of days to the nearest integer\n // because the number of milliseconds in a week is not constant\n // (e.g. it's different in the week of the daylight saving time clock shift)\n return Math.round(diff / MILLISECONDS_IN_WEEK) + 1;\n}","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport startOfUTCWeek from \"../startOfUTCWeek/index.js\";\nimport toInteger from \"../toInteger/index.js\";\nimport { getDefaultOptions } from \"../defaultOptions/index.js\";\nexport default function getUTCWeekYear(dirtyDate, options) {\n var _ref, _ref2, _ref3, _options$firstWeekCon, _options$locale, _options$locale$optio, _defaultOptions$local, _defaultOptions$local2;\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var year = date.getUTCFullYear();\n var defaultOptions = getDefaultOptions();\n var firstWeekContainsDate = toInteger((_ref = (_ref2 = (_ref3 = (_options$firstWeekCon = options === null || options === void 0 ? void 0 : options.firstWeekContainsDate) !== null && _options$firstWeekCon !== void 0 ? _options$firstWeekCon : options === null || options === void 0 ? void 0 : (_options$locale = options.locale) === null || _options$locale === void 0 ? void 0 : (_options$locale$optio = _options$locale.options) === null || _options$locale$optio === void 0 ? void 0 : _options$locale$optio.firstWeekContainsDate) !== null && _ref3 !== void 0 ? _ref3 : defaultOptions.firstWeekContainsDate) !== null && _ref2 !== void 0 ? _ref2 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.firstWeekContainsDate) !== null && _ref !== void 0 ? _ref : 1);\n\n // Test if weekStartsOn is between 1 and 7 _and_ is not NaN\n if (!(firstWeekContainsDate >= 1 && firstWeekContainsDate <= 7)) {\n throw new RangeError('firstWeekContainsDate must be between 1 and 7 inclusively');\n }\n var firstWeekOfNextYear = new Date(0);\n firstWeekOfNextYear.setUTCFullYear(year + 1, 0, firstWeekContainsDate);\n firstWeekOfNextYear.setUTCHours(0, 0, 0, 0);\n var startOfNextYear = startOfUTCWeek(firstWeekOfNextYear, options);\n var firstWeekOfThisYear = new Date(0);\n firstWeekOfThisYear.setUTCFullYear(year, 0, firstWeekContainsDate);\n firstWeekOfThisYear.setUTCHours(0, 0, 0, 0);\n var startOfThisYear = startOfUTCWeek(firstWeekOfThisYear, options);\n if (date.getTime() >= startOfNextYear.getTime()) {\n return year + 1;\n } else if (date.getTime() >= startOfThisYear.getTime()) {\n return year;\n } else {\n return year - 1;\n }\n}","var protectedDayOfYearTokens = ['D', 'DD'];\nvar protectedWeekYearTokens = ['YY', 'YYYY'];\nexport function isProtectedDayOfYearToken(token) {\n return protectedDayOfYearTokens.indexOf(token) !== -1;\n}\nexport function isProtectedWeekYearToken(token) {\n return protectedWeekYearTokens.indexOf(token) !== -1;\n}\nexport function throwProtectedError(token, format, input) {\n if (token === 'YYYY') {\n throw new RangeError(\"Use `yyyy` instead of `YYYY` (in `\".concat(format, \"`) for formatting years to the input `\").concat(input, \"`; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\"));\n } else if (token === 'YY') {\n throw new RangeError(\"Use `yy` instead of `YY` (in `\".concat(format, \"`) for formatting years to the input `\").concat(input, \"`; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\"));\n } else if (token === 'D') {\n throw new RangeError(\"Use `d` instead of `D` (in `\".concat(format, \"`) for formatting days of the month to the input `\").concat(input, \"`; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\"));\n } else if (token === 'DD') {\n throw new RangeError(\"Use `dd` instead of `DD` (in `\".concat(format, \"`) for formatting days of the month to the input `\").concat(input, \"`; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\"));\n }\n}","export default function requiredArgs(required, args) {\n if (args.length < required) {\n throw new TypeError(required + ' argument' + (required > 1 ? 's' : '') + ' required, but only ' + args.length + ' present');\n }\n}","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nexport default function startOfUTCISOWeek(dirtyDate) {\n requiredArgs(1, arguments);\n var weekStartsOn = 1;\n var date = toDate(dirtyDate);\n var day = date.getUTCDay();\n var diff = (day < weekStartsOn ? 7 : 0) + day - weekStartsOn;\n date.setUTCDate(date.getUTCDate() - diff);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n}","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport toInteger from \"../toInteger/index.js\";\nimport { getDefaultOptions } from \"../defaultOptions/index.js\";\nexport default function startOfUTCWeek(dirtyDate, options) {\n var _ref, _ref2, _ref3, _options$weekStartsOn, _options$locale, _options$locale$optio, _defaultOptions$local, _defaultOptions$local2;\n requiredArgs(1, arguments);\n var defaultOptions = getDefaultOptions();\n var weekStartsOn = toInteger((_ref = (_ref2 = (_ref3 = (_options$weekStartsOn = options === null || options === void 0 ? void 0 : options.weekStartsOn) !== null && _options$weekStartsOn !== void 0 ? _options$weekStartsOn : options === null || options === void 0 ? void 0 : (_options$locale = options.locale) === null || _options$locale === void 0 ? void 0 : (_options$locale$optio = _options$locale.options) === null || _options$locale$optio === void 0 ? void 0 : _options$locale$optio.weekStartsOn) !== null && _ref3 !== void 0 ? _ref3 : defaultOptions.weekStartsOn) !== null && _ref2 !== void 0 ? _ref2 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.weekStartsOn) !== null && _ref !== void 0 ? _ref : 0);\n\n // Test if weekStartsOn is between 0 and 6 _and_ is not NaN\n if (!(weekStartsOn >= 0 && weekStartsOn <= 6)) {\n throw new RangeError('weekStartsOn must be between 0 and 6 inclusively');\n }\n var date = toDate(dirtyDate);\n var day = date.getUTCDay();\n var diff = (day < weekStartsOn ? 7 : 0) + day - weekStartsOn;\n date.setUTCDate(date.getUTCDate() - diff);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n}","export default function toInteger(dirtyNumber) {\n if (dirtyNumber === null || dirtyNumber === true || dirtyNumber === false) {\n return NaN;\n }\n var number = Number(dirtyNumber);\n if (isNaN(number)) {\n return number;\n }\n return number < 0 ? Math.ceil(number) : Math.floor(number);\n}","import toInteger from \"../_lib/toInteger/index.js\";\nimport toDate from \"../toDate/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name addMilliseconds\n * @category Millisecond Helpers\n * @summary Add the specified number of milliseconds to the given date.\n *\n * @description\n * Add the specified number of milliseconds to the given date.\n *\n * @param {Date|Number} date - the date to be changed\n * @param {Number} amount - the amount of milliseconds to be added. Positive decimals will be rounded using `Math.floor`, decimals less than zero will be rounded using `Math.ceil`.\n * @returns {Date} the new date with the milliseconds added\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // Add 750 milliseconds to 10 July 2014 12:45:30.000:\n * const result = addMilliseconds(new Date(2014, 6, 10, 12, 45, 30, 0), 750)\n * //=> Thu Jul 10 2014 12:45:30.750\n */\nexport default function addMilliseconds(dirtyDate, dirtyAmount) {\n requiredArgs(2, arguments);\n var timestamp = toDate(dirtyDate).getTime();\n var amount = toInteger(dirtyAmount);\n return new Date(timestamp + amount);\n}","import toDate from \"../toDate/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name compareAsc\n * @category Common Helpers\n * @summary Compare the two dates and return -1, 0 or 1.\n *\n * @description\n * Compare the two dates and return 1 if the first date is after the second,\n * -1 if the first date is before the second or 0 if dates are equal.\n *\n * @param {Date|Number} dateLeft - the first date to compare\n * @param {Date|Number} dateRight - the second date to compare\n * @returns {Number} the result of the comparison\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // Compare 11 February 1987 and 10 July 1989:\n * const result = compareAsc(new Date(1987, 1, 11), new Date(1989, 6, 10))\n * //=> -1\n *\n * @example\n * // Sort the array of dates:\n * const result = [\n * new Date(1995, 6, 2),\n * new Date(1987, 1, 11),\n * new Date(1989, 6, 10)\n * ].sort(compareAsc)\n * //=> [\n * // Wed Feb 11 1987 00:00:00,\n * // Mon Jul 10 1989 00:00:00,\n * // Sun Jul 02 1995 00:00:00\n * // ]\n */\nexport default function compareAsc(dirtyDateLeft, dirtyDateRight) {\n requiredArgs(2, arguments);\n var dateLeft = toDate(dirtyDateLeft);\n var dateRight = toDate(dirtyDateRight);\n var diff = dateLeft.getTime() - dateRight.getTime();\n if (diff < 0) {\n return -1;\n } else if (diff > 0) {\n return 1;\n // Return 0 if diff is 0; return NaN if diff is NaN\n } else {\n return diff;\n }\n}","/**\n * Days in 1 week.\n *\n * @name daysInWeek\n * @constant\n * @type {number}\n * @default\n */\nexport var daysInWeek = 7;\n\n/**\n * Days in 1 year\n * One years equals 365.2425 days according to the formula:\n *\n * > Leap year occures every 4 years, except for years that are divisable by 100 and not divisable by 400.\n * > 1 mean year = (365+1/4-1/100+1/400) days = 365.2425 days\n *\n * @name daysInYear\n * @constant\n * @type {number}\n * @default\n */\nexport var daysInYear = 365.2425;\n\n/**\n * Maximum allowed time.\n *\n * @name maxTime\n * @constant\n * @type {number}\n * @default\n */\nexport var maxTime = Math.pow(10, 8) * 24 * 60 * 60 * 1000;\n\n/**\n * Milliseconds in 1 minute\n *\n * @name millisecondsInMinute\n * @constant\n * @type {number}\n * @default\n */\nexport var millisecondsInMinute = 60000;\n\n/**\n * Milliseconds in 1 hour\n *\n * @name millisecondsInHour\n * @constant\n * @type {number}\n * @default\n */\nexport var millisecondsInHour = 3600000;\n\n/**\n * Milliseconds in 1 second\n *\n * @name millisecondsInSecond\n * @constant\n * @type {number}\n * @default\n */\nexport var millisecondsInSecond = 1000;\n\n/**\n * Minimum allowed time.\n *\n * @name minTime\n * @constant\n * @type {number}\n * @default\n */\nexport var minTime = -maxTime;\n\n/**\n * Minutes in 1 hour\n *\n * @name minutesInHour\n * @constant\n * @type {number}\n * @default\n */\nexport var minutesInHour = 60;\n\n/**\n * Months in 1 quarter\n *\n * @name monthsInQuarter\n * @constant\n * @type {number}\n * @default\n */\nexport var monthsInQuarter = 3;\n\n/**\n * Months in 1 year\n *\n * @name monthsInYear\n * @constant\n * @type {number}\n * @default\n */\nexport var monthsInYear = 12;\n\n/**\n * Quarters in 1 year\n *\n * @name quartersInYear\n * @constant\n * @type {number}\n * @default\n */\nexport var quartersInYear = 4;\n\n/**\n * Seconds in 1 hour\n *\n * @name secondsInHour\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInHour = 3600;\n\n/**\n * Seconds in 1 minute\n *\n * @name secondsInMinute\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInMinute = 60;\n\n/**\n * Seconds in 1 day\n *\n * @name secondsInDay\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInDay = secondsInHour * 24;\n\n/**\n * Seconds in 1 week\n *\n * @name secondsInWeek\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInWeek = secondsInDay * 7;\n\n/**\n * Seconds in 1 year\n *\n * @name secondsInYear\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInYear = secondsInDay * daysInYear;\n\n/**\n * Seconds in 1 month\n *\n * @name secondsInMonth\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInMonth = secondsInYear / 12;\n\n/**\n * Seconds in 1 quarter\n *\n * @name secondsInQuarter\n * @constant\n * @type {number}\n * @default\n */\nexport var secondsInQuarter = secondsInMonth * 3;","import toDate from \"../toDate/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name differenceInMilliseconds\n * @category Millisecond Helpers\n * @summary Get the number of milliseconds between the given dates.\n *\n * @description\n * Get the number of milliseconds between the given dates.\n *\n * @param {Date|Number} dateLeft - the later date\n * @param {Date|Number} dateRight - the earlier date\n * @returns {Number} the number of milliseconds\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // How many milliseconds are between\n * // 2 July 2014 12:30:20.600 and 2 July 2014 12:30:21.700?\n * const result = differenceInMilliseconds(\n * new Date(2014, 6, 2, 12, 30, 21, 700),\n * new Date(2014, 6, 2, 12, 30, 20, 600)\n * )\n * //=> 1100\n */\nexport default function differenceInMilliseconds(dateLeft, dateRight) {\n requiredArgs(2, arguments);\n return toDate(dateLeft).getTime() - toDate(dateRight).getTime();\n}","var roundingMap = {\n ceil: Math.ceil,\n round: Math.round,\n floor: Math.floor,\n trunc: function trunc(value) {\n return value < 0 ? Math.ceil(value) : Math.floor(value);\n } // Math.trunc is not supported by IE\n};\n\nvar defaultRoundingMethod = 'trunc';\nexport function getRoundingMethod(method) {\n return method ? roundingMap[method] : roundingMap[defaultRoundingMethod];\n}","import { millisecondsInMinute } from \"../constants/index.js\";\nimport differenceInMilliseconds from \"../differenceInMilliseconds/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\nimport { getRoundingMethod } from \"../_lib/roundingMethods/index.js\";\n/**\n * @name differenceInMinutes\n * @category Minute Helpers\n * @summary Get the number of minutes between the given dates.\n *\n * @description\n * Get the signed number of full (rounded towards 0) minutes between the given dates.\n *\n * @param {Date|Number} dateLeft - the later date\n * @param {Date|Number} dateRight - the earlier date\n * @param {Object} [options] - an object with options.\n * @param {String} [options.roundingMethod='trunc'] - a rounding method (`ceil`, `floor`, `round` or `trunc`)\n * @returns {Number} the number of minutes\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // How many minutes are between 2 July 2014 12:07:59 and 2 July 2014 12:20:00?\n * const result = differenceInMinutes(\n * new Date(2014, 6, 2, 12, 20, 0),\n * new Date(2014, 6, 2, 12, 7, 59)\n * )\n * //=> 12\n *\n * @example\n * // How many minutes are between 10:01:59 and 10:00:00\n * const result = differenceInMinutes(\n * new Date(2000, 0, 1, 10, 0, 0),\n * new Date(2000, 0, 1, 10, 1, 59)\n * )\n * //=> -1\n */\nexport default function differenceInMinutes(dateLeft, dateRight, options) {\n requiredArgs(2, arguments);\n var diff = differenceInMilliseconds(dateLeft, dateRight) / millisecondsInMinute;\n return getRoundingMethod(options === null || options === void 0 ? void 0 : options.roundingMethod)(diff);\n}","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nvar MILLISECONDS_IN_DAY = 86400000;\nexport default function getUTCDayOfYear(dirtyDate) {\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var timestamp = date.getTime();\n date.setUTCMonth(0, 1);\n date.setUTCHours(0, 0, 0, 0);\n var startOfYearTimestamp = date.getTime();\n var difference = timestamp - startOfYearTimestamp;\n return Math.floor(difference / MILLISECONDS_IN_DAY) + 1;\n}","import addLeadingZeros from \"../../addLeadingZeros/index.js\";\n/*\n * | | Unit | | Unit |\n * |-----|--------------------------------|-----|--------------------------------|\n * | a | AM, PM | A* | |\n * | d | Day of month | D | |\n * | h | Hour [1-12] | H | Hour [0-23] |\n * | m | Minute | M | Month |\n * | s | Second | S | Fraction of second |\n * | y | Year (abs) | Y | |\n *\n * Letters marked by * are not implemented but reserved by Unicode standard.\n */\nvar formatters = {\n // Year\n y: function y(date, token) {\n // From http://www.unicode.org/reports/tr35/tr35-31/tr35-dates.html#Date_Format_tokens\n // | Year | y | yy | yyy | yyyy | yyyyy |\n // |----------|-------|----|-------|-------|-------|\n // | AD 1 | 1 | 01 | 001 | 0001 | 00001 |\n // | AD 12 | 12 | 12 | 012 | 0012 | 00012 |\n // | AD 123 | 123 | 23 | 123 | 0123 | 00123 |\n // | AD 1234 | 1234 | 34 | 1234 | 1234 | 01234 |\n // | AD 12345 | 12345 | 45 | 12345 | 12345 | 12345 |\n\n var signedYear = date.getUTCFullYear();\n // Returns 1 for 1 BC (which is year 0 in JavaScript)\n var year = signedYear > 0 ? signedYear : 1 - signedYear;\n return addLeadingZeros(token === 'yy' ? year % 100 : year, token.length);\n },\n // Month\n M: function M(date, token) {\n var month = date.getUTCMonth();\n return token === 'M' ? String(month + 1) : addLeadingZeros(month + 1, 2);\n },\n // Day of the month\n d: function d(date, token) {\n return addLeadingZeros(date.getUTCDate(), token.length);\n },\n // AM or PM\n a: function a(date, token) {\n var dayPeriodEnumValue = date.getUTCHours() / 12 >= 1 ? 'pm' : 'am';\n switch (token) {\n case 'a':\n case 'aa':\n return dayPeriodEnumValue.toUpperCase();\n case 'aaa':\n return dayPeriodEnumValue;\n case 'aaaaa':\n return dayPeriodEnumValue[0];\n case 'aaaa':\n default:\n return dayPeriodEnumValue === 'am' ? 'a.m.' : 'p.m.';\n }\n },\n // Hour [1-12]\n h: function h(date, token) {\n return addLeadingZeros(date.getUTCHours() % 12 || 12, token.length);\n },\n // Hour [0-23]\n H: function H(date, token) {\n return addLeadingZeros(date.getUTCHours(), token.length);\n },\n // Minute\n m: function m(date, token) {\n return addLeadingZeros(date.getUTCMinutes(), token.length);\n },\n // Second\n s: function s(date, token) {\n return addLeadingZeros(date.getUTCSeconds(), token.length);\n },\n // Fraction of second\n S: function S(date, token) {\n var numberOfDigits = token.length;\n var milliseconds = date.getUTCMilliseconds();\n var fractionalSeconds = Math.floor(milliseconds * Math.pow(10, numberOfDigits - 3));\n return addLeadingZeros(fractionalSeconds, token.length);\n }\n};\nexport default formatters;","import getUTCDayOfYear from \"../../../_lib/getUTCDayOfYear/index.js\";\nimport getUTCISOWeek from \"../../../_lib/getUTCISOWeek/index.js\";\nimport getUTCISOWeekYear from \"../../../_lib/getUTCISOWeekYear/index.js\";\nimport getUTCWeek from \"../../../_lib/getUTCWeek/index.js\";\nimport getUTCWeekYear from \"../../../_lib/getUTCWeekYear/index.js\";\nimport addLeadingZeros from \"../../addLeadingZeros/index.js\";\nimport lightFormatters from \"../lightFormatters/index.js\";\nvar dayPeriodEnum = {\n am: 'am',\n pm: 'pm',\n midnight: 'midnight',\n noon: 'noon',\n morning: 'morning',\n afternoon: 'afternoon',\n evening: 'evening',\n night: 'night'\n};\n/*\n * | | Unit | | Unit |\n * |-----|--------------------------------|-----|--------------------------------|\n * | a | AM, PM | A* | Milliseconds in day |\n * | b | AM, PM, noon, midnight | B | Flexible day period |\n * | c | Stand-alone local day of week | C* | Localized hour w/ day period |\n * | d | Day of month | D | Day of year |\n * | e | Local day of week | E | Day of week |\n * | f | | F* | Day of week in month |\n * | g* | Modified Julian day | G | Era |\n * | h | Hour [1-12] | H | Hour [0-23] |\n * | i! | ISO day of week | I! | ISO week of year |\n * | j* | Localized hour w/ day period | J* | Localized hour w/o day period |\n * | k | Hour [1-24] | K | Hour [0-11] |\n * | l* | (deprecated) | L | Stand-alone month |\n * | m | Minute | M | Month |\n * | n | | N | |\n * | o! | Ordinal number modifier | O | Timezone (GMT) |\n * | p! | Long localized time | P! | Long localized date |\n * | q | Stand-alone quarter | Q | Quarter |\n * | r* | Related Gregorian year | R! | ISO week-numbering year |\n * | s | Second | S | Fraction of second |\n * | t! | Seconds timestamp | T! | Milliseconds timestamp |\n * | u | Extended year | U* | Cyclic year |\n * | v* | Timezone (generic non-locat.) | V* | Timezone (location) |\n * | w | Local week of year | W* | Week of month |\n * | x | Timezone (ISO-8601 w/o Z) | X | Timezone (ISO-8601) |\n * | y | Year (abs) | Y | Local week-numbering year |\n * | z | Timezone (specific non-locat.) | Z* | Timezone (aliases) |\n *\n * Letters marked by * are not implemented but reserved by Unicode standard.\n *\n * Letters marked by ! are non-standard, but implemented by date-fns:\n * - `o` modifies the previous token to turn it into an ordinal (see `format` docs)\n * - `i` is ISO day of week. For `i` and `ii` is returns numeric ISO week days,\n * i.e. 7 for Sunday, 1 for Monday, etc.\n * - `I` is ISO week of year, as opposed to `w` which is local week of year.\n * - `R` is ISO week-numbering year, as opposed to `Y` which is local week-numbering year.\n * `R` is supposed to be used in conjunction with `I` and `i`\n * for universal ISO week-numbering date, whereas\n * `Y` is supposed to be used in conjunction with `w` and `e`\n * for week-numbering date specific to the locale.\n * - `P` is long localized date format\n * - `p` is long localized time format\n */\n\nvar formatters = {\n // Era\n G: function G(date, token, localize) {\n var era = date.getUTCFullYear() > 0 ? 1 : 0;\n switch (token) {\n // AD, BC\n case 'G':\n case 'GG':\n case 'GGG':\n return localize.era(era, {\n width: 'abbreviated'\n });\n // A, B\n case 'GGGGG':\n return localize.era(era, {\n width: 'narrow'\n });\n // Anno Domini, Before Christ\n case 'GGGG':\n default:\n return localize.era(era, {\n width: 'wide'\n });\n }\n },\n // Year\n y: function y(date, token, localize) {\n // Ordinal number\n if (token === 'yo') {\n var signedYear = date.getUTCFullYear();\n // Returns 1 for 1 BC (which is year 0 in JavaScript)\n var year = signedYear > 0 ? signedYear : 1 - signedYear;\n return localize.ordinalNumber(year, {\n unit: 'year'\n });\n }\n return lightFormatters.y(date, token);\n },\n // Local week-numbering year\n Y: function Y(date, token, localize, options) {\n var signedWeekYear = getUTCWeekYear(date, options);\n // Returns 1 for 1 BC (which is year 0 in JavaScript)\n var weekYear = signedWeekYear > 0 ? signedWeekYear : 1 - signedWeekYear;\n\n // Two digit year\n if (token === 'YY') {\n var twoDigitYear = weekYear % 100;\n return addLeadingZeros(twoDigitYear, 2);\n }\n\n // Ordinal number\n if (token === 'Yo') {\n return localize.ordinalNumber(weekYear, {\n unit: 'year'\n });\n }\n\n // Padding\n return addLeadingZeros(weekYear, token.length);\n },\n // ISO week-numbering year\n R: function R(date, token) {\n var isoWeekYear = getUTCISOWeekYear(date);\n\n // Padding\n return addLeadingZeros(isoWeekYear, token.length);\n },\n // Extended year. This is a single number designating the year of this calendar system.\n // The main difference between `y` and `u` localizers are B.C. years:\n // | Year | `y` | `u` |\n // |------|-----|-----|\n // | AC 1 | 1 | 1 |\n // | BC 1 | 1 | 0 |\n // | BC 2 | 2 | -1 |\n // Also `yy` always returns the last two digits of a year,\n // while `uu` pads single digit years to 2 characters and returns other years unchanged.\n u: function u(date, token) {\n var year = date.getUTCFullYear();\n return addLeadingZeros(year, token.length);\n },\n // Quarter\n Q: function Q(date, token, localize) {\n var quarter = Math.ceil((date.getUTCMonth() + 1) / 3);\n switch (token) {\n // 1, 2, 3, 4\n case 'Q':\n return String(quarter);\n // 01, 02, 03, 04\n case 'QQ':\n return addLeadingZeros(quarter, 2);\n // 1st, 2nd, 3rd, 4th\n case 'Qo':\n return localize.ordinalNumber(quarter, {\n unit: 'quarter'\n });\n // Q1, Q2, Q3, Q4\n case 'QQQ':\n return localize.quarter(quarter, {\n width: 'abbreviated',\n context: 'formatting'\n });\n // 1, 2, 3, 4 (narrow quarter; could be not numerical)\n case 'QQQQQ':\n return localize.quarter(quarter, {\n width: 'narrow',\n context: 'formatting'\n });\n // 1st quarter, 2nd quarter, ...\n case 'QQQQ':\n default:\n return localize.quarter(quarter, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // Stand-alone quarter\n q: function q(date, token, localize) {\n var quarter = Math.ceil((date.getUTCMonth() + 1) / 3);\n switch (token) {\n // 1, 2, 3, 4\n case 'q':\n return String(quarter);\n // 01, 02, 03, 04\n case 'qq':\n return addLeadingZeros(quarter, 2);\n // 1st, 2nd, 3rd, 4th\n case 'qo':\n return localize.ordinalNumber(quarter, {\n unit: 'quarter'\n });\n // Q1, Q2, Q3, Q4\n case 'qqq':\n return localize.quarter(quarter, {\n width: 'abbreviated',\n context: 'standalone'\n });\n // 1, 2, 3, 4 (narrow quarter; could be not numerical)\n case 'qqqqq':\n return localize.quarter(quarter, {\n width: 'narrow',\n context: 'standalone'\n });\n // 1st quarter, 2nd quarter, ...\n case 'qqqq':\n default:\n return localize.quarter(quarter, {\n width: 'wide',\n context: 'standalone'\n });\n }\n },\n // Month\n M: function M(date, token, localize) {\n var month = date.getUTCMonth();\n switch (token) {\n case 'M':\n case 'MM':\n return lightFormatters.M(date, token);\n // 1st, 2nd, ..., 12th\n case 'Mo':\n return localize.ordinalNumber(month + 1, {\n unit: 'month'\n });\n // Jan, Feb, ..., Dec\n case 'MMM':\n return localize.month(month, {\n width: 'abbreviated',\n context: 'formatting'\n });\n // J, F, ..., D\n case 'MMMMM':\n return localize.month(month, {\n width: 'narrow',\n context: 'formatting'\n });\n // January, February, ..., December\n case 'MMMM':\n default:\n return localize.month(month, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // Stand-alone month\n L: function L(date, token, localize) {\n var month = date.getUTCMonth();\n switch (token) {\n // 1, 2, ..., 12\n case 'L':\n return String(month + 1);\n // 01, 02, ..., 12\n case 'LL':\n return addLeadingZeros(month + 1, 2);\n // 1st, 2nd, ..., 12th\n case 'Lo':\n return localize.ordinalNumber(month + 1, {\n unit: 'month'\n });\n // Jan, Feb, ..., Dec\n case 'LLL':\n return localize.month(month, {\n width: 'abbreviated',\n context: 'standalone'\n });\n // J, F, ..., D\n case 'LLLLL':\n return localize.month(month, {\n width: 'narrow',\n context: 'standalone'\n });\n // January, February, ..., December\n case 'LLLL':\n default:\n return localize.month(month, {\n width: 'wide',\n context: 'standalone'\n });\n }\n },\n // Local week of year\n w: function w(date, token, localize, options) {\n var week = getUTCWeek(date, options);\n if (token === 'wo') {\n return localize.ordinalNumber(week, {\n unit: 'week'\n });\n }\n return addLeadingZeros(week, token.length);\n },\n // ISO week of year\n I: function I(date, token, localize) {\n var isoWeek = getUTCISOWeek(date);\n if (token === 'Io') {\n return localize.ordinalNumber(isoWeek, {\n unit: 'week'\n });\n }\n return addLeadingZeros(isoWeek, token.length);\n },\n // Day of the month\n d: function d(date, token, localize) {\n if (token === 'do') {\n return localize.ordinalNumber(date.getUTCDate(), {\n unit: 'date'\n });\n }\n return lightFormatters.d(date, token);\n },\n // Day of year\n D: function D(date, token, localize) {\n var dayOfYear = getUTCDayOfYear(date);\n if (token === 'Do') {\n return localize.ordinalNumber(dayOfYear, {\n unit: 'dayOfYear'\n });\n }\n return addLeadingZeros(dayOfYear, token.length);\n },\n // Day of week\n E: function E(date, token, localize) {\n var dayOfWeek = date.getUTCDay();\n switch (token) {\n // Tue\n case 'E':\n case 'EE':\n case 'EEE':\n return localize.day(dayOfWeek, {\n width: 'abbreviated',\n context: 'formatting'\n });\n // T\n case 'EEEEE':\n return localize.day(dayOfWeek, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tu\n case 'EEEEEE':\n return localize.day(dayOfWeek, {\n width: 'short',\n context: 'formatting'\n });\n // Tuesday\n case 'EEEE':\n default:\n return localize.day(dayOfWeek, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // Local day of week\n e: function e(date, token, localize, options) {\n var dayOfWeek = date.getUTCDay();\n var localDayOfWeek = (dayOfWeek - options.weekStartsOn + 8) % 7 || 7;\n switch (token) {\n // Numerical value (Nth day of week with current locale or weekStartsOn)\n case 'e':\n return String(localDayOfWeek);\n // Padded numerical value\n case 'ee':\n return addLeadingZeros(localDayOfWeek, 2);\n // 1st, 2nd, ..., 7th\n case 'eo':\n return localize.ordinalNumber(localDayOfWeek, {\n unit: 'day'\n });\n case 'eee':\n return localize.day(dayOfWeek, {\n width: 'abbreviated',\n context: 'formatting'\n });\n // T\n case 'eeeee':\n return localize.day(dayOfWeek, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tu\n case 'eeeeee':\n return localize.day(dayOfWeek, {\n width: 'short',\n context: 'formatting'\n });\n // Tuesday\n case 'eeee':\n default:\n return localize.day(dayOfWeek, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // Stand-alone local day of week\n c: function c(date, token, localize, options) {\n var dayOfWeek = date.getUTCDay();\n var localDayOfWeek = (dayOfWeek - options.weekStartsOn + 8) % 7 || 7;\n switch (token) {\n // Numerical value (same as in `e`)\n case 'c':\n return String(localDayOfWeek);\n // Padded numerical value\n case 'cc':\n return addLeadingZeros(localDayOfWeek, token.length);\n // 1st, 2nd, ..., 7th\n case 'co':\n return localize.ordinalNumber(localDayOfWeek, {\n unit: 'day'\n });\n case 'ccc':\n return localize.day(dayOfWeek, {\n width: 'abbreviated',\n context: 'standalone'\n });\n // T\n case 'ccccc':\n return localize.day(dayOfWeek, {\n width: 'narrow',\n context: 'standalone'\n });\n // Tu\n case 'cccccc':\n return localize.day(dayOfWeek, {\n width: 'short',\n context: 'standalone'\n });\n // Tuesday\n case 'cccc':\n default:\n return localize.day(dayOfWeek, {\n width: 'wide',\n context: 'standalone'\n });\n }\n },\n // ISO day of week\n i: function i(date, token, localize) {\n var dayOfWeek = date.getUTCDay();\n var isoDayOfWeek = dayOfWeek === 0 ? 7 : dayOfWeek;\n switch (token) {\n // 2\n case 'i':\n return String(isoDayOfWeek);\n // 02\n case 'ii':\n return addLeadingZeros(isoDayOfWeek, token.length);\n // 2nd\n case 'io':\n return localize.ordinalNumber(isoDayOfWeek, {\n unit: 'day'\n });\n // Tue\n case 'iii':\n return localize.day(dayOfWeek, {\n width: 'abbreviated',\n context: 'formatting'\n });\n // T\n case 'iiiii':\n return localize.day(dayOfWeek, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tu\n case 'iiiiii':\n return localize.day(dayOfWeek, {\n width: 'short',\n context: 'formatting'\n });\n // Tuesday\n case 'iiii':\n default:\n return localize.day(dayOfWeek, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // AM or PM\n a: function a(date, token, localize) {\n var hours = date.getUTCHours();\n var dayPeriodEnumValue = hours / 12 >= 1 ? 'pm' : 'am';\n switch (token) {\n case 'a':\n case 'aa':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'abbreviated',\n context: 'formatting'\n });\n case 'aaa':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'abbreviated',\n context: 'formatting'\n }).toLowerCase();\n case 'aaaaa':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'aaaa':\n default:\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // AM, PM, midnight, noon\n b: function b(date, token, localize) {\n var hours = date.getUTCHours();\n var dayPeriodEnumValue;\n if (hours === 12) {\n dayPeriodEnumValue = dayPeriodEnum.noon;\n } else if (hours === 0) {\n dayPeriodEnumValue = dayPeriodEnum.midnight;\n } else {\n dayPeriodEnumValue = hours / 12 >= 1 ? 'pm' : 'am';\n }\n switch (token) {\n case 'b':\n case 'bb':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'abbreviated',\n context: 'formatting'\n });\n case 'bbb':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'abbreviated',\n context: 'formatting'\n }).toLowerCase();\n case 'bbbbb':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'bbbb':\n default:\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // in the morning, in the afternoon, in the evening, at night\n B: function B(date, token, localize) {\n var hours = date.getUTCHours();\n var dayPeriodEnumValue;\n if (hours >= 17) {\n dayPeriodEnumValue = dayPeriodEnum.evening;\n } else if (hours >= 12) {\n dayPeriodEnumValue = dayPeriodEnum.afternoon;\n } else if (hours >= 4) {\n dayPeriodEnumValue = dayPeriodEnum.morning;\n } else {\n dayPeriodEnumValue = dayPeriodEnum.night;\n }\n switch (token) {\n case 'B':\n case 'BB':\n case 'BBB':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'abbreviated',\n context: 'formatting'\n });\n case 'BBBBB':\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'BBBB':\n default:\n return localize.dayPeriod(dayPeriodEnumValue, {\n width: 'wide',\n context: 'formatting'\n });\n }\n },\n // Hour [1-12]\n h: function h(date, token, localize) {\n if (token === 'ho') {\n var hours = date.getUTCHours() % 12;\n if (hours === 0) hours = 12;\n return localize.ordinalNumber(hours, {\n unit: 'hour'\n });\n }\n return lightFormatters.h(date, token);\n },\n // Hour [0-23]\n H: function H(date, token, localize) {\n if (token === 'Ho') {\n return localize.ordinalNumber(date.getUTCHours(), {\n unit: 'hour'\n });\n }\n return lightFormatters.H(date, token);\n },\n // Hour [0-11]\n K: function K(date, token, localize) {\n var hours = date.getUTCHours() % 12;\n if (token === 'Ko') {\n return localize.ordinalNumber(hours, {\n unit: 'hour'\n });\n }\n return addLeadingZeros(hours, token.length);\n },\n // Hour [1-24]\n k: function k(date, token, localize) {\n var hours = date.getUTCHours();\n if (hours === 0) hours = 24;\n if (token === 'ko') {\n return localize.ordinalNumber(hours, {\n unit: 'hour'\n });\n }\n return addLeadingZeros(hours, token.length);\n },\n // Minute\n m: function m(date, token, localize) {\n if (token === 'mo') {\n return localize.ordinalNumber(date.getUTCMinutes(), {\n unit: 'minute'\n });\n }\n return lightFormatters.m(date, token);\n },\n // Second\n s: function s(date, token, localize) {\n if (token === 'so') {\n return localize.ordinalNumber(date.getUTCSeconds(), {\n unit: 'second'\n });\n }\n return lightFormatters.s(date, token);\n },\n // Fraction of second\n S: function S(date, token) {\n return lightFormatters.S(date, token);\n },\n // Timezone (ISO-8601. If offset is 0, output is always `'Z'`)\n X: function X(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timezoneOffset = originalDate.getTimezoneOffset();\n if (timezoneOffset === 0) {\n return 'Z';\n }\n switch (token) {\n // Hours and optional minutes\n case 'X':\n return formatTimezoneWithOptionalMinutes(timezoneOffset);\n\n // Hours, minutes and optional seconds without `:` delimiter\n // Note: neither ISO-8601 nor JavaScript supports seconds in timezone offsets\n // so this token always has the same output as `XX`\n case 'XXXX':\n case 'XX':\n // Hours and minutes without `:` delimiter\n return formatTimezone(timezoneOffset);\n\n // Hours, minutes and optional seconds with `:` delimiter\n // Note: neither ISO-8601 nor JavaScript supports seconds in timezone offsets\n // so this token always has the same output as `XXX`\n case 'XXXXX':\n case 'XXX': // Hours and minutes with `:` delimiter\n default:\n return formatTimezone(timezoneOffset, ':');\n }\n },\n // Timezone (ISO-8601. If offset is 0, output is `'+00:00'` or equivalent)\n x: function x(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timezoneOffset = originalDate.getTimezoneOffset();\n switch (token) {\n // Hours and optional minutes\n case 'x':\n return formatTimezoneWithOptionalMinutes(timezoneOffset);\n\n // Hours, minutes and optional seconds without `:` delimiter\n // Note: neither ISO-8601 nor JavaScript supports seconds in timezone offsets\n // so this token always has the same output as `xx`\n case 'xxxx':\n case 'xx':\n // Hours and minutes without `:` delimiter\n return formatTimezone(timezoneOffset);\n\n // Hours, minutes and optional seconds with `:` delimiter\n // Note: neither ISO-8601 nor JavaScript supports seconds in timezone offsets\n // so this token always has the same output as `xxx`\n case 'xxxxx':\n case 'xxx': // Hours and minutes with `:` delimiter\n default:\n return formatTimezone(timezoneOffset, ':');\n }\n },\n // Timezone (GMT)\n O: function O(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timezoneOffset = originalDate.getTimezoneOffset();\n switch (token) {\n // Short\n case 'O':\n case 'OO':\n case 'OOO':\n return 'GMT' + formatTimezoneShort(timezoneOffset, ':');\n // Long\n case 'OOOO':\n default:\n return 'GMT' + formatTimezone(timezoneOffset, ':');\n }\n },\n // Timezone (specific non-location)\n z: function z(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timezoneOffset = originalDate.getTimezoneOffset();\n switch (token) {\n // Short\n case 'z':\n case 'zz':\n case 'zzz':\n return 'GMT' + formatTimezoneShort(timezoneOffset, ':');\n // Long\n case 'zzzz':\n default:\n return 'GMT' + formatTimezone(timezoneOffset, ':');\n }\n },\n // Seconds timestamp\n t: function t(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timestamp = Math.floor(originalDate.getTime() / 1000);\n return addLeadingZeros(timestamp, token.length);\n },\n // Milliseconds timestamp\n T: function T(date, token, _localize, options) {\n var originalDate = options._originalDate || date;\n var timestamp = originalDate.getTime();\n return addLeadingZeros(timestamp, token.length);\n }\n};\nfunction formatTimezoneShort(offset, dirtyDelimiter) {\n var sign = offset > 0 ? '-' : '+';\n var absOffset = Math.abs(offset);\n var hours = Math.floor(absOffset / 60);\n var minutes = absOffset % 60;\n if (minutes === 0) {\n return sign + String(hours);\n }\n var delimiter = dirtyDelimiter || '';\n return sign + String(hours) + delimiter + addLeadingZeros(minutes, 2);\n}\nfunction formatTimezoneWithOptionalMinutes(offset, dirtyDelimiter) {\n if (offset % 60 === 0) {\n var sign = offset > 0 ? '-' : '+';\n return sign + addLeadingZeros(Math.abs(offset) / 60, 2);\n }\n return formatTimezone(offset, dirtyDelimiter);\n}\nfunction formatTimezone(offset, dirtyDelimiter) {\n var delimiter = dirtyDelimiter || '';\n var sign = offset > 0 ? '-' : '+';\n var absOffset = Math.abs(offset);\n var hours = addLeadingZeros(Math.floor(absOffset / 60), 2);\n var minutes = addLeadingZeros(absOffset % 60, 2);\n return sign + hours + delimiter + minutes;\n}\nexport default formatters;","import isValid from \"../isValid/index.js\";\nimport subMilliseconds from \"../subMilliseconds/index.js\";\nimport toDate from \"../toDate/index.js\";\nimport formatters from \"../_lib/format/formatters/index.js\";\nimport longFormatters from \"../_lib/format/longFormatters/index.js\";\nimport getTimezoneOffsetInMilliseconds from \"../_lib/getTimezoneOffsetInMilliseconds/index.js\";\nimport { isProtectedDayOfYearToken, isProtectedWeekYearToken, throwProtectedError } from \"../_lib/protectedTokens/index.js\";\nimport toInteger from \"../_lib/toInteger/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\nimport { getDefaultOptions } from \"../_lib/defaultOptions/index.js\";\nimport defaultLocale from \"../_lib/defaultLocale/index.js\"; // This RegExp consists of three parts separated by `|`:\n// - [yYQqMLwIdDecihHKkms]o matches any available ordinal number token\n// (one of the certain letters followed by `o`)\n// - (\\w)\\1* matches any sequences of the same letter\n// - '' matches two quote characters in a row\n// - '(''|[^'])+('|$) matches anything surrounded by two quote characters ('),\n// except a single quote symbol, which ends the sequence.\n// Two quote characters do not end the sequence.\n// If there is no matching single quote\n// then the sequence will continue until the end of the string.\n// - . matches any single character unmatched by previous parts of the RegExps\nvar formattingTokensRegExp = /[yYQqMLwIdDecihHKkms]o|(\\w)\\1*|''|'(''|[^'])+('|$)|./g;\n\n// This RegExp catches symbols escaped by quotes, and also\n// sequences of symbols P, p, and the combinations like `PPPPPPPppppp`\nvar longFormattingTokensRegExp = /P+p+|P+|p+|''|'(''|[^'])+('|$)|./g;\nvar escapedStringRegExp = /^'([^]*?)'?$/;\nvar doubleQuoteRegExp = /''/g;\nvar unescapedLatinCharacterRegExp = /[a-zA-Z]/;\n\n/**\n * @name format\n * @category Common Helpers\n * @summary Format the date.\n *\n * @description\n * Return the formatted date string in the given format. The result may vary by locale.\n *\n * > ⚠️ Please note that the `format` tokens differ from Moment.js and other libraries.\n * > See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * The characters wrapped between two single quotes characters (') are escaped.\n * Two single quotes in a row, whether inside or outside a quoted sequence, represent a 'real' single quote.\n * (see the last example)\n *\n * Format of the string is based on Unicode Technical Standard #35:\n * https://www.unicode.org/reports/tr35/tr35-dates.html#Date_Field_Symbol_Table\n * with a few additions (see note 7 below the table).\n *\n * Accepted patterns:\n * | Unit | Pattern | Result examples | Notes |\n * |---------------------------------|---------|-----------------------------------|-------|\n * | Era | G..GGG | AD, BC | |\n * | | GGGG | Anno Domini, Before Christ | 2 |\n * | | GGGGG | A, B | |\n * | Calendar year | y | 44, 1, 1900, 2017 | 5 |\n * | | yo | 44th, 1st, 0th, 17th | 5,7 |\n * | | yy | 44, 01, 00, 17 | 5 |\n * | | yyy | 044, 001, 1900, 2017 | 5 |\n * | | yyyy | 0044, 0001, 1900, 2017 | 5 |\n * | | yyyyy | ... | 3,5 |\n * | Local week-numbering year | Y | 44, 1, 1900, 2017 | 5 |\n * | | Yo | 44th, 1st, 1900th, 2017th | 5,7 |\n * | | YY | 44, 01, 00, 17 | 5,8 |\n * | | YYY | 044, 001, 1900, 2017 | 5 |\n * | | YYYY | 0044, 0001, 1900, 2017 | 5,8 |\n * | | YYYYY | ... | 3,5 |\n * | ISO week-numbering year | R | -43, 0, 1, 1900, 2017 | 5,7 |\n * | | RR | -43, 00, 01, 1900, 2017 | 5,7 |\n * | | RRR | -043, 000, 001, 1900, 2017 | 5,7 |\n * | | RRRR | -0043, 0000, 0001, 1900, 2017 | 5,7 |\n * | | RRRRR | ... | 3,5,7 |\n * | Extended year | u | -43, 0, 1, 1900, 2017 | 5 |\n * | | uu | -43, 01, 1900, 2017 | 5 |\n * | | uuu | -043, 001, 1900, 2017 | 5 |\n * | | uuuu | -0043, 0001, 1900, 2017 | 5 |\n * | | uuuuu | ... | 3,5 |\n * | Quarter (formatting) | Q | 1, 2, 3, 4 | |\n * | | Qo | 1st, 2nd, 3rd, 4th | 7 |\n * | | QQ | 01, 02, 03, 04 | |\n * | | QQQ | Q1, Q2, Q3, Q4 | |\n * | | QQQQ | 1st quarter, 2nd quarter, ... | 2 |\n * | | QQQQQ | 1, 2, 3, 4 | 4 |\n * | Quarter (stand-alone) | q | 1, 2, 3, 4 | |\n * | | qo | 1st, 2nd, 3rd, 4th | 7 |\n * | | qq | 01, 02, 03, 04 | |\n * | | qqq | Q1, Q2, Q3, Q4 | |\n * | | qqqq | 1st quarter, 2nd quarter, ... | 2 |\n * | | qqqqq | 1, 2, 3, 4 | 4 |\n * | Month (formatting) | M | 1, 2, ..., 12 | |\n * | | Mo | 1st, 2nd, ..., 12th | 7 |\n * | | MM | 01, 02, ..., 12 | |\n * | | MMM | Jan, Feb, ..., Dec | |\n * | | MMMM | January, February, ..., December | 2 |\n * | | MMMMM | J, F, ..., D | |\n * | Month (stand-alone) | L | 1, 2, ..., 12 | |\n * | | Lo | 1st, 2nd, ..., 12th | 7 |\n * | | LL | 01, 02, ..., 12 | |\n * | | LLL | Jan, Feb, ..., Dec | |\n * | | LLLL | January, February, ..., December | 2 |\n * | | LLLLL | J, F, ..., D | |\n * | Local week of year | w | 1, 2, ..., 53 | |\n * | | wo | 1st, 2nd, ..., 53th | 7 |\n * | | ww | 01, 02, ..., 53 | |\n * | ISO week of year | I | 1, 2, ..., 53 | 7 |\n * | | Io | 1st, 2nd, ..., 53th | 7 |\n * | | II | 01, 02, ..., 53 | 7 |\n * | Day of month | d | 1, 2, ..., 31 | |\n * | | do | 1st, 2nd, ..., 31st | 7 |\n * | | dd | 01, 02, ..., 31 | |\n * | Day of year | D | 1, 2, ..., 365, 366 | 9 |\n * | | Do | 1st, 2nd, ..., 365th, 366th | 7 |\n * | | DD | 01, 02, ..., 365, 366 | 9 |\n * | | DDD | 001, 002, ..., 365, 366 | |\n * | | DDDD | ... | 3 |\n * | Day of week (formatting) | E..EEE | Mon, Tue, Wed, ..., Sun | |\n * | | EEEE | Monday, Tuesday, ..., Sunday | 2 |\n * | | EEEEE | M, T, W, T, F, S, S | |\n * | | EEEEEE | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | ISO day of week (formatting) | i | 1, 2, 3, ..., 7 | 7 |\n * | | io | 1st, 2nd, ..., 7th | 7 |\n * | | ii | 01, 02, ..., 07 | 7 |\n * | | iii | Mon, Tue, Wed, ..., Sun | 7 |\n * | | iiii | Monday, Tuesday, ..., Sunday | 2,7 |\n * | | iiiii | M, T, W, T, F, S, S | 7 |\n * | | iiiiii | Mo, Tu, We, Th, Fr, Sa, Su | 7 |\n * | Local day of week (formatting) | e | 2, 3, 4, ..., 1 | |\n * | | eo | 2nd, 3rd, ..., 1st | 7 |\n * | | ee | 02, 03, ..., 01 | |\n * | | eee | Mon, Tue, Wed, ..., Sun | |\n * | | eeee | Monday, Tuesday, ..., Sunday | 2 |\n * | | eeeee | M, T, W, T, F, S, S | |\n * | | eeeeee | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | Local day of week (stand-alone) | c | 2, 3, 4, ..., 1 | |\n * | | co | 2nd, 3rd, ..., 1st | 7 |\n * | | cc | 02, 03, ..., 01 | |\n * | | ccc | Mon, Tue, Wed, ..., Sun | |\n * | | cccc | Monday, Tuesday, ..., Sunday | 2 |\n * | | ccccc | M, T, W, T, F, S, S | |\n * | | cccccc | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | AM, PM | a..aa | AM, PM | |\n * | | aaa | am, pm | |\n * | | aaaa | a.m., p.m. | 2 |\n * | | aaaaa | a, p | |\n * | AM, PM, noon, midnight | b..bb | AM, PM, noon, midnight | |\n * | | bbb | am, pm, noon, midnight | |\n * | | bbbb | a.m., p.m., noon, midnight | 2 |\n * | | bbbbb | a, p, n, mi | |\n * | Flexible day period | B..BBB | at night, in the morning, ... | |\n * | | BBBB | at night, in the morning, ... | 2 |\n * | | BBBBB | at night, in the morning, ... | |\n * | Hour [1-12] | h | 1, 2, ..., 11, 12 | |\n * | | ho | 1st, 2nd, ..., 11th, 12th | 7 |\n * | | hh | 01, 02, ..., 11, 12 | |\n * | Hour [0-23] | H | 0, 1, 2, ..., 23 | |\n * | | Ho | 0th, 1st, 2nd, ..., 23rd | 7 |\n * | | HH | 00, 01, 02, ..., 23 | |\n * | Hour [0-11] | K | 1, 2, ..., 11, 0 | |\n * | | Ko | 1st, 2nd, ..., 11th, 0th | 7 |\n * | | KK | 01, 02, ..., 11, 00 | |\n * | Hour [1-24] | k | 24, 1, 2, ..., 23 | |\n * | | ko | 24th, 1st, 2nd, ..., 23rd | 7 |\n * | | kk | 24, 01, 02, ..., 23 | |\n * | Minute | m | 0, 1, ..., 59 | |\n * | | mo | 0th, 1st, ..., 59th | 7 |\n * | | mm | 00, 01, ..., 59 | |\n * | Second | s | 0, 1, ..., 59 | |\n * | | so | 0th, 1st, ..., 59th | 7 |\n * | | ss | 00, 01, ..., 59 | |\n * | Fraction of second | S | 0, 1, ..., 9 | |\n * | | SS | 00, 01, ..., 99 | |\n * | | SSS | 000, 001, ..., 999 | |\n * | | SSSS | ... | 3 |\n * | Timezone (ISO-8601 w/ Z) | X | -08, +0530, Z | |\n * | | XX | -0800, +0530, Z | |\n * | | XXX | -08:00, +05:30, Z | |\n * | | XXXX | -0800, +0530, Z, +123456 | 2 |\n * | | XXXXX | -08:00, +05:30, Z, +12:34:56 | |\n * | Timezone (ISO-8601 w/o Z) | x | -08, +0530, +00 | |\n * | | xx | -0800, +0530, +0000 | |\n * | | xxx | -08:00, +05:30, +00:00 | 2 |\n * | | xxxx | -0800, +0530, +0000, +123456 | |\n * | | xxxxx | -08:00, +05:30, +00:00, +12:34:56 | |\n * | Timezone (GMT) | O...OOO | GMT-8, GMT+5:30, GMT+0 | |\n * | | OOOO | GMT-08:00, GMT+05:30, GMT+00:00 | 2 |\n * | Timezone (specific non-locat.) | z...zzz | GMT-8, GMT+5:30, GMT+0 | 6 |\n * | | zzzz | GMT-08:00, GMT+05:30, GMT+00:00 | 2,6 |\n * | Seconds timestamp | t | 512969520 | 7 |\n * | | tt | ... | 3,7 |\n * | Milliseconds timestamp | T | 512969520900 | 7 |\n * | | TT | ... | 3,7 |\n * | Long localized date | P | 04/29/1453 | 7 |\n * | | PP | Apr 29, 1453 | 7 |\n * | | PPP | April 29th, 1453 | 7 |\n * | | PPPP | Friday, April 29th, 1453 | 2,7 |\n * | Long localized time | p | 12:00 AM | 7 |\n * | | pp | 12:00:00 AM | 7 |\n * | | ppp | 12:00:00 AM GMT+2 | 7 |\n * | | pppp | 12:00:00 AM GMT+02:00 | 2,7 |\n * | Combination of date and time | Pp | 04/29/1453, 12:00 AM | 7 |\n * | | PPpp | Apr 29, 1453, 12:00:00 AM | 7 |\n * | | PPPppp | April 29th, 1453 at ... | 7 |\n * | | PPPPpppp| Friday, April 29th, 1453 at ... | 2,7 |\n * Notes:\n * 1. \"Formatting\" units (e.g. formatting quarter) in the default en-US locale\n * are the same as \"stand-alone\" units, but are different in some languages.\n * \"Formatting\" units are declined according to the rules of the language\n * in the context of a date. \"Stand-alone\" units are always nominative singular:\n *\n * `format(new Date(2017, 10, 6), 'do LLLL', {locale: cs}) //=> '6. listopad'`\n *\n * `format(new Date(2017, 10, 6), 'do MMMM', {locale: cs}) //=> '6. listopadu'`\n *\n * 2. Any sequence of the identical letters is a pattern, unless it is escaped by\n * the single quote characters (see below).\n * If the sequence is longer than listed in table (e.g. `EEEEEEEEEEE`)\n * the output will be the same as default pattern for this unit, usually\n * the longest one (in case of ISO weekdays, `EEEE`). Default patterns for units\n * are marked with \"2\" in the last column of the table.\n *\n * `format(new Date(2017, 10, 6), 'MMM') //=> 'Nov'`\n *\n * `format(new Date(2017, 10, 6), 'MMMM') //=> 'November'`\n *\n * `format(new Date(2017, 10, 6), 'MMMMM') //=> 'N'`\n *\n * `format(new Date(2017, 10, 6), 'MMMMMM') //=> 'November'`\n *\n * `format(new Date(2017, 10, 6), 'MMMMMMM') //=> 'November'`\n *\n * 3. Some patterns could be unlimited length (such as `yyyyyyyy`).\n * The output will be padded with zeros to match the length of the pattern.\n *\n * `format(new Date(2017, 10, 6), 'yyyyyyyy') //=> '00002017'`\n *\n * 4. `QQQQQ` and `qqqqq` could be not strictly numerical in some locales.\n * These tokens represent the shortest form of the quarter.\n *\n * 5. The main difference between `y` and `u` patterns are B.C. years:\n *\n * | Year | `y` | `u` |\n * |------|-----|-----|\n * | AC 1 | 1 | 1 |\n * | BC 1 | 1 | 0 |\n * | BC 2 | 2 | -1 |\n *\n * Also `yy` always returns the last two digits of a year,\n * while `uu` pads single digit years to 2 characters and returns other years unchanged:\n *\n * | Year | `yy` | `uu` |\n * |------|------|------|\n * | 1 | 01 | 01 |\n * | 14 | 14 | 14 |\n * | 376 | 76 | 376 |\n * | 1453 | 53 | 1453 |\n *\n * The same difference is true for local and ISO week-numbering years (`Y` and `R`),\n * except local week-numbering years are dependent on `options.weekStartsOn`\n * and `options.firstWeekContainsDate` (compare [getISOWeekYear]{@link https://date-fns.org/docs/getISOWeekYear}\n * and [getWeekYear]{@link https://date-fns.org/docs/getWeekYear}).\n *\n * 6. Specific non-location timezones are currently unavailable in `date-fns`,\n * so right now these tokens fall back to GMT timezones.\n *\n * 7. These patterns are not in the Unicode Technical Standard #35:\n * - `i`: ISO day of week\n * - `I`: ISO week of year\n * - `R`: ISO week-numbering year\n * - `t`: seconds timestamp\n * - `T`: milliseconds timestamp\n * - `o`: ordinal number modifier\n * - `P`: long localized date\n * - `p`: long localized time\n *\n * 8. `YY` and `YYYY` tokens represent week-numbering years but they are often confused with years.\n * You should enable `options.useAdditionalWeekYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * 9. `D` and `DD` tokens represent days of the year but they are often confused with days of the month.\n * You should enable `options.useAdditionalDayOfYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * @param {Date|Number} date - the original date\n * @param {String} format - the string of tokens\n * @param {Object} [options] - an object with options.\n * @param {Locale} [options.locale=defaultLocale] - the locale object. See [Locale]{@link https://date-fns.org/docs/Locale}\n * @param {0|1|2|3|4|5|6} [options.weekStartsOn=0] - the index of the first day of the week (0 - Sunday)\n * @param {Number} [options.firstWeekContainsDate=1] - the day of January, which is\n * @param {Boolean} [options.useAdditionalWeekYearTokens=false] - if true, allows usage of the week-numbering year tokens `YY` and `YYYY`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @param {Boolean} [options.useAdditionalDayOfYearTokens=false] - if true, allows usage of the day of year tokens `D` and `DD`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @returns {String} the formatted date string\n * @throws {TypeError} 2 arguments required\n * @throws {RangeError} `date` must not be Invalid Date\n * @throws {RangeError} `options.locale` must contain `localize` property\n * @throws {RangeError} `options.locale` must contain `formatLong` property\n * @throws {RangeError} `options.weekStartsOn` must be between 0 and 6\n * @throws {RangeError} `options.firstWeekContainsDate` must be between 1 and 7\n * @throws {RangeError} use `yyyy` instead of `YYYY` for formatting years using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `yy` instead of `YY` for formatting years using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `d` instead of `D` for formatting days of the month using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `dd` instead of `DD` for formatting days of the month using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} format string contains an unescaped latin alphabet character\n *\n * @example\n * // Represent 11 February 2014 in middle-endian format:\n * const result = format(new Date(2014, 1, 11), 'MM/dd/yyyy')\n * //=> '02/11/2014'\n *\n * @example\n * // Represent 2 July 2014 in Esperanto:\n * import { eoLocale } from 'date-fns/locale/eo'\n * const result = format(new Date(2014, 6, 2), \"do 'de' MMMM yyyy\", {\n * locale: eoLocale\n * })\n * //=> '2-a de julio 2014'\n *\n * @example\n * // Escape string by single quote characters:\n * const result = format(new Date(2014, 6, 2, 15), \"h 'o''clock'\")\n * //=> \"3 o'clock\"\n */\n\nexport default function format(dirtyDate, dirtyFormatStr, options) {\n var _ref, _options$locale, _ref2, _ref3, _ref4, _options$firstWeekCon, _options$locale2, _options$locale2$opti, _defaultOptions$local, _defaultOptions$local2, _ref5, _ref6, _ref7, _options$weekStartsOn, _options$locale3, _options$locale3$opti, _defaultOptions$local3, _defaultOptions$local4;\n requiredArgs(2, arguments);\n var formatStr = String(dirtyFormatStr);\n var defaultOptions = getDefaultOptions();\n var locale = (_ref = (_options$locale = options === null || options === void 0 ? void 0 : options.locale) !== null && _options$locale !== void 0 ? _options$locale : defaultOptions.locale) !== null && _ref !== void 0 ? _ref : defaultLocale;\n var firstWeekContainsDate = toInteger((_ref2 = (_ref3 = (_ref4 = (_options$firstWeekCon = options === null || options === void 0 ? void 0 : options.firstWeekContainsDate) !== null && _options$firstWeekCon !== void 0 ? _options$firstWeekCon : options === null || options === void 0 ? void 0 : (_options$locale2 = options.locale) === null || _options$locale2 === void 0 ? void 0 : (_options$locale2$opti = _options$locale2.options) === null || _options$locale2$opti === void 0 ? void 0 : _options$locale2$opti.firstWeekContainsDate) !== null && _ref4 !== void 0 ? _ref4 : defaultOptions.firstWeekContainsDate) !== null && _ref3 !== void 0 ? _ref3 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.firstWeekContainsDate) !== null && _ref2 !== void 0 ? _ref2 : 1);\n\n // Test if weekStartsOn is between 1 and 7 _and_ is not NaN\n if (!(firstWeekContainsDate >= 1 && firstWeekContainsDate <= 7)) {\n throw new RangeError('firstWeekContainsDate must be between 1 and 7 inclusively');\n }\n var weekStartsOn = toInteger((_ref5 = (_ref6 = (_ref7 = (_options$weekStartsOn = options === null || options === void 0 ? void 0 : options.weekStartsOn) !== null && _options$weekStartsOn !== void 0 ? _options$weekStartsOn : options === null || options === void 0 ? void 0 : (_options$locale3 = options.locale) === null || _options$locale3 === void 0 ? void 0 : (_options$locale3$opti = _options$locale3.options) === null || _options$locale3$opti === void 0 ? void 0 : _options$locale3$opti.weekStartsOn) !== null && _ref7 !== void 0 ? _ref7 : defaultOptions.weekStartsOn) !== null && _ref6 !== void 0 ? _ref6 : (_defaultOptions$local3 = defaultOptions.locale) === null || _defaultOptions$local3 === void 0 ? void 0 : (_defaultOptions$local4 = _defaultOptions$local3.options) === null || _defaultOptions$local4 === void 0 ? void 0 : _defaultOptions$local4.weekStartsOn) !== null && _ref5 !== void 0 ? _ref5 : 0);\n\n // Test if weekStartsOn is between 0 and 6 _and_ is not NaN\n if (!(weekStartsOn >= 0 && weekStartsOn <= 6)) {\n throw new RangeError('weekStartsOn must be between 0 and 6 inclusively');\n }\n if (!locale.localize) {\n throw new RangeError('locale must contain localize property');\n }\n if (!locale.formatLong) {\n throw new RangeError('locale must contain formatLong property');\n }\n var originalDate = toDate(dirtyDate);\n if (!isValid(originalDate)) {\n throw new RangeError('Invalid time value');\n }\n\n // Convert the date in system timezone to the same date in UTC+00:00 timezone.\n // This ensures that when UTC functions will be implemented, locales will be compatible with them.\n // See an issue about UTC functions: https://github.com/date-fns/date-fns/issues/376\n var timezoneOffset = getTimezoneOffsetInMilliseconds(originalDate);\n var utcDate = subMilliseconds(originalDate, timezoneOffset);\n var formatterOptions = {\n firstWeekContainsDate: firstWeekContainsDate,\n weekStartsOn: weekStartsOn,\n locale: locale,\n _originalDate: originalDate\n };\n var result = formatStr.match(longFormattingTokensRegExp).map(function (substring) {\n var firstCharacter = substring[0];\n if (firstCharacter === 'p' || firstCharacter === 'P') {\n var longFormatter = longFormatters[firstCharacter];\n return longFormatter(substring, locale.formatLong);\n }\n return substring;\n }).join('').match(formattingTokensRegExp).map(function (substring) {\n // Replace two single quote characters with one single quote character\n if (substring === \"''\") {\n return \"'\";\n }\n var firstCharacter = substring[0];\n if (firstCharacter === \"'\") {\n return cleanEscapedString(substring);\n }\n var formatter = formatters[firstCharacter];\n if (formatter) {\n if (!(options !== null && options !== void 0 && options.useAdditionalWeekYearTokens) && isProtectedWeekYearToken(substring)) {\n throwProtectedError(substring, dirtyFormatStr, String(dirtyDate));\n }\n if (!(options !== null && options !== void 0 && options.useAdditionalDayOfYearTokens) && isProtectedDayOfYearToken(substring)) {\n throwProtectedError(substring, dirtyFormatStr, String(dirtyDate));\n }\n return formatter(utcDate, substring, locale.localize, formatterOptions);\n }\n if (firstCharacter.match(unescapedLatinCharacterRegExp)) {\n throw new RangeError('Format string contains an unescaped latin alphabet character `' + firstCharacter + '`');\n }\n return substring;\n }).join('');\n return result;\n}\nfunction cleanEscapedString(input) {\n var matched = input.match(escapedStringRegExp);\n if (!matched) {\n return input;\n }\n return matched[1].replace(doubleQuoteRegExp, \"'\");\n}","import toDate from \"../toDate/index.js\";\nimport addLeadingZeros from \"../_lib/addLeadingZeros/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name formatISO\n * @category Common Helpers\n * @summary Format the date according to the ISO 8601 standard (https://support.sas.com/documentation/cdl/en/lrdict/64316/HTML/default/viewer.htm#a003169814.htm).\n *\n * @description\n * Return the formatted date string in ISO 8601 format. Options may be passed to control the parts and notations of the date.\n *\n * @param {Date|Number} date - the original date\n * @param {Object} [options] - an object with options.\n * @param {'extended'|'basic'} [options.format='extended'] - if 'basic', hide delimiters between date and time values.\n * @param {'complete'|'date'|'time'} [options.representation='complete'] - format date, time with local time zone, or both.\n * @returns {String} the formatted date string (in local time zone)\n * @throws {TypeError} 1 argument required\n * @throws {RangeError} `date` must not be Invalid Date\n * @throws {RangeError} `options.format` must be 'extended' or 'basic'\n * @throws {RangeError} `options.representation` must be 'date', 'time' or 'complete'\n *\n * @example\n * // Represent 18 September 2019 in ISO 8601 format (local time zone is UTC):\n * const result = formatISO(new Date(2019, 8, 18, 19, 0, 52))\n * //=> '2019-09-18T19:00:52Z'\n *\n * @example\n * // Represent 18 September 2019 in ISO 8601, short format (local time zone is UTC):\n * const result = formatISO(new Date(2019, 8, 18, 19, 0, 52), { format: 'basic' })\n * //=> '20190918T190052'\n *\n * @example\n * // Represent 18 September 2019 in ISO 8601 format, date only:\n * const result = formatISO(new Date(2019, 8, 18, 19, 0, 52), { representation: 'date' })\n * //=> '2019-09-18'\n *\n * @example\n * // Represent 18 September 2019 in ISO 8601 format, time only (local time zone is UTC):\n * const result = formatISO(new Date(2019, 8, 18, 19, 0, 52), { representation: 'time' })\n * //=> '19:00:52Z'\n */\nexport default function formatISO(date, options) {\n var _options$format, _options$representati;\n requiredArgs(1, arguments);\n var originalDate = toDate(date);\n if (isNaN(originalDate.getTime())) {\n throw new RangeError('Invalid time value');\n }\n var format = String((_options$format = options === null || options === void 0 ? void 0 : options.format) !== null && _options$format !== void 0 ? _options$format : 'extended');\n var representation = String((_options$representati = options === null || options === void 0 ? void 0 : options.representation) !== null && _options$representati !== void 0 ? _options$representati : 'complete');\n if (format !== 'extended' && format !== 'basic') {\n throw new RangeError(\"format must be 'extended' or 'basic'\");\n }\n if (representation !== 'date' && representation !== 'time' && representation !== 'complete') {\n throw new RangeError(\"representation must be 'date', 'time', or 'complete'\");\n }\n var result = '';\n var tzOffset = '';\n var dateDelimiter = format === 'extended' ? '-' : '';\n var timeDelimiter = format === 'extended' ? ':' : '';\n\n // Representation is either 'date' or 'complete'\n if (representation !== 'time') {\n var day = addLeadingZeros(originalDate.getDate(), 2);\n var month = addLeadingZeros(originalDate.getMonth() + 1, 2);\n var year = addLeadingZeros(originalDate.getFullYear(), 4);\n\n // yyyyMMdd or yyyy-MM-dd.\n result = \"\".concat(year).concat(dateDelimiter).concat(month).concat(dateDelimiter).concat(day);\n }\n\n // Representation is either 'time' or 'complete'\n if (representation !== 'date') {\n // Add the timezone.\n var offset = originalDate.getTimezoneOffset();\n if (offset !== 0) {\n var absoluteOffset = Math.abs(offset);\n var hourOffset = addLeadingZeros(Math.floor(absoluteOffset / 60), 2);\n var minuteOffset = addLeadingZeros(absoluteOffset % 60, 2);\n // If less than 0, the sign is +, because it is ahead of time.\n var sign = offset < 0 ? '+' : '-';\n tzOffset = \"\".concat(sign).concat(hourOffset, \":\").concat(minuteOffset);\n } else {\n tzOffset = 'Z';\n }\n var hour = addLeadingZeros(originalDate.getHours(), 2);\n var minute = addLeadingZeros(originalDate.getMinutes(), 2);\n var second = addLeadingZeros(originalDate.getSeconds(), 2);\n\n // If there's also date, separate it with time with 'T'\n var separator = result === '' ? '' : 'T';\n\n // Creates a time string consisting of hour, minute, and second, separated by delimiters, if defined.\n var time = [hour, minute, second].join(timeDelimiter);\n\n // HHmmss or HH:mm:ss.\n result = \"\".concat(result).concat(separator).concat(time).concat(tzOffset);\n }\n return result;\n}","import _typeof from \"@babel/runtime/helpers/esm/typeof\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name isDate\n * @category Common Helpers\n * @summary Is the given value a date?\n *\n * @description\n * Returns true if the given value is an instance of Date. The function works for dates transferred across iframes.\n *\n * @param {*} value - the value to check\n * @returns {boolean} true if the given value is a date\n * @throws {TypeError} 1 arguments required\n *\n * @example\n * // For a valid date:\n * const result = isDate(new Date())\n * //=> true\n *\n * @example\n * // For an invalid date:\n * const result = isDate(new Date(NaN))\n * //=> true\n *\n * @example\n * // For some value:\n * const result = isDate('2014-02-31')\n * //=> false\n *\n * @example\n * // For an object:\n * const result = isDate({})\n * //=> false\n */\nexport default function isDate(value) {\n requiredArgs(1, arguments);\n return value instanceof Date || _typeof(value) === 'object' && Object.prototype.toString.call(value) === '[object Date]';\n}","import isDate from \"../isDate/index.js\";\nimport toDate from \"../toDate/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name isValid\n * @category Common Helpers\n * @summary Is the given date valid?\n *\n * @description\n * Returns false if argument is Invalid Date and true otherwise.\n * Argument is converted to Date using `toDate`. See [toDate]{@link https://date-fns.org/docs/toDate}\n * Invalid Date is a Date, whose time value is NaN.\n *\n * Time value of Date: http://es5.github.io/#x15.9.1.1\n *\n * @param {*} date - the date to check\n * @returns {Boolean} the date is valid\n * @throws {TypeError} 1 argument required\n *\n * @example\n * // For the valid date:\n * const result = isValid(new Date(2014, 1, 31))\n * //=> true\n *\n * @example\n * // For the value, convertable into a date:\n * const result = isValid(1393804800000)\n * //=> true\n *\n * @example\n * // For the invalid date:\n * const result = isValid(new Date(''))\n * //=> false\n */\nexport default function isValid(dirtyDate) {\n requiredArgs(1, arguments);\n if (!isDate(dirtyDate) && typeof dirtyDate !== 'number') {\n return false;\n }\n var date = toDate(dirtyDate);\n return !isNaN(Number(date));\n}","function _arrayLikeToArray(r, a) {\n (null == a || a > r.length) && (a = r.length);\n for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];\n return n;\n}\nexport { _arrayLikeToArray as default };","import unsupportedIterableToArray from \"./unsupportedIterableToArray.js\";\nfunction _createForOfIteratorHelper(r, e) {\n var t = \"undefined\" != typeof Symbol && r[Symbol.iterator] || r[\"@@iterator\"];\n if (!t) {\n if (Array.isArray(r) || (t = unsupportedIterableToArray(r)) || e && r && \"number\" == typeof r.length) {\n t && (r = t);\n var _n = 0,\n F = function F() {};\n return {\n s: F,\n n: function n() {\n return _n >= r.length ? {\n done: !0\n } : {\n done: !1,\n value: r[_n++]\n };\n },\n e: function e(r) {\n throw r;\n },\n f: F\n };\n }\n throw new TypeError(\"Invalid attempt to iterate non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n var o,\n a = !0,\n u = !1;\n return {\n s: function s() {\n t = t.call(r);\n },\n n: function n() {\n var r = t.next();\n return a = r.done, r;\n },\n e: function e(r) {\n u = !0, o = r;\n },\n f: function f() {\n try {\n a || null == t[\"return\"] || t[\"return\"]();\n } finally {\n if (u) throw o;\n }\n }\n };\n}\nexport { _createForOfIteratorHelper as default };","import arrayLikeToArray from \"./arrayLikeToArray.js\";\nfunction _unsupportedIterableToArray(r, a) {\n if (r) {\n if (\"string\" == typeof r) return arrayLikeToArray(r, a);\n var t = {}.toString.call(r).slice(8, -1);\n return \"Object\" === t && r.constructor && (t = r.constructor.name), \"Map\" === t || \"Set\" === t ? Array.from(r) : \"Arguments\" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? arrayLikeToArray(r, a) : void 0;\n }\n}\nexport { _unsupportedIterableToArray as default };","export default function assign(target, object) {\n if (target == null) {\n throw new TypeError('assign requires that input parameter not be null or undefined');\n }\n for (var property in object) {\n if (Object.prototype.hasOwnProperty.call(object, property)) {\n ;\n target[property] = object[property];\n }\n }\n return target;\n}","function _assertThisInitialized(e) {\n if (void 0 === e) throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\");\n return e;\n}\nexport { _assertThisInitialized as default };","function _setPrototypeOf(t, e) {\n return _setPrototypeOf = Object.setPrototypeOf ? Object.setPrototypeOf.bind() : function (t, e) {\n return t.__proto__ = e, t;\n }, _setPrototypeOf(t, e);\n}\nexport { _setPrototypeOf as default };","import setPrototypeOf from \"./setPrototypeOf.js\";\nfunction _inherits(t, e) {\n if (\"function\" != typeof e && null !== e) throw new TypeError(\"Super expression must either be null or a function\");\n t.prototype = Object.create(e && e.prototype, {\n constructor: {\n value: t,\n writable: !0,\n configurable: !0\n }\n }), Object.defineProperty(t, \"prototype\", {\n writable: !1\n }), e && setPrototypeOf(t, e);\n}\nexport { _inherits as default };","function _getPrototypeOf(t) {\n return _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf.bind() : function (t) {\n return t.__proto__ || Object.getPrototypeOf(t);\n }, _getPrototypeOf(t);\n}\nexport { _getPrototypeOf as default };","function _isNativeReflectConstruct() {\n try {\n var t = !Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {}));\n } catch (t) {}\n return (_isNativeReflectConstruct = function _isNativeReflectConstruct() {\n return !!t;\n })();\n}\nexport { _isNativeReflectConstruct as default };","import _typeof from \"./typeof.js\";\nimport assertThisInitialized from \"./assertThisInitialized.js\";\nfunction _possibleConstructorReturn(t, e) {\n if (e && (\"object\" == _typeof(e) || \"function\" == typeof e)) return e;\n if (void 0 !== e) throw new TypeError(\"Derived constructors may only return object or undefined\");\n return assertThisInitialized(t);\n}\nexport { _possibleConstructorReturn as default };","import getPrototypeOf from \"./getPrototypeOf.js\";\nimport isNativeReflectConstruct from \"./isNativeReflectConstruct.js\";\nimport possibleConstructorReturn from \"./possibleConstructorReturn.js\";\nfunction _createSuper(t) {\n var r = isNativeReflectConstruct();\n return function () {\n var e,\n o = getPrototypeOf(t);\n if (r) {\n var s = getPrototypeOf(this).constructor;\n e = Reflect.construct(o, arguments, s);\n } else e = o.apply(this, arguments);\n return possibleConstructorReturn(this, e);\n };\n}\nexport { _createSuper as default };","function _classCallCheck(a, n) {\n if (!(a instanceof n)) throw new TypeError(\"Cannot call a class as a function\");\n}\nexport { _classCallCheck as default };","import toPropertyKey from \"./toPropertyKey.js\";\nfunction _defineProperties(e, r) {\n for (var t = 0; t < r.length; t++) {\n var o = r[t];\n o.enumerable = o.enumerable || !1, o.configurable = !0, \"value\" in o && (o.writable = !0), Object.defineProperty(e, toPropertyKey(o.key), o);\n }\n}\nfunction _createClass(e, r, t) {\n return r && _defineProperties(e.prototype, r), t && _defineProperties(e, t), Object.defineProperty(e, \"prototype\", {\n writable: !1\n }), e;\n}\nexport { _createClass as default };","import _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nvar TIMEZONE_UNIT_PRIORITY = 10;\nexport var Setter = /*#__PURE__*/function () {\n function Setter() {\n _classCallCheck(this, Setter);\n _defineProperty(this, \"priority\", void 0);\n _defineProperty(this, \"subPriority\", 0);\n }\n _createClass(Setter, [{\n key: \"validate\",\n value: function validate(_utcDate, _options) {\n return true;\n }\n }]);\n return Setter;\n}();\nexport var ValueSetter = /*#__PURE__*/function (_Setter) {\n _inherits(ValueSetter, _Setter);\n var _super = _createSuper(ValueSetter);\n function ValueSetter(value, validateValue, setValue, priority, subPriority) {\n var _this;\n _classCallCheck(this, ValueSetter);\n _this = _super.call(this);\n _this.value = value;\n _this.validateValue = validateValue;\n _this.setValue = setValue;\n _this.priority = priority;\n if (subPriority) {\n _this.subPriority = subPriority;\n }\n return _this;\n }\n _createClass(ValueSetter, [{\n key: \"validate\",\n value: function validate(utcDate, options) {\n return this.validateValue(utcDate, this.value, options);\n }\n }, {\n key: \"set\",\n value: function set(utcDate, flags, options) {\n return this.setValue(utcDate, flags, this.value, options);\n }\n }]);\n return ValueSetter;\n}(Setter);\nexport var DateToSystemTimezoneSetter = /*#__PURE__*/function (_Setter2) {\n _inherits(DateToSystemTimezoneSetter, _Setter2);\n var _super2 = _createSuper(DateToSystemTimezoneSetter);\n function DateToSystemTimezoneSetter() {\n var _this2;\n _classCallCheck(this, DateToSystemTimezoneSetter);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this2 = _super2.call.apply(_super2, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this2), \"priority\", TIMEZONE_UNIT_PRIORITY);\n _defineProperty(_assertThisInitialized(_this2), \"subPriority\", -1);\n return _this2;\n }\n _createClass(DateToSystemTimezoneSetter, [{\n key: \"set\",\n value: function set(date, flags) {\n if (flags.timestampIsSet) {\n return date;\n }\n var convertedDate = new Date(0);\n convertedDate.setFullYear(date.getUTCFullYear(), date.getUTCMonth(), date.getUTCDate());\n convertedDate.setHours(date.getUTCHours(), date.getUTCMinutes(), date.getUTCSeconds(), date.getUTCMilliseconds());\n return convertedDate;\n }\n }]);\n return DateToSystemTimezoneSetter;\n}(Setter);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { ValueSetter } from \"./Setter.js\";\nexport var Parser = /*#__PURE__*/function () {\n function Parser() {\n _classCallCheck(this, Parser);\n _defineProperty(this, \"incompatibleTokens\", void 0);\n _defineProperty(this, \"priority\", void 0);\n _defineProperty(this, \"subPriority\", void 0);\n }\n _createClass(Parser, [{\n key: \"run\",\n value: function run(dateString, token, match, options) {\n var result = this.parse(dateString, token, match, options);\n if (!result) {\n return null;\n }\n return {\n setter: new ValueSetter(result.value, this.validate, this.set, this.priority, this.subPriority),\n rest: result.rest\n };\n }\n }, {\n key: \"validate\",\n value: function validate(_utcDate, _value, _options) {\n return true;\n }\n }]);\n return Parser;\n}();","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nexport var EraParser = /*#__PURE__*/function (_Parser) {\n _inherits(EraParser, _Parser);\n var _super = _createSuper(EraParser);\n function EraParser() {\n var _this;\n _classCallCheck(this, EraParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 140);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['R', 'u', 't', 'T']);\n return _this;\n }\n _createClass(EraParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n // AD, BC\n case 'G':\n case 'GG':\n case 'GGG':\n return match.era(dateString, {\n width: 'abbreviated'\n }) || match.era(dateString, {\n width: 'narrow'\n });\n // A, B\n case 'GGGGG':\n return match.era(dateString, {\n width: 'narrow'\n });\n // Anno Domini, Before Christ\n case 'GGGG':\n default:\n return match.era(dateString, {\n width: 'wide'\n }) || match.era(dateString, {\n width: 'abbreviated'\n }) || match.era(dateString, {\n width: 'narrow'\n });\n }\n }\n }, {\n key: \"set\",\n value: function set(date, flags, value) {\n flags.era = value;\n date.setUTCFullYear(value, 0, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return EraParser;\n}(Parser);","export var numericPatterns = {\n month: /^(1[0-2]|0?\\d)/,\n // 0 to 12\n date: /^(3[0-1]|[0-2]?\\d)/,\n // 0 to 31\n dayOfYear: /^(36[0-6]|3[0-5]\\d|[0-2]?\\d?\\d)/,\n // 0 to 366\n week: /^(5[0-3]|[0-4]?\\d)/,\n // 0 to 53\n hour23h: /^(2[0-3]|[0-1]?\\d)/,\n // 0 to 23\n hour24h: /^(2[0-4]|[0-1]?\\d)/,\n // 0 to 24\n hour11h: /^(1[0-1]|0?\\d)/,\n // 0 to 11\n hour12h: /^(1[0-2]|0?\\d)/,\n // 0 to 12\n minute: /^[0-5]?\\d/,\n // 0 to 59\n second: /^[0-5]?\\d/,\n // 0 to 59\n\n singleDigit: /^\\d/,\n // 0 to 9\n twoDigits: /^\\d{1,2}/,\n // 0 to 99\n threeDigits: /^\\d{1,3}/,\n // 0 to 999\n fourDigits: /^\\d{1,4}/,\n // 0 to 9999\n\n anyDigitsSigned: /^-?\\d+/,\n singleDigitSigned: /^-?\\d/,\n // 0 to 9, -0 to -9\n twoDigitsSigned: /^-?\\d{1,2}/,\n // 0 to 99, -0 to -99\n threeDigitsSigned: /^-?\\d{1,3}/,\n // 0 to 999, -0 to -999\n fourDigitsSigned: /^-?\\d{1,4}/ // 0 to 9999, -0 to -9999\n};\n\nexport var timezonePatterns = {\n basicOptionalMinutes: /^([+-])(\\d{2})(\\d{2})?|Z/,\n basic: /^([+-])(\\d{2})(\\d{2})|Z/,\n basicOptionalSeconds: /^([+-])(\\d{2})(\\d{2})((\\d{2}))?|Z/,\n extended: /^([+-])(\\d{2}):(\\d{2})|Z/,\n extendedOptionalSeconds: /^([+-])(\\d{2}):(\\d{2})(:(\\d{2}))?|Z/\n};","import { millisecondsInHour, millisecondsInMinute, millisecondsInSecond } from \"../../constants/index.js\";\nimport { numericPatterns } from \"./constants.js\";\nexport function mapValue(parseFnResult, mapFn) {\n if (!parseFnResult) {\n return parseFnResult;\n }\n return {\n value: mapFn(parseFnResult.value),\n rest: parseFnResult.rest\n };\n}\nexport function parseNumericPattern(pattern, dateString) {\n var matchResult = dateString.match(pattern);\n if (!matchResult) {\n return null;\n }\n return {\n value: parseInt(matchResult[0], 10),\n rest: dateString.slice(matchResult[0].length)\n };\n}\nexport function parseTimezonePattern(pattern, dateString) {\n var matchResult = dateString.match(pattern);\n if (!matchResult) {\n return null;\n }\n\n // Input is 'Z'\n if (matchResult[0] === 'Z') {\n return {\n value: 0,\n rest: dateString.slice(1)\n };\n }\n var sign = matchResult[1] === '+' ? 1 : -1;\n var hours = matchResult[2] ? parseInt(matchResult[2], 10) : 0;\n var minutes = matchResult[3] ? parseInt(matchResult[3], 10) : 0;\n var seconds = matchResult[5] ? parseInt(matchResult[5], 10) : 0;\n return {\n value: sign * (hours * millisecondsInHour + minutes * millisecondsInMinute + seconds * millisecondsInSecond),\n rest: dateString.slice(matchResult[0].length)\n };\n}\nexport function parseAnyDigitsSigned(dateString) {\n return parseNumericPattern(numericPatterns.anyDigitsSigned, dateString);\n}\nexport function parseNDigits(n, dateString) {\n switch (n) {\n case 1:\n return parseNumericPattern(numericPatterns.singleDigit, dateString);\n case 2:\n return parseNumericPattern(numericPatterns.twoDigits, dateString);\n case 3:\n return parseNumericPattern(numericPatterns.threeDigits, dateString);\n case 4:\n return parseNumericPattern(numericPatterns.fourDigits, dateString);\n default:\n return parseNumericPattern(new RegExp('^\\\\d{1,' + n + '}'), dateString);\n }\n}\nexport function parseNDigitsSigned(n, dateString) {\n switch (n) {\n case 1:\n return parseNumericPattern(numericPatterns.singleDigitSigned, dateString);\n case 2:\n return parseNumericPattern(numericPatterns.twoDigitsSigned, dateString);\n case 3:\n return parseNumericPattern(numericPatterns.threeDigitsSigned, dateString);\n case 4:\n return parseNumericPattern(numericPatterns.fourDigitsSigned, dateString);\n default:\n return parseNumericPattern(new RegExp('^-?\\\\d{1,' + n + '}'), dateString);\n }\n}\nexport function dayPeriodEnumToHours(dayPeriod) {\n switch (dayPeriod) {\n case 'morning':\n return 4;\n case 'evening':\n return 17;\n case 'pm':\n case 'noon':\n case 'afternoon':\n return 12;\n case 'am':\n case 'midnight':\n case 'night':\n default:\n return 0;\n }\n}\nexport function normalizeTwoDigitYear(twoDigitYear, currentYear) {\n var isCommonEra = currentYear > 0;\n // Absolute number of the current year:\n // 1 -> 1 AC\n // 0 -> 1 BC\n // -1 -> 2 BC\n var absCurrentYear = isCommonEra ? currentYear : 1 - currentYear;\n var result;\n if (absCurrentYear <= 50) {\n result = twoDigitYear || 100;\n } else {\n var rangeEnd = absCurrentYear + 50;\n var rangeEndCentury = Math.floor(rangeEnd / 100) * 100;\n var isPreviousCentury = twoDigitYear >= rangeEnd % 100;\n result = twoDigitYear + rangeEndCentury - (isPreviousCentury ? 100 : 0);\n }\n return isCommonEra ? result : 1 - result;\n}\nexport function isLeapYearIndex(year) {\n return year % 400 === 0 || year % 4 === 0 && year % 100 !== 0;\n}","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { mapValue, normalizeTwoDigitYear, parseNDigits } from \"../utils.js\";\n// From http://www.unicode.org/reports/tr35/tr35-31/tr35-dates.html#Date_Format_Patterns\n// | Year | y | yy | yyy | yyyy | yyyyy |\n// |----------|-------|----|-------|-------|-------|\n// | AD 1 | 1 | 01 | 001 | 0001 | 00001 |\n// | AD 12 | 12 | 12 | 012 | 0012 | 00012 |\n// | AD 123 | 123 | 23 | 123 | 0123 | 00123 |\n// | AD 1234 | 1234 | 34 | 1234 | 1234 | 01234 |\n// | AD 12345 | 12345 | 45 | 12345 | 12345 | 12345 |\nexport var YearParser = /*#__PURE__*/function (_Parser) {\n _inherits(YearParser, _Parser);\n var _super = _createSuper(YearParser);\n function YearParser() {\n var _this;\n _classCallCheck(this, YearParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 130);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'u', 'w', 'I', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(YearParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n var valueCallback = function valueCallback(year) {\n return {\n year: year,\n isTwoDigitYear: token === 'yy'\n };\n };\n switch (token) {\n case 'y':\n return mapValue(parseNDigits(4, dateString), valueCallback);\n case 'yo':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'year'\n }), valueCallback);\n default:\n return mapValue(parseNDigits(token.length, dateString), valueCallback);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value.isTwoDigitYear || value.year > 0;\n }\n }, {\n key: \"set\",\n value: function set(date, flags, value) {\n var currentYear = date.getUTCFullYear();\n if (value.isTwoDigitYear) {\n var normalizedTwoDigitYear = normalizeTwoDigitYear(value.year, currentYear);\n date.setUTCFullYear(normalizedTwoDigitYear, 0, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n var year = !('era' in flags) || flags.era === 1 ? value.year : 1 - value.year;\n date.setUTCFullYear(year, 0, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return YearParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseNDigits, normalizeTwoDigitYear, mapValue } from \"../utils.js\";\nimport getUTCWeekYear from \"../../../_lib/getUTCWeekYear/index.js\";\nimport startOfUTCWeek from \"../../../_lib/startOfUTCWeek/index.js\";\n// Local week-numbering year\nexport var LocalWeekYearParser = /*#__PURE__*/function (_Parser) {\n _inherits(LocalWeekYearParser, _Parser);\n var _super = _createSuper(LocalWeekYearParser);\n function LocalWeekYearParser() {\n var _this;\n _classCallCheck(this, LocalWeekYearParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 130);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'R', 'u', 'Q', 'q', 'M', 'L', 'I', 'd', 'D', 'i', 't', 'T']);\n return _this;\n }\n _createClass(LocalWeekYearParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n var valueCallback = function valueCallback(year) {\n return {\n year: year,\n isTwoDigitYear: token === 'YY'\n };\n };\n switch (token) {\n case 'Y':\n return mapValue(parseNDigits(4, dateString), valueCallback);\n case 'Yo':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'year'\n }), valueCallback);\n default:\n return mapValue(parseNDigits(token.length, dateString), valueCallback);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value.isTwoDigitYear || value.year > 0;\n }\n }, {\n key: \"set\",\n value: function set(date, flags, value, options) {\n var currentYear = getUTCWeekYear(date, options);\n if (value.isTwoDigitYear) {\n var normalizedTwoDigitYear = normalizeTwoDigitYear(value.year, currentYear);\n date.setUTCFullYear(normalizedTwoDigitYear, 0, options.firstWeekContainsDate);\n date.setUTCHours(0, 0, 0, 0);\n return startOfUTCWeek(date, options);\n }\n var year = !('era' in flags) || flags.era === 1 ? value.year : 1 - value.year;\n date.setUTCFullYear(year, 0, options.firstWeekContainsDate);\n date.setUTCHours(0, 0, 0, 0);\n return startOfUTCWeek(date, options);\n }\n }]);\n return LocalWeekYearParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseNDigitsSigned } from \"../utils.js\";\nimport startOfUTCISOWeek from \"../../../_lib/startOfUTCISOWeek/index.js\"; // ISO week-numbering year\nexport var ISOWeekYearParser = /*#__PURE__*/function (_Parser) {\n _inherits(ISOWeekYearParser, _Parser);\n var _super = _createSuper(ISOWeekYearParser);\n function ISOWeekYearParser() {\n var _this;\n _classCallCheck(this, ISOWeekYearParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 130);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['G', 'y', 'Y', 'u', 'Q', 'q', 'M', 'L', 'w', 'd', 'D', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(ISOWeekYearParser, [{\n key: \"parse\",\n value: function parse(dateString, token) {\n if (token === 'R') {\n return parseNDigitsSigned(4, dateString);\n }\n return parseNDigitsSigned(token.length, dateString);\n }\n }, {\n key: \"set\",\n value: function set(_date, _flags, value) {\n var firstWeekOfYear = new Date(0);\n firstWeekOfYear.setUTCFullYear(value, 0, 4);\n firstWeekOfYear.setUTCHours(0, 0, 0, 0);\n return startOfUTCISOWeek(firstWeekOfYear);\n }\n }]);\n return ISOWeekYearParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseNDigitsSigned } from \"../utils.js\";\nexport var ExtendedYearParser = /*#__PURE__*/function (_Parser) {\n _inherits(ExtendedYearParser, _Parser);\n var _super = _createSuper(ExtendedYearParser);\n function ExtendedYearParser() {\n var _this;\n _classCallCheck(this, ExtendedYearParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 130);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['G', 'y', 'Y', 'R', 'w', 'I', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(ExtendedYearParser, [{\n key: \"parse\",\n value: function parse(dateString, token) {\n if (token === 'u') {\n return parseNDigitsSigned(4, dateString);\n }\n return parseNDigitsSigned(token.length, dateString);\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCFullYear(value, 0, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return ExtendedYearParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseNDigits } from \"../utils.js\";\nexport var QuarterParser = /*#__PURE__*/function (_Parser) {\n _inherits(QuarterParser, _Parser);\n var _super = _createSuper(QuarterParser);\n function QuarterParser() {\n var _this;\n _classCallCheck(this, QuarterParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 120);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'q', 'M', 'L', 'w', 'I', 'd', 'D', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(QuarterParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n // 1, 2, 3, 4\n case 'Q':\n case 'QQ':\n // 01, 02, 03, 04\n return parseNDigits(token.length, dateString);\n // 1st, 2nd, 3rd, 4th\n case 'Qo':\n return match.ordinalNumber(dateString, {\n unit: 'quarter'\n });\n // Q1, Q2, Q3, Q4\n case 'QQQ':\n return match.quarter(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.quarter(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // 1, 2, 3, 4 (narrow quarter; could be not numerical)\n case 'QQQQQ':\n return match.quarter(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // 1st quarter, 2nd quarter, ...\n case 'QQQQ':\n default:\n return match.quarter(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.quarter(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.quarter(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 4;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMonth((value - 1) * 3, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return QuarterParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseNDigits } from \"../utils.js\";\nexport var StandAloneQuarterParser = /*#__PURE__*/function (_Parser) {\n _inherits(StandAloneQuarterParser, _Parser);\n var _super = _createSuper(StandAloneQuarterParser);\n function StandAloneQuarterParser() {\n var _this;\n _classCallCheck(this, StandAloneQuarterParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 120);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'Q', 'M', 'L', 'w', 'I', 'd', 'D', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(StandAloneQuarterParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n // 1, 2, 3, 4\n case 'q':\n case 'qq':\n // 01, 02, 03, 04\n return parseNDigits(token.length, dateString);\n // 1st, 2nd, 3rd, 4th\n case 'qo':\n return match.ordinalNumber(dateString, {\n unit: 'quarter'\n });\n // Q1, Q2, Q3, Q4\n case 'qqq':\n return match.quarter(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.quarter(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // 1, 2, 3, 4 (narrow quarter; could be not numerical)\n case 'qqqqq':\n return match.quarter(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // 1st quarter, 2nd quarter, ...\n case 'qqqq':\n default:\n return match.quarter(dateString, {\n width: 'wide',\n context: 'standalone'\n }) || match.quarter(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.quarter(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 4;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMonth((value - 1) * 3, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return StandAloneQuarterParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { mapValue, parseNDigits, parseNumericPattern } from \"../utils.js\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nexport var MonthParser = /*#__PURE__*/function (_Parser) {\n _inherits(MonthParser, _Parser);\n var _super = _createSuper(MonthParser);\n function MonthParser() {\n var _this;\n _classCallCheck(this, MonthParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'q', 'Q', 'L', 'w', 'I', 'D', 'i', 'e', 'c', 't', 'T']);\n _defineProperty(_assertThisInitialized(_this), \"priority\", 110);\n return _this;\n }\n _createClass(MonthParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n var valueCallback = function valueCallback(value) {\n return value - 1;\n };\n switch (token) {\n // 1, 2, ..., 12\n case 'M':\n return mapValue(parseNumericPattern(numericPatterns.month, dateString), valueCallback);\n // 01, 02, ..., 12\n case 'MM':\n return mapValue(parseNDigits(2, dateString), valueCallback);\n // 1st, 2nd, ..., 12th\n case 'Mo':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'month'\n }), valueCallback);\n // Jan, Feb, ..., Dec\n case 'MMM':\n return match.month(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.month(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // J, F, ..., D\n case 'MMMMM':\n return match.month(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // January, February, ..., December\n case 'MMMM':\n default:\n return match.month(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.month(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.month(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 11;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMonth(value, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return MonthParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits, mapValue } from \"../utils.js\";\nexport var StandAloneMonthParser = /*#__PURE__*/function (_Parser) {\n _inherits(StandAloneMonthParser, _Parser);\n var _super = _createSuper(StandAloneMonthParser);\n function StandAloneMonthParser() {\n var _this;\n _classCallCheck(this, StandAloneMonthParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 110);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'q', 'Q', 'M', 'w', 'I', 'D', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(StandAloneMonthParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n var valueCallback = function valueCallback(value) {\n return value - 1;\n };\n switch (token) {\n // 1, 2, ..., 12\n case 'L':\n return mapValue(parseNumericPattern(numericPatterns.month, dateString), valueCallback);\n // 01, 02, ..., 12\n case 'LL':\n return mapValue(parseNDigits(2, dateString), valueCallback);\n // 1st, 2nd, ..., 12th\n case 'Lo':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'month'\n }), valueCallback);\n // Jan, Feb, ..., Dec\n case 'LLL':\n return match.month(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.month(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // J, F, ..., D\n case 'LLLLL':\n return match.month(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // January, February, ..., December\n case 'LLLL':\n default:\n return match.month(dateString, {\n width: 'wide',\n context: 'standalone'\n }) || match.month(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.month(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 11;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMonth(value, 1);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return StandAloneMonthParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nimport setUTCWeek from \"../../../_lib/setUTCWeek/index.js\";\nimport startOfUTCWeek from \"../../../_lib/startOfUTCWeek/index.js\"; // Local week of year\nexport var LocalWeekParser = /*#__PURE__*/function (_Parser) {\n _inherits(LocalWeekParser, _Parser);\n var _super = _createSuper(LocalWeekParser);\n function LocalWeekParser() {\n var _this;\n _classCallCheck(this, LocalWeekParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 100);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'R', 'u', 'q', 'Q', 'M', 'L', 'I', 'd', 'D', 'i', 't', 'T']);\n return _this;\n }\n _createClass(LocalWeekParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'w':\n return parseNumericPattern(numericPatterns.week, dateString);\n case 'wo':\n return match.ordinalNumber(dateString, {\n unit: 'week'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 53;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value, options) {\n return startOfUTCWeek(setUTCWeek(date, value, options), options);\n }\n }]);\n return LocalWeekParser;\n}(Parser);","import toInteger from \"../toInteger/index.js\";\nimport toDate from \"../../toDate/index.js\";\nimport getUTCWeek from \"../getUTCWeek/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nexport default function setUTCWeek(dirtyDate, dirtyWeek, options) {\n requiredArgs(2, arguments);\n var date = toDate(dirtyDate);\n var week = toInteger(dirtyWeek);\n var diff = getUTCWeek(date, options) - week;\n date.setUTCDate(date.getUTCDate() - diff * 7);\n return date;\n}","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nimport setUTCISOWeek from \"../../../_lib/setUTCISOWeek/index.js\";\nimport startOfUTCISOWeek from \"../../../_lib/startOfUTCISOWeek/index.js\"; // ISO week of year\nexport var ISOWeekParser = /*#__PURE__*/function (_Parser) {\n _inherits(ISOWeekParser, _Parser);\n var _super = _createSuper(ISOWeekParser);\n function ISOWeekParser() {\n var _this;\n _classCallCheck(this, ISOWeekParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 100);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'Y', 'u', 'q', 'Q', 'M', 'L', 'w', 'd', 'D', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(ISOWeekParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'I':\n return parseNumericPattern(numericPatterns.week, dateString);\n case 'Io':\n return match.ordinalNumber(dateString, {\n unit: 'week'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 53;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n return startOfUTCISOWeek(setUTCISOWeek(date, value));\n }\n }]);\n return ISOWeekParser;\n}(Parser);","import toInteger from \"../toInteger/index.js\";\nimport toDate from \"../../toDate/index.js\";\nimport getUTCISOWeek from \"../getUTCISOWeek/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nexport default function setUTCISOWeek(dirtyDate, dirtyISOWeek) {\n requiredArgs(2, arguments);\n var date = toDate(dirtyDate);\n var isoWeek = toInteger(dirtyISOWeek);\n var diff = getUTCISOWeek(date) - isoWeek;\n date.setUTCDate(date.getUTCDate() - diff * 7);\n return date;\n}","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { isLeapYearIndex, parseNDigits, parseNumericPattern } from \"../utils.js\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nvar DAYS_IN_MONTH = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];\nvar DAYS_IN_MONTH_LEAP_YEAR = [31, 29, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];\n\n// Day of the month\nexport var DateParser = /*#__PURE__*/function (_Parser) {\n _inherits(DateParser, _Parser);\n var _super = _createSuper(DateParser);\n function DateParser() {\n var _this;\n _classCallCheck(this, DateParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"subPriority\", 1);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'q', 'Q', 'w', 'I', 'D', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(DateParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'd':\n return parseNumericPattern(numericPatterns.date, dateString);\n case 'do':\n return match.ordinalNumber(dateString, {\n unit: 'date'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(date, value) {\n var year = date.getUTCFullYear();\n var isLeapYear = isLeapYearIndex(year);\n var month = date.getUTCMonth();\n if (isLeapYear) {\n return value >= 1 && value <= DAYS_IN_MONTH_LEAP_YEAR[month];\n } else {\n return value >= 1 && value <= DAYS_IN_MONTH[month];\n }\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCDate(value);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return DateParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits, isLeapYearIndex } from \"../utils.js\";\nexport var DayOfYearParser = /*#__PURE__*/function (_Parser) {\n _inherits(DayOfYearParser, _Parser);\n var _super = _createSuper(DayOfYearParser);\n function DayOfYearParser() {\n var _this;\n _classCallCheck(this, DayOfYearParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"subpriority\", 1);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['Y', 'R', 'q', 'Q', 'M', 'L', 'w', 'I', 'd', 'E', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(DayOfYearParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'D':\n case 'DD':\n return parseNumericPattern(numericPatterns.dayOfYear, dateString);\n case 'Do':\n return match.ordinalNumber(dateString, {\n unit: 'date'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(date, value) {\n var year = date.getUTCFullYear();\n var isLeapYear = isLeapYearIndex(year);\n if (isLeapYear) {\n return value >= 1 && value <= 366;\n } else {\n return value >= 1 && value <= 365;\n }\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMonth(0, value);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return DayOfYearParser;\n}(Parser);","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport toInteger from \"../toInteger/index.js\";\nimport { getDefaultOptions } from \"../defaultOptions/index.js\";\nexport default function setUTCDay(dirtyDate, dirtyDay, options) {\n var _ref, _ref2, _ref3, _options$weekStartsOn, _options$locale, _options$locale$optio, _defaultOptions$local, _defaultOptions$local2;\n requiredArgs(2, arguments);\n var defaultOptions = getDefaultOptions();\n var weekStartsOn = toInteger((_ref = (_ref2 = (_ref3 = (_options$weekStartsOn = options === null || options === void 0 ? void 0 : options.weekStartsOn) !== null && _options$weekStartsOn !== void 0 ? _options$weekStartsOn : options === null || options === void 0 ? void 0 : (_options$locale = options.locale) === null || _options$locale === void 0 ? void 0 : (_options$locale$optio = _options$locale.options) === null || _options$locale$optio === void 0 ? void 0 : _options$locale$optio.weekStartsOn) !== null && _ref3 !== void 0 ? _ref3 : defaultOptions.weekStartsOn) !== null && _ref2 !== void 0 ? _ref2 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.weekStartsOn) !== null && _ref !== void 0 ? _ref : 0);\n\n // Test if weekStartsOn is between 0 and 6 _and_ is not NaN\n if (!(weekStartsOn >= 0 && weekStartsOn <= 6)) {\n throw new RangeError('weekStartsOn must be between 0 and 6 inclusively');\n }\n var date = toDate(dirtyDate);\n var day = toInteger(dirtyDay);\n var currentDay = date.getUTCDay();\n var remainder = day % 7;\n var dayIndex = (remainder + 7) % 7;\n var diff = (dayIndex < weekStartsOn ? 7 : 0) + day - currentDay;\n date.setUTCDate(date.getUTCDate() + diff);\n return date;\n}","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport setUTCDay from \"../../../_lib/setUTCDay/index.js\"; // Day of week\nexport var DayParser = /*#__PURE__*/function (_Parser) {\n _inherits(DayParser, _Parser);\n var _super = _createSuper(DayParser);\n function DayParser() {\n var _this;\n _classCallCheck(this, DayParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['D', 'i', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(DayParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n // Tue\n case 'E':\n case 'EE':\n case 'EEE':\n return match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // T\n case 'EEEEE':\n return match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tu\n case 'EEEEEE':\n return match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tuesday\n case 'EEEE':\n default:\n return match.day(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 6;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value, options) {\n date = setUTCDay(date, value, options);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return DayParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { mapValue, parseNDigits } from \"../utils.js\";\nimport setUTCDay from \"../../../_lib/setUTCDay/index.js\"; // Local day of week\nexport var LocalDayParser = /*#__PURE__*/function (_Parser) {\n _inherits(LocalDayParser, _Parser);\n var _super = _createSuper(LocalDayParser);\n function LocalDayParser() {\n var _this;\n _classCallCheck(this, LocalDayParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'R', 'u', 'q', 'Q', 'M', 'L', 'I', 'd', 'D', 'E', 'i', 'c', 't', 'T']);\n return _this;\n }\n _createClass(LocalDayParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match, options) {\n var valueCallback = function valueCallback(value) {\n var wholeWeekDays = Math.floor((value - 1) / 7) * 7;\n return (value + options.weekStartsOn + 6) % 7 + wholeWeekDays;\n };\n switch (token) {\n // 3\n case 'e':\n case 'ee':\n // 03\n return mapValue(parseNDigits(token.length, dateString), valueCallback);\n // 3rd\n case 'eo':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'day'\n }), valueCallback);\n // Tue\n case 'eee':\n return match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // T\n case 'eeeee':\n return match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tu\n case 'eeeeee':\n return match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n // Tuesday\n case 'eeee':\n default:\n return match.day(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 6;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value, options) {\n date = setUTCDay(date, value, options);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return LocalDayParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { mapValue, parseNDigits } from \"../utils.js\";\nimport setUTCDay from \"../../../_lib/setUTCDay/index.js\"; // Stand-alone local day of week\nexport var StandAloneLocalDayParser = /*#__PURE__*/function (_Parser) {\n _inherits(StandAloneLocalDayParser, _Parser);\n var _super = _createSuper(StandAloneLocalDayParser);\n function StandAloneLocalDayParser() {\n var _this;\n _classCallCheck(this, StandAloneLocalDayParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'R', 'u', 'q', 'Q', 'M', 'L', 'I', 'd', 'D', 'E', 'i', 'e', 't', 'T']);\n return _this;\n }\n _createClass(StandAloneLocalDayParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match, options) {\n var valueCallback = function valueCallback(value) {\n var wholeWeekDays = Math.floor((value - 1) / 7) * 7;\n return (value + options.weekStartsOn + 6) % 7 + wholeWeekDays;\n };\n switch (token) {\n // 3\n case 'c':\n case 'cc':\n // 03\n return mapValue(parseNDigits(token.length, dateString), valueCallback);\n // 3rd\n case 'co':\n return mapValue(match.ordinalNumber(dateString, {\n unit: 'day'\n }), valueCallback);\n // Tue\n case 'ccc':\n return match.day(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'short',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // T\n case 'ccccc':\n return match.day(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // Tu\n case 'cccccc':\n return match.day(dateString, {\n width: 'short',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n // Tuesday\n case 'cccc':\n default:\n return match.day(dateString, {\n width: 'wide',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'abbreviated',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'short',\n context: 'standalone'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'standalone'\n });\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 6;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value, options) {\n date = setUTCDay(date, value, options);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return StandAloneLocalDayParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { mapValue, parseNDigits } from \"../utils.js\";\nimport setUTCISODay from \"../../../_lib/setUTCISODay/index.js\"; // ISO day of week\nexport var ISODayParser = /*#__PURE__*/function (_Parser) {\n _inherits(ISODayParser, _Parser);\n var _super = _createSuper(ISODayParser);\n function ISODayParser() {\n var _this;\n _classCallCheck(this, ISODayParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 90);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['y', 'Y', 'u', 'q', 'Q', 'M', 'L', 'w', 'd', 'D', 'E', 'e', 'c', 't', 'T']);\n return _this;\n }\n _createClass(ISODayParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n var valueCallback = function valueCallback(value) {\n if (value === 0) {\n return 7;\n }\n return value;\n };\n switch (token) {\n // 2\n case 'i':\n case 'ii':\n // 02\n return parseNDigits(token.length, dateString);\n // 2nd\n case 'io':\n return match.ordinalNumber(dateString, {\n unit: 'day'\n });\n // Tue\n case 'iii':\n return mapValue(match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n }), valueCallback);\n // T\n case 'iiiii':\n return mapValue(match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n }), valueCallback);\n // Tu\n case 'iiiiii':\n return mapValue(match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n }), valueCallback);\n // Tuesday\n case 'iiii':\n default:\n return mapValue(match.day(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'short',\n context: 'formatting'\n }) || match.day(dateString, {\n width: 'narrow',\n context: 'formatting'\n }), valueCallback);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 7;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date = setUTCISODay(date, value);\n date.setUTCHours(0, 0, 0, 0);\n return date;\n }\n }]);\n return ISODayParser;\n}(Parser);","import toDate from \"../../toDate/index.js\";\nimport requiredArgs from \"../requiredArgs/index.js\";\nimport toInteger from \"../toInteger/index.js\";\nexport default function setUTCISODay(dirtyDate, dirtyDay) {\n requiredArgs(2, arguments);\n var day = toInteger(dirtyDay);\n if (day % 7 === 0) {\n day = day - 7;\n }\n var weekStartsOn = 1;\n var date = toDate(dirtyDate);\n var currentDay = date.getUTCDay();\n var remainder = day % 7;\n var dayIndex = (remainder + 7) % 7;\n var diff = (dayIndex < weekStartsOn ? 7 : 0) + day - currentDay;\n date.setUTCDate(date.getUTCDate() + diff);\n return date;\n}","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { dayPeriodEnumToHours } from \"../utils.js\";\nexport var AMPMParser = /*#__PURE__*/function (_Parser) {\n _inherits(AMPMParser, _Parser);\n var _super = _createSuper(AMPMParser);\n function AMPMParser() {\n var _this;\n _classCallCheck(this, AMPMParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 80);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['b', 'B', 'H', 'k', 't', 'T']);\n return _this;\n }\n _createClass(AMPMParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'a':\n case 'aa':\n case 'aaa':\n return match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'aaaaa':\n return match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'aaaa':\n default:\n return match.dayPeriod(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCHours(dayPeriodEnumToHours(value), 0, 0, 0);\n return date;\n }\n }]);\n return AMPMParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { dayPeriodEnumToHours } from \"../utils.js\";\nexport var AMPMMidnightParser = /*#__PURE__*/function (_Parser) {\n _inherits(AMPMMidnightParser, _Parser);\n var _super = _createSuper(AMPMMidnightParser);\n function AMPMMidnightParser() {\n var _this;\n _classCallCheck(this, AMPMMidnightParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 80);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['a', 'B', 'H', 'k', 't', 'T']);\n return _this;\n }\n _createClass(AMPMMidnightParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'b':\n case 'bb':\n case 'bbb':\n return match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'bbbbb':\n return match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'bbbb':\n default:\n return match.dayPeriod(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCHours(dayPeriodEnumToHours(value), 0, 0, 0);\n return date;\n }\n }]);\n return AMPMMidnightParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { dayPeriodEnumToHours } from \"../utils.js\"; // in the morning, in the afternoon, in the evening, at night\nexport var DayPeriodParser = /*#__PURE__*/function (_Parser) {\n _inherits(DayPeriodParser, _Parser);\n var _super = _createSuper(DayPeriodParser);\n function DayPeriodParser() {\n var _this;\n _classCallCheck(this, DayPeriodParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 80);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['a', 'b', 't', 'T']);\n return _this;\n }\n _createClass(DayPeriodParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'B':\n case 'BB':\n case 'BBB':\n return match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'BBBBB':\n return match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n case 'BBBB':\n default:\n return match.dayPeriod(dateString, {\n width: 'wide',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'abbreviated',\n context: 'formatting'\n }) || match.dayPeriod(dateString, {\n width: 'narrow',\n context: 'formatting'\n });\n }\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCHours(dayPeriodEnumToHours(value), 0, 0, 0);\n return date;\n }\n }]);\n return DayPeriodParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var Hour1to12Parser = /*#__PURE__*/function (_Parser) {\n _inherits(Hour1to12Parser, _Parser);\n var _super = _createSuper(Hour1to12Parser);\n function Hour1to12Parser() {\n var _this;\n _classCallCheck(this, Hour1to12Parser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 70);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['H', 'K', 'k', 't', 'T']);\n return _this;\n }\n _createClass(Hour1to12Parser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'h':\n return parseNumericPattern(numericPatterns.hour12h, dateString);\n case 'ho':\n return match.ordinalNumber(dateString, {\n unit: 'hour'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 12;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n var isPM = date.getUTCHours() >= 12;\n if (isPM && value < 12) {\n date.setUTCHours(value + 12, 0, 0, 0);\n } else if (!isPM && value === 12) {\n date.setUTCHours(0, 0, 0, 0);\n } else {\n date.setUTCHours(value, 0, 0, 0);\n }\n return date;\n }\n }]);\n return Hour1to12Parser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var Hour0to23Parser = /*#__PURE__*/function (_Parser) {\n _inherits(Hour0to23Parser, _Parser);\n var _super = _createSuper(Hour0to23Parser);\n function Hour0to23Parser() {\n var _this;\n _classCallCheck(this, Hour0to23Parser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 70);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['a', 'b', 'h', 'K', 'k', 't', 'T']);\n return _this;\n }\n _createClass(Hour0to23Parser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'H':\n return parseNumericPattern(numericPatterns.hour23h, dateString);\n case 'Ho':\n return match.ordinalNumber(dateString, {\n unit: 'hour'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 23;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCHours(value, 0, 0, 0);\n return date;\n }\n }]);\n return Hour0to23Parser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var Hour0To11Parser = /*#__PURE__*/function (_Parser) {\n _inherits(Hour0To11Parser, _Parser);\n var _super = _createSuper(Hour0To11Parser);\n function Hour0To11Parser() {\n var _this;\n _classCallCheck(this, Hour0To11Parser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 70);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['h', 'H', 'k', 't', 'T']);\n return _this;\n }\n _createClass(Hour0To11Parser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'K':\n return parseNumericPattern(numericPatterns.hour11h, dateString);\n case 'Ko':\n return match.ordinalNumber(dateString, {\n unit: 'hour'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 11;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n var isPM = date.getUTCHours() >= 12;\n if (isPM && value < 12) {\n date.setUTCHours(value + 12, 0, 0, 0);\n } else {\n date.setUTCHours(value, 0, 0, 0);\n }\n return date;\n }\n }]);\n return Hour0To11Parser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var Hour1To24Parser = /*#__PURE__*/function (_Parser) {\n _inherits(Hour1To24Parser, _Parser);\n var _super = _createSuper(Hour1To24Parser);\n function Hour1To24Parser() {\n var _this;\n _classCallCheck(this, Hour1To24Parser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 70);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['a', 'b', 'h', 'H', 'K', 't', 'T']);\n return _this;\n }\n _createClass(Hour1To24Parser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'k':\n return parseNumericPattern(numericPatterns.hour24h, dateString);\n case 'ko':\n return match.ordinalNumber(dateString, {\n unit: 'hour'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 1 && value <= 24;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n var hours = value <= 24 ? value % 24 : value;\n date.setUTCHours(hours, 0, 0, 0);\n return date;\n }\n }]);\n return Hour1To24Parser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var MinuteParser = /*#__PURE__*/function (_Parser) {\n _inherits(MinuteParser, _Parser);\n var _super = _createSuper(MinuteParser);\n function MinuteParser() {\n var _this;\n _classCallCheck(this, MinuteParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 60);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['t', 'T']);\n return _this;\n }\n _createClass(MinuteParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 'm':\n return parseNumericPattern(numericPatterns.minute, dateString);\n case 'mo':\n return match.ordinalNumber(dateString, {\n unit: 'minute'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 59;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMinutes(value, 0, 0);\n return date;\n }\n }]);\n return MinuteParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { numericPatterns } from \"../constants.js\";\nimport { parseNumericPattern, parseNDigits } from \"../utils.js\";\nexport var SecondParser = /*#__PURE__*/function (_Parser) {\n _inherits(SecondParser, _Parser);\n var _super = _createSuper(SecondParser);\n function SecondParser() {\n var _this;\n _classCallCheck(this, SecondParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 50);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['t', 'T']);\n return _this;\n }\n _createClass(SecondParser, [{\n key: \"parse\",\n value: function parse(dateString, token, match) {\n switch (token) {\n case 's':\n return parseNumericPattern(numericPatterns.second, dateString);\n case 'so':\n return match.ordinalNumber(dateString, {\n unit: 'second'\n });\n default:\n return parseNDigits(token.length, dateString);\n }\n }\n }, {\n key: \"validate\",\n value: function validate(_date, value) {\n return value >= 0 && value <= 59;\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCSeconds(value, 0);\n return date;\n }\n }]);\n return SecondParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { mapValue, parseNDigits } from \"../utils.js\";\nexport var FractionOfSecondParser = /*#__PURE__*/function (_Parser) {\n _inherits(FractionOfSecondParser, _Parser);\n var _super = _createSuper(FractionOfSecondParser);\n function FractionOfSecondParser() {\n var _this;\n _classCallCheck(this, FractionOfSecondParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 30);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['t', 'T']);\n return _this;\n }\n _createClass(FractionOfSecondParser, [{\n key: \"parse\",\n value: function parse(dateString, token) {\n var valueCallback = function valueCallback(value) {\n return Math.floor(value * Math.pow(10, -token.length + 3));\n };\n return mapValue(parseNDigits(token.length, dateString), valueCallback);\n }\n }, {\n key: \"set\",\n value: function set(date, _flags, value) {\n date.setUTCMilliseconds(value);\n return date;\n }\n }]);\n return FractionOfSecondParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { timezonePatterns } from \"../constants.js\";\nimport { parseTimezonePattern } from \"../utils.js\"; // Timezone (ISO-8601. +00:00 is `'Z'`)\nexport var ISOTimezoneWithZParser = /*#__PURE__*/function (_Parser) {\n _inherits(ISOTimezoneWithZParser, _Parser);\n var _super = _createSuper(ISOTimezoneWithZParser);\n function ISOTimezoneWithZParser() {\n var _this;\n _classCallCheck(this, ISOTimezoneWithZParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 10);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['t', 'T', 'x']);\n return _this;\n }\n _createClass(ISOTimezoneWithZParser, [{\n key: \"parse\",\n value: function parse(dateString, token) {\n switch (token) {\n case 'X':\n return parseTimezonePattern(timezonePatterns.basicOptionalMinutes, dateString);\n case 'XX':\n return parseTimezonePattern(timezonePatterns.basic, dateString);\n case 'XXXX':\n return parseTimezonePattern(timezonePatterns.basicOptionalSeconds, dateString);\n case 'XXXXX':\n return parseTimezonePattern(timezonePatterns.extendedOptionalSeconds, dateString);\n case 'XXX':\n default:\n return parseTimezonePattern(timezonePatterns.extended, dateString);\n }\n }\n }, {\n key: \"set\",\n value: function set(date, flags, value) {\n if (flags.timestampIsSet) {\n return date;\n }\n return new Date(date.getTime() - value);\n }\n }]);\n return ISOTimezoneWithZParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { timezonePatterns } from \"../constants.js\";\nimport { parseTimezonePattern } from \"../utils.js\"; // Timezone (ISO-8601)\nexport var ISOTimezoneParser = /*#__PURE__*/function (_Parser) {\n _inherits(ISOTimezoneParser, _Parser);\n var _super = _createSuper(ISOTimezoneParser);\n function ISOTimezoneParser() {\n var _this;\n _classCallCheck(this, ISOTimezoneParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 10);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", ['t', 'T', 'X']);\n return _this;\n }\n _createClass(ISOTimezoneParser, [{\n key: \"parse\",\n value: function parse(dateString, token) {\n switch (token) {\n case 'x':\n return parseTimezonePattern(timezonePatterns.basicOptionalMinutes, dateString);\n case 'xx':\n return parseTimezonePattern(timezonePatterns.basic, dateString);\n case 'xxxx':\n return parseTimezonePattern(timezonePatterns.basicOptionalSeconds, dateString);\n case 'xxxxx':\n return parseTimezonePattern(timezonePatterns.extendedOptionalSeconds, dateString);\n case 'xxx':\n default:\n return parseTimezonePattern(timezonePatterns.extended, dateString);\n }\n }\n }, {\n key: \"set\",\n value: function set(date, flags, value) {\n if (flags.timestampIsSet) {\n return date;\n }\n return new Date(date.getTime() - value);\n }\n }]);\n return ISOTimezoneParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseAnyDigitsSigned } from \"../utils.js\";\nexport var TimestampSecondsParser = /*#__PURE__*/function (_Parser) {\n _inherits(TimestampSecondsParser, _Parser);\n var _super = _createSuper(TimestampSecondsParser);\n function TimestampSecondsParser() {\n var _this;\n _classCallCheck(this, TimestampSecondsParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 40);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", '*');\n return _this;\n }\n _createClass(TimestampSecondsParser, [{\n key: \"parse\",\n value: function parse(dateString) {\n return parseAnyDigitsSigned(dateString);\n }\n }, {\n key: \"set\",\n value: function set(_date, _flags, value) {\n return [new Date(value * 1000), {\n timestampIsSet: true\n }];\n }\n }]);\n return TimestampSecondsParser;\n}(Parser);","import _classCallCheck from \"@babel/runtime/helpers/esm/classCallCheck\";\nimport _createClass from \"@babel/runtime/helpers/esm/createClass\";\nimport _assertThisInitialized from \"@babel/runtime/helpers/esm/assertThisInitialized\";\nimport _inherits from \"@babel/runtime/helpers/esm/inherits\";\nimport _createSuper from \"@babel/runtime/helpers/esm/createSuper\";\nimport _defineProperty from \"@babel/runtime/helpers/esm/defineProperty\";\nimport { Parser } from \"../Parser.js\";\nimport { parseAnyDigitsSigned } from \"../utils.js\";\nexport var TimestampMillisecondsParser = /*#__PURE__*/function (_Parser) {\n _inherits(TimestampMillisecondsParser, _Parser);\n var _super = _createSuper(TimestampMillisecondsParser);\n function TimestampMillisecondsParser() {\n var _this;\n _classCallCheck(this, TimestampMillisecondsParser);\n for (var _len = arguments.length, args = new Array(_len), _key = 0; _key < _len; _key++) {\n args[_key] = arguments[_key];\n }\n _this = _super.call.apply(_super, [this].concat(args));\n _defineProperty(_assertThisInitialized(_this), \"priority\", 20);\n _defineProperty(_assertThisInitialized(_this), \"incompatibleTokens\", '*');\n return _this;\n }\n _createClass(TimestampMillisecondsParser, [{\n key: \"parse\",\n value: function parse(dateString) {\n return parseAnyDigitsSigned(dateString);\n }\n }, {\n key: \"set\",\n value: function set(_date, _flags, value) {\n return [new Date(value), {\n timestampIsSet: true\n }];\n }\n }]);\n return TimestampMillisecondsParser;\n}(Parser);","import { EraParser } from \"./EraParser.js\";\nimport { YearParser } from \"./YearParser.js\";\nimport { LocalWeekYearParser } from \"./LocalWeekYearParser.js\";\nimport { ISOWeekYearParser } from \"./ISOWeekYearParser.js\";\nimport { ExtendedYearParser } from \"./ExtendedYearParser.js\";\nimport { QuarterParser } from \"./QuarterParser.js\";\nimport { StandAloneQuarterParser } from \"./StandAloneQuarterParser.js\";\nimport { MonthParser } from \"./MonthParser.js\";\nimport { StandAloneMonthParser } from \"./StandAloneMonthParser.js\";\nimport { LocalWeekParser } from \"./LocalWeekParser.js\";\nimport { ISOWeekParser } from \"./ISOWeekParser.js\";\nimport { DateParser } from \"./DateParser.js\";\nimport { DayOfYearParser } from \"./DayOfYearParser.js\";\nimport { DayParser } from \"./DayParser.js\";\nimport { LocalDayParser } from \"./LocalDayParser.js\";\nimport { StandAloneLocalDayParser } from \"./StandAloneLocalDayParser.js\";\nimport { ISODayParser } from \"./ISODayParser.js\";\nimport { AMPMParser } from \"./AMPMParser.js\";\nimport { AMPMMidnightParser } from \"./AMPMMidnightParser.js\";\nimport { DayPeriodParser } from \"./DayPeriodParser.js\";\nimport { Hour1to12Parser } from \"./Hour1to12Parser.js\";\nimport { Hour0to23Parser } from \"./Hour0to23Parser.js\";\nimport { Hour0To11Parser } from \"./Hour0To11Parser.js\";\nimport { Hour1To24Parser } from \"./Hour1To24Parser.js\";\nimport { MinuteParser } from \"./MinuteParser.js\";\nimport { SecondParser } from \"./SecondParser.js\";\nimport { FractionOfSecondParser } from \"./FractionOfSecondParser.js\";\nimport { ISOTimezoneWithZParser } from \"./ISOTimezoneWithZParser.js\";\nimport { ISOTimezoneParser } from \"./ISOTimezoneParser.js\";\nimport { TimestampSecondsParser } from \"./TimestampSecondsParser.js\";\nimport { TimestampMillisecondsParser } from \"./TimestampMillisecondsParser.js\";\n/*\n * | | Unit | | Unit |\n * |-----|--------------------------------|-----|--------------------------------|\n * | a | AM, PM | A* | Milliseconds in day |\n * | b | AM, PM, noon, midnight | B | Flexible day period |\n * | c | Stand-alone local day of week | C* | Localized hour w/ day period |\n * | d | Day of month | D | Day of year |\n * | e | Local day of week | E | Day of week |\n * | f | | F* | Day of week in month |\n * | g* | Modified Julian day | G | Era |\n * | h | Hour [1-12] | H | Hour [0-23] |\n * | i! | ISO day of week | I! | ISO week of year |\n * | j* | Localized hour w/ day period | J* | Localized hour w/o day period |\n * | k | Hour [1-24] | K | Hour [0-11] |\n * | l* | (deprecated) | L | Stand-alone month |\n * | m | Minute | M | Month |\n * | n | | N | |\n * | o! | Ordinal number modifier | O* | Timezone (GMT) |\n * | p | | P | |\n * | q | Stand-alone quarter | Q | Quarter |\n * | r* | Related Gregorian year | R! | ISO week-numbering year |\n * | s | Second | S | Fraction of second |\n * | t! | Seconds timestamp | T! | Milliseconds timestamp |\n * | u | Extended year | U* | Cyclic year |\n * | v* | Timezone (generic non-locat.) | V* | Timezone (location) |\n * | w | Local week of year | W* | Week of month |\n * | x | Timezone (ISO-8601 w/o Z) | X | Timezone (ISO-8601) |\n * | y | Year (abs) | Y | Local week-numbering year |\n * | z* | Timezone (specific non-locat.) | Z* | Timezone (aliases) |\n *\n * Letters marked by * are not implemented but reserved by Unicode standard.\n *\n * Letters marked by ! are non-standard, but implemented by date-fns:\n * - `o` modifies the previous token to turn it into an ordinal (see `parse` docs)\n * - `i` is ISO day of week. For `i` and `ii` is returns numeric ISO week days,\n * i.e. 7 for Sunday, 1 for Monday, etc.\n * - `I` is ISO week of year, as opposed to `w` which is local week of year.\n * - `R` is ISO week-numbering year, as opposed to `Y` which is local week-numbering year.\n * `R` is supposed to be used in conjunction with `I` and `i`\n * for universal ISO week-numbering date, whereas\n * `Y` is supposed to be used in conjunction with `w` and `e`\n * for week-numbering date specific to the locale.\n */\nexport var parsers = {\n G: new EraParser(),\n y: new YearParser(),\n Y: new LocalWeekYearParser(),\n R: new ISOWeekYearParser(),\n u: new ExtendedYearParser(),\n Q: new QuarterParser(),\n q: new StandAloneQuarterParser(),\n M: new MonthParser(),\n L: new StandAloneMonthParser(),\n w: new LocalWeekParser(),\n I: new ISOWeekParser(),\n d: new DateParser(),\n D: new DayOfYearParser(),\n E: new DayParser(),\n e: new LocalDayParser(),\n c: new StandAloneLocalDayParser(),\n i: new ISODayParser(),\n a: new AMPMParser(),\n b: new AMPMMidnightParser(),\n B: new DayPeriodParser(),\n h: new Hour1to12Parser(),\n H: new Hour0to23Parser(),\n K: new Hour0To11Parser(),\n k: new Hour1To24Parser(),\n m: new MinuteParser(),\n s: new SecondParser(),\n S: new FractionOfSecondParser(),\n X: new ISOTimezoneWithZParser(),\n x: new ISOTimezoneParser(),\n t: new TimestampSecondsParser(),\n T: new TimestampMillisecondsParser()\n};","import _typeof from \"@babel/runtime/helpers/esm/typeof\";\nimport _createForOfIteratorHelper from \"@babel/runtime/helpers/esm/createForOfIteratorHelper\";\nimport defaultLocale from \"../_lib/defaultLocale/index.js\";\nimport subMilliseconds from \"../subMilliseconds/index.js\";\nimport toDate from \"../toDate/index.js\";\nimport assign from \"../_lib/assign/index.js\";\nimport longFormatters from \"../_lib/format/longFormatters/index.js\";\nimport getTimezoneOffsetInMilliseconds from \"../_lib/getTimezoneOffsetInMilliseconds/index.js\";\nimport { isProtectedDayOfYearToken, isProtectedWeekYearToken, throwProtectedError } from \"../_lib/protectedTokens/index.js\";\nimport toInteger from \"../_lib/toInteger/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\nimport { DateToSystemTimezoneSetter } from \"./_lib/Setter.js\";\nimport { parsers } from \"./_lib/parsers/index.js\";\nimport { getDefaultOptions } from \"../_lib/defaultOptions/index.js\"; // This RegExp consists of three parts separated by `|`:\n// - [yYQqMLwIdDecihHKkms]o matches any available ordinal number token\n// (one of the certain letters followed by `o`)\n// - (\\w)\\1* matches any sequences of the same letter\n// - '' matches two quote characters in a row\n// - '(''|[^'])+('|$) matches anything surrounded by two quote characters ('),\n// except a single quote symbol, which ends the sequence.\n// Two quote characters do not end the sequence.\n// If there is no matching single quote\n// then the sequence will continue until the end of the string.\n// - . matches any single character unmatched by previous parts of the RegExps\nvar formattingTokensRegExp = /[yYQqMLwIdDecihHKkms]o|(\\w)\\1*|''|'(''|[^'])+('|$)|./g;\n\n// This RegExp catches symbols escaped by quotes, and also\n// sequences of symbols P, p, and the combinations like `PPPPPPPppppp`\nvar longFormattingTokensRegExp = /P+p+|P+|p+|''|'(''|[^'])+('|$)|./g;\nvar escapedStringRegExp = /^'([^]*?)'?$/;\nvar doubleQuoteRegExp = /''/g;\nvar notWhitespaceRegExp = /\\S/;\nvar unescapedLatinCharacterRegExp = /[a-zA-Z]/;\n\n/**\n * @name parse\n * @category Common Helpers\n * @summary Parse the date.\n *\n * @description\n * Return the date parsed from string using the given format string.\n *\n * > ⚠️ Please note that the `format` tokens differ from Moment.js and other libraries.\n * > See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * The characters in the format string wrapped between two single quotes characters (') are escaped.\n * Two single quotes in a row, whether inside or outside a quoted sequence, represent a 'real' single quote.\n *\n * Format of the format string is based on Unicode Technical Standard #35:\n * https://www.unicode.org/reports/tr35/tr35-dates.html#Date_Field_Symbol_Table\n * with a few additions (see note 5 below the table).\n *\n * Not all tokens are compatible. Combinations that don't make sense or could lead to bugs are prohibited\n * and will throw `RangeError`. For example usage of 24-hour format token with AM/PM token will throw an exception:\n *\n * ```javascript\n * parse('23 AM', 'HH a', new Date())\n * //=> RangeError: The format string mustn't contain `HH` and `a` at the same time\n * ```\n *\n * See the compatibility table: https://docs.google.com/spreadsheets/d/e/2PACX-1vQOPU3xUhplll6dyoMmVUXHKl_8CRDs6_ueLmex3SoqwhuolkuN3O05l4rqx5h1dKX8eb46Ul-CCSrq/pubhtml?gid=0&single=true\n *\n * Accepted format string patterns:\n * | Unit |Prior| Pattern | Result examples | Notes |\n * |---------------------------------|-----|---------|-----------------------------------|-------|\n * | Era | 140 | G..GGG | AD, BC | |\n * | | | GGGG | Anno Domini, Before Christ | 2 |\n * | | | GGGGG | A, B | |\n * | Calendar year | 130 | y | 44, 1, 1900, 2017, 9999 | 4 |\n * | | | yo | 44th, 1st, 1900th, 9999999th | 4,5 |\n * | | | yy | 44, 01, 00, 17 | 4 |\n * | | | yyy | 044, 001, 123, 999 | 4 |\n * | | | yyyy | 0044, 0001, 1900, 2017 | 4 |\n * | | | yyyyy | ... | 2,4 |\n * | Local week-numbering year | 130 | Y | 44, 1, 1900, 2017, 9000 | 4 |\n * | | | Yo | 44th, 1st, 1900th, 9999999th | 4,5 |\n * | | | YY | 44, 01, 00, 17 | 4,6 |\n * | | | YYY | 044, 001, 123, 999 | 4 |\n * | | | YYYY | 0044, 0001, 1900, 2017 | 4,6 |\n * | | | YYYYY | ... | 2,4 |\n * | ISO week-numbering year | 130 | R | -43, 1, 1900, 2017, 9999, -9999 | 4,5 |\n * | | | RR | -43, 01, 00, 17 | 4,5 |\n * | | | RRR | -043, 001, 123, 999, -999 | 4,5 |\n * | | | RRRR | -0043, 0001, 2017, 9999, -9999 | 4,5 |\n * | | | RRRRR | ... | 2,4,5 |\n * | Extended year | 130 | u | -43, 1, 1900, 2017, 9999, -999 | 4 |\n * | | | uu | -43, 01, 99, -99 | 4 |\n * | | | uuu | -043, 001, 123, 999, -999 | 4 |\n * | | | uuuu | -0043, 0001, 2017, 9999, -9999 | 4 |\n * | | | uuuuu | ... | 2,4 |\n * | Quarter (formatting) | 120 | Q | 1, 2, 3, 4 | |\n * | | | Qo | 1st, 2nd, 3rd, 4th | 5 |\n * | | | QQ | 01, 02, 03, 04 | |\n * | | | QQQ | Q1, Q2, Q3, Q4 | |\n * | | | QQQQ | 1st quarter, 2nd quarter, ... | 2 |\n * | | | QQQQQ | 1, 2, 3, 4 | 4 |\n * | Quarter (stand-alone) | 120 | q | 1, 2, 3, 4 | |\n * | | | qo | 1st, 2nd, 3rd, 4th | 5 |\n * | | | qq | 01, 02, 03, 04 | |\n * | | | qqq | Q1, Q2, Q3, Q4 | |\n * | | | qqqq | 1st quarter, 2nd quarter, ... | 2 |\n * | | | qqqqq | 1, 2, 3, 4 | 3 |\n * | Month (formatting) | 110 | M | 1, 2, ..., 12 | |\n * | | | Mo | 1st, 2nd, ..., 12th | 5 |\n * | | | MM | 01, 02, ..., 12 | |\n * | | | MMM | Jan, Feb, ..., Dec | |\n * | | | MMMM | January, February, ..., December | 2 |\n * | | | MMMMM | J, F, ..., D | |\n * | Month (stand-alone) | 110 | L | 1, 2, ..., 12 | |\n * | | | Lo | 1st, 2nd, ..., 12th | 5 |\n * | | | LL | 01, 02, ..., 12 | |\n * | | | LLL | Jan, Feb, ..., Dec | |\n * | | | LLLL | January, February, ..., December | 2 |\n * | | | LLLLL | J, F, ..., D | |\n * | Local week of year | 100 | w | 1, 2, ..., 53 | |\n * | | | wo | 1st, 2nd, ..., 53th | 5 |\n * | | | ww | 01, 02, ..., 53 | |\n * | ISO week of year | 100 | I | 1, 2, ..., 53 | 5 |\n * | | | Io | 1st, 2nd, ..., 53th | 5 |\n * | | | II | 01, 02, ..., 53 | 5 |\n * | Day of month | 90 | d | 1, 2, ..., 31 | |\n * | | | do | 1st, 2nd, ..., 31st | 5 |\n * | | | dd | 01, 02, ..., 31 | |\n * | Day of year | 90 | D | 1, 2, ..., 365, 366 | 7 |\n * | | | Do | 1st, 2nd, ..., 365th, 366th | 5 |\n * | | | DD | 01, 02, ..., 365, 366 | 7 |\n * | | | DDD | 001, 002, ..., 365, 366 | |\n * | | | DDDD | ... | 2 |\n * | Day of week (formatting) | 90 | E..EEE | Mon, Tue, Wed, ..., Sun | |\n * | | | EEEE | Monday, Tuesday, ..., Sunday | 2 |\n * | | | EEEEE | M, T, W, T, F, S, S | |\n * | | | EEEEEE | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | ISO day of week (formatting) | 90 | i | 1, 2, 3, ..., 7 | 5 |\n * | | | io | 1st, 2nd, ..., 7th | 5 |\n * | | | ii | 01, 02, ..., 07 | 5 |\n * | | | iii | Mon, Tue, Wed, ..., Sun | 5 |\n * | | | iiii | Monday, Tuesday, ..., Sunday | 2,5 |\n * | | | iiiii | M, T, W, T, F, S, S | 5 |\n * | | | iiiiii | Mo, Tu, We, Th, Fr, Sa, Su | 5 |\n * | Local day of week (formatting) | 90 | e | 2, 3, 4, ..., 1 | |\n * | | | eo | 2nd, 3rd, ..., 1st | 5 |\n * | | | ee | 02, 03, ..., 01 | |\n * | | | eee | Mon, Tue, Wed, ..., Sun | |\n * | | | eeee | Monday, Tuesday, ..., Sunday | 2 |\n * | | | eeeee | M, T, W, T, F, S, S | |\n * | | | eeeeee | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | Local day of week (stand-alone) | 90 | c | 2, 3, 4, ..., 1 | |\n * | | | co | 2nd, 3rd, ..., 1st | 5 |\n * | | | cc | 02, 03, ..., 01 | |\n * | | | ccc | Mon, Tue, Wed, ..., Sun | |\n * | | | cccc | Monday, Tuesday, ..., Sunday | 2 |\n * | | | ccccc | M, T, W, T, F, S, S | |\n * | | | cccccc | Mo, Tu, We, Th, Fr, Sa, Su | |\n * | AM, PM | 80 | a..aaa | AM, PM | |\n * | | | aaaa | a.m., p.m. | 2 |\n * | | | aaaaa | a, p | |\n * | AM, PM, noon, midnight | 80 | b..bbb | AM, PM, noon, midnight | |\n * | | | bbbb | a.m., p.m., noon, midnight | 2 |\n * | | | bbbbb | a, p, n, mi | |\n * | Flexible day period | 80 | B..BBB | at night, in the morning, ... | |\n * | | | BBBB | at night, in the morning, ... | 2 |\n * | | | BBBBB | at night, in the morning, ... | |\n * | Hour [1-12] | 70 | h | 1, 2, ..., 11, 12 | |\n * | | | ho | 1st, 2nd, ..., 11th, 12th | 5 |\n * | | | hh | 01, 02, ..., 11, 12 | |\n * | Hour [0-23] | 70 | H | 0, 1, 2, ..., 23 | |\n * | | | Ho | 0th, 1st, 2nd, ..., 23rd | 5 |\n * | | | HH | 00, 01, 02, ..., 23 | |\n * | Hour [0-11] | 70 | K | 1, 2, ..., 11, 0 | |\n * | | | Ko | 1st, 2nd, ..., 11th, 0th | 5 |\n * | | | KK | 01, 02, ..., 11, 00 | |\n * | Hour [1-24] | 70 | k | 24, 1, 2, ..., 23 | |\n * | | | ko | 24th, 1st, 2nd, ..., 23rd | 5 |\n * | | | kk | 24, 01, 02, ..., 23 | |\n * | Minute | 60 | m | 0, 1, ..., 59 | |\n * | | | mo | 0th, 1st, ..., 59th | 5 |\n * | | | mm | 00, 01, ..., 59 | |\n * | Second | 50 | s | 0, 1, ..., 59 | |\n * | | | so | 0th, 1st, ..., 59th | 5 |\n * | | | ss | 00, 01, ..., 59 | |\n * | Seconds timestamp | 40 | t | 512969520 | |\n * | | | tt | ... | 2 |\n * | Fraction of second | 30 | S | 0, 1, ..., 9 | |\n * | | | SS | 00, 01, ..., 99 | |\n * | | | SSS | 000, 001, ..., 999 | |\n * | | | SSSS | ... | 2 |\n * | Milliseconds timestamp | 20 | T | 512969520900 | |\n * | | | TT | ... | 2 |\n * | Timezone (ISO-8601 w/ Z) | 10 | X | -08, +0530, Z | |\n * | | | XX | -0800, +0530, Z | |\n * | | | XXX | -08:00, +05:30, Z | |\n * | | | XXXX | -0800, +0530, Z, +123456 | 2 |\n * | | | XXXXX | -08:00, +05:30, Z, +12:34:56 | |\n * | Timezone (ISO-8601 w/o Z) | 10 | x | -08, +0530, +00 | |\n * | | | xx | -0800, +0530, +0000 | |\n * | | | xxx | -08:00, +05:30, +00:00 | 2 |\n * | | | xxxx | -0800, +0530, +0000, +123456 | |\n * | | | xxxxx | -08:00, +05:30, +00:00, +12:34:56 | |\n * | Long localized date | NA | P | 05/29/1453 | 5,8 |\n * | | | PP | May 29, 1453 | |\n * | | | PPP | May 29th, 1453 | |\n * | | | PPPP | Sunday, May 29th, 1453 | 2,5,8 |\n * | Long localized time | NA | p | 12:00 AM | 5,8 |\n * | | | pp | 12:00:00 AM | |\n * | Combination of date and time | NA | Pp | 05/29/1453, 12:00 AM | |\n * | | | PPpp | May 29, 1453, 12:00:00 AM | |\n * | | | PPPpp | May 29th, 1453 at ... | |\n * | | | PPPPpp | Sunday, May 29th, 1453 at ... | 2,5,8 |\n * Notes:\n * 1. \"Formatting\" units (e.g. formatting quarter) in the default en-US locale\n * are the same as \"stand-alone\" units, but are different in some languages.\n * \"Formatting\" units are declined according to the rules of the language\n * in the context of a date. \"Stand-alone\" units are always nominative singular.\n * In `format` function, they will produce different result:\n *\n * `format(new Date(2017, 10, 6), 'do LLLL', {locale: cs}) //=> '6. listopad'`\n *\n * `format(new Date(2017, 10, 6), 'do MMMM', {locale: cs}) //=> '6. listopadu'`\n *\n * `parse` will try to match both formatting and stand-alone units interchangably.\n *\n * 2. Any sequence of the identical letters is a pattern, unless it is escaped by\n * the single quote characters (see below).\n * If the sequence is longer than listed in table:\n * - for numerical units (`yyyyyyyy`) `parse` will try to match a number\n * as wide as the sequence\n * - for text units (`MMMMMMMM`) `parse` will try to match the widest variation of the unit.\n * These variations are marked with \"2\" in the last column of the table.\n *\n * 3. `QQQQQ` and `qqqqq` could be not strictly numerical in some locales.\n * These tokens represent the shortest form of the quarter.\n *\n * 4. The main difference between `y` and `u` patterns are B.C. years:\n *\n * | Year | `y` | `u` |\n * |------|-----|-----|\n * | AC 1 | 1 | 1 |\n * | BC 1 | 1 | 0 |\n * | BC 2 | 2 | -1 |\n *\n * Also `yy` will try to guess the century of two digit year by proximity with `referenceDate`:\n *\n * `parse('50', 'yy', new Date(2018, 0, 1)) //=> Sat Jan 01 2050 00:00:00`\n *\n * `parse('75', 'yy', new Date(2018, 0, 1)) //=> Wed Jan 01 1975 00:00:00`\n *\n * while `uu` will just assign the year as is:\n *\n * `parse('50', 'uu', new Date(2018, 0, 1)) //=> Sat Jan 01 0050 00:00:00`\n *\n * `parse('75', 'uu', new Date(2018, 0, 1)) //=> Tue Jan 01 0075 00:00:00`\n *\n * The same difference is true for local and ISO week-numbering years (`Y` and `R`),\n * except local week-numbering years are dependent on `options.weekStartsOn`\n * and `options.firstWeekContainsDate` (compare [setISOWeekYear]{@link https://date-fns.org/docs/setISOWeekYear}\n * and [setWeekYear]{@link https://date-fns.org/docs/setWeekYear}).\n *\n * 5. These patterns are not in the Unicode Technical Standard #35:\n * - `i`: ISO day of week\n * - `I`: ISO week of year\n * - `R`: ISO week-numbering year\n * - `o`: ordinal number modifier\n * - `P`: long localized date\n * - `p`: long localized time\n *\n * 6. `YY` and `YYYY` tokens represent week-numbering years but they are often confused with years.\n * You should enable `options.useAdditionalWeekYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * 7. `D` and `DD` tokens represent days of the year but they are ofthen confused with days of the month.\n * You should enable `options.useAdditionalDayOfYearTokens` to use them. See: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n *\n * 8. `P+` tokens do not have a defined priority since they are merely aliases to other tokens based\n * on the given locale.\n *\n * using `en-US` locale: `P` => `MM/dd/yyyy`\n * using `en-US` locale: `p` => `hh:mm a`\n * using `pt-BR` locale: `P` => `dd/MM/yyyy`\n * using `pt-BR` locale: `p` => `HH:mm`\n *\n * Values will be assigned to the date in the descending order of its unit's priority.\n * Units of an equal priority overwrite each other in the order of appearance.\n *\n * If no values of higher priority are parsed (e.g. when parsing string 'January 1st' without a year),\n * the values will be taken from 3rd argument `referenceDate` which works as a context of parsing.\n *\n * `referenceDate` must be passed for correct work of the function.\n * If you're not sure which `referenceDate` to supply, create a new instance of Date:\n * `parse('02/11/2014', 'MM/dd/yyyy', new Date())`\n * In this case parsing will be done in the context of the current date.\n * If `referenceDate` is `Invalid Date` or a value not convertible to valid `Date`,\n * then `Invalid Date` will be returned.\n *\n * The result may vary by locale.\n *\n * If `formatString` matches with `dateString` but does not provides tokens, `referenceDate` will be returned.\n *\n * If parsing failed, `Invalid Date` will be returned.\n * Invalid Date is a Date, whose time value is NaN.\n * Time value of Date: http://es5.github.io/#x15.9.1.1\n *\n * @param {String} dateString - the string to parse\n * @param {String} formatString - the string of tokens\n * @param {Date|Number} referenceDate - defines values missing from the parsed dateString\n * @param {Object} [options] - an object with options.\n * @param {Locale} [options.locale=defaultLocale] - the locale object. See [Locale]{@link https://date-fns.org/docs/Locale}\n * @param {0|1|2|3|4|5|6} [options.weekStartsOn=0] - the index of the first day of the week (0 - Sunday)\n * @param {1|2|3|4|5|6|7} [options.firstWeekContainsDate=1] - the day of January, which is always in the first week of the year\n * @param {Boolean} [options.useAdditionalWeekYearTokens=false] - if true, allows usage of the week-numbering year tokens `YY` and `YYYY`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @param {Boolean} [options.useAdditionalDayOfYearTokens=false] - if true, allows usage of the day of year tokens `D` and `DD`;\n * see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @returns {Date} the parsed date\n * @throws {TypeError} 3 arguments required\n * @throws {RangeError} `options.weekStartsOn` must be between 0 and 6\n * @throws {RangeError} `options.firstWeekContainsDate` must be between 1 and 7\n * @throws {RangeError} `options.locale` must contain `match` property\n * @throws {RangeError} use `yyyy` instead of `YYYY` for formatting years using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `yy` instead of `YY` for formatting years using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `d` instead of `D` for formatting days of the month using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} use `dd` instead of `DD` for formatting days of the month using [format provided] to the input [input provided]; see: https://github.com/date-fns/date-fns/blob/master/docs/unicodeTokens.md\n * @throws {RangeError} format string contains an unescaped latin alphabet character\n *\n * @example\n * // Parse 11 February 2014 from middle-endian format:\n * var result = parse('02/11/2014', 'MM/dd/yyyy', new Date())\n * //=> Tue Feb 11 2014 00:00:00\n *\n * @example\n * // Parse 28th of February in Esperanto locale in the context of 2010 year:\n * import eo from 'date-fns/locale/eo'\n * var result = parse('28-a de februaro', \"do 'de' MMMM\", new Date(2010, 0, 1), {\n * locale: eo\n * })\n * //=> Sun Feb 28 2010 00:00:00\n */\nexport default function parse(dirtyDateString, dirtyFormatString, dirtyReferenceDate, options) {\n var _ref, _options$locale, _ref2, _ref3, _ref4, _options$firstWeekCon, _options$locale2, _options$locale2$opti, _defaultOptions$local, _defaultOptions$local2, _ref5, _ref6, _ref7, _options$weekStartsOn, _options$locale3, _options$locale3$opti, _defaultOptions$local3, _defaultOptions$local4;\n requiredArgs(3, arguments);\n var dateString = String(dirtyDateString);\n var formatString = String(dirtyFormatString);\n var defaultOptions = getDefaultOptions();\n var locale = (_ref = (_options$locale = options === null || options === void 0 ? void 0 : options.locale) !== null && _options$locale !== void 0 ? _options$locale : defaultOptions.locale) !== null && _ref !== void 0 ? _ref : defaultLocale;\n if (!locale.match) {\n throw new RangeError('locale must contain match property');\n }\n var firstWeekContainsDate = toInteger((_ref2 = (_ref3 = (_ref4 = (_options$firstWeekCon = options === null || options === void 0 ? void 0 : options.firstWeekContainsDate) !== null && _options$firstWeekCon !== void 0 ? _options$firstWeekCon : options === null || options === void 0 ? void 0 : (_options$locale2 = options.locale) === null || _options$locale2 === void 0 ? void 0 : (_options$locale2$opti = _options$locale2.options) === null || _options$locale2$opti === void 0 ? void 0 : _options$locale2$opti.firstWeekContainsDate) !== null && _ref4 !== void 0 ? _ref4 : defaultOptions.firstWeekContainsDate) !== null && _ref3 !== void 0 ? _ref3 : (_defaultOptions$local = defaultOptions.locale) === null || _defaultOptions$local === void 0 ? void 0 : (_defaultOptions$local2 = _defaultOptions$local.options) === null || _defaultOptions$local2 === void 0 ? void 0 : _defaultOptions$local2.firstWeekContainsDate) !== null && _ref2 !== void 0 ? _ref2 : 1);\n\n // Test if weekStartsOn is between 1 and 7 _and_ is not NaN\n if (!(firstWeekContainsDate >= 1 && firstWeekContainsDate <= 7)) {\n throw new RangeError('firstWeekContainsDate must be between 1 and 7 inclusively');\n }\n var weekStartsOn = toInteger((_ref5 = (_ref6 = (_ref7 = (_options$weekStartsOn = options === null || options === void 0 ? void 0 : options.weekStartsOn) !== null && _options$weekStartsOn !== void 0 ? _options$weekStartsOn : options === null || options === void 0 ? void 0 : (_options$locale3 = options.locale) === null || _options$locale3 === void 0 ? void 0 : (_options$locale3$opti = _options$locale3.options) === null || _options$locale3$opti === void 0 ? void 0 : _options$locale3$opti.weekStartsOn) !== null && _ref7 !== void 0 ? _ref7 : defaultOptions.weekStartsOn) !== null && _ref6 !== void 0 ? _ref6 : (_defaultOptions$local3 = defaultOptions.locale) === null || _defaultOptions$local3 === void 0 ? void 0 : (_defaultOptions$local4 = _defaultOptions$local3.options) === null || _defaultOptions$local4 === void 0 ? void 0 : _defaultOptions$local4.weekStartsOn) !== null && _ref5 !== void 0 ? _ref5 : 0);\n\n // Test if weekStartsOn is between 0 and 6 _and_ is not NaN\n if (!(weekStartsOn >= 0 && weekStartsOn <= 6)) {\n throw new RangeError('weekStartsOn must be between 0 and 6 inclusively');\n }\n if (formatString === '') {\n if (dateString === '') {\n return toDate(dirtyReferenceDate);\n } else {\n return new Date(NaN);\n }\n }\n var subFnOptions = {\n firstWeekContainsDate: firstWeekContainsDate,\n weekStartsOn: weekStartsOn,\n locale: locale\n };\n\n // If timezone isn't specified, it will be set to the system timezone\n var setters = [new DateToSystemTimezoneSetter()];\n var tokens = formatString.match(longFormattingTokensRegExp).map(function (substring) {\n var firstCharacter = substring[0];\n if (firstCharacter in longFormatters) {\n var longFormatter = longFormatters[firstCharacter];\n return longFormatter(substring, locale.formatLong);\n }\n return substring;\n }).join('').match(formattingTokensRegExp);\n var usedTokens = [];\n var _iterator = _createForOfIteratorHelper(tokens),\n _step;\n try {\n var _loop = function _loop() {\n var token = _step.value;\n if (!(options !== null && options !== void 0 && options.useAdditionalWeekYearTokens) && isProtectedWeekYearToken(token)) {\n throwProtectedError(token, formatString, dirtyDateString);\n }\n if (!(options !== null && options !== void 0 && options.useAdditionalDayOfYearTokens) && isProtectedDayOfYearToken(token)) {\n throwProtectedError(token, formatString, dirtyDateString);\n }\n var firstCharacter = token[0];\n var parser = parsers[firstCharacter];\n if (parser) {\n var incompatibleTokens = parser.incompatibleTokens;\n if (Array.isArray(incompatibleTokens)) {\n var incompatibleToken = usedTokens.find(function (usedToken) {\n return incompatibleTokens.includes(usedToken.token) || usedToken.token === firstCharacter;\n });\n if (incompatibleToken) {\n throw new RangeError(\"The format string mustn't contain `\".concat(incompatibleToken.fullToken, \"` and `\").concat(token, \"` at the same time\"));\n }\n } else if (parser.incompatibleTokens === '*' && usedTokens.length > 0) {\n throw new RangeError(\"The format string mustn't contain `\".concat(token, \"` and any other token at the same time\"));\n }\n usedTokens.push({\n token: firstCharacter,\n fullToken: token\n });\n var parseResult = parser.run(dateString, token, locale.match, subFnOptions);\n if (!parseResult) {\n return {\n v: new Date(NaN)\n };\n }\n setters.push(parseResult.setter);\n dateString = parseResult.rest;\n } else {\n if (firstCharacter.match(unescapedLatinCharacterRegExp)) {\n throw new RangeError('Format string contains an unescaped latin alphabet character `' + firstCharacter + '`');\n }\n\n // Replace two single quote characters with one single quote character\n if (token === \"''\") {\n token = \"'\";\n } else if (firstCharacter === \"'\") {\n token = cleanEscapedString(token);\n }\n\n // Cut token from string, or, if string doesn't match the token, return Invalid Date\n if (dateString.indexOf(token) === 0) {\n dateString = dateString.slice(token.length);\n } else {\n return {\n v: new Date(NaN)\n };\n }\n }\n };\n for (_iterator.s(); !(_step = _iterator.n()).done;) {\n var _ret = _loop();\n if (_typeof(_ret) === \"object\") return _ret.v;\n }\n\n // Check if the remaining input contains something other than whitespace\n } catch (err) {\n _iterator.e(err);\n } finally {\n _iterator.f();\n }\n if (dateString.length > 0 && notWhitespaceRegExp.test(dateString)) {\n return new Date(NaN);\n }\n var uniquePrioritySetters = setters.map(function (setter) {\n return setter.priority;\n }).sort(function (a, b) {\n return b - a;\n }).filter(function (priority, index, array) {\n return array.indexOf(priority) === index;\n }).map(function (priority) {\n return setters.filter(function (setter) {\n return setter.priority === priority;\n }).sort(function (a, b) {\n return b.subPriority - a.subPriority;\n });\n }).map(function (setterArray) {\n return setterArray[0];\n });\n var date = toDate(dirtyReferenceDate);\n if (isNaN(date.getTime())) {\n return new Date(NaN);\n }\n\n // Convert the date in system timezone to the same date in UTC+00:00 timezone.\n var utcDate = subMilliseconds(date, getTimezoneOffsetInMilliseconds(date));\n var flags = {};\n var _iterator2 = _createForOfIteratorHelper(uniquePrioritySetters),\n _step2;\n try {\n for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {\n var setter = _step2.value;\n if (!setter.validate(utcDate, subFnOptions)) {\n return new Date(NaN);\n }\n var result = setter.set(utcDate, flags, subFnOptions);\n // Result is tuple (date, flags)\n if (Array.isArray(result)) {\n utcDate = result[0];\n assign(flags, result[1]);\n // Result is date\n } else {\n utcDate = result;\n }\n }\n } catch (err) {\n _iterator2.e(err);\n } finally {\n _iterator2.f();\n }\n return utcDate;\n}\nfunction cleanEscapedString(input) {\n return input.match(escapedStringRegExp)[1].replace(doubleQuoteRegExp, \"'\");\n}","import { millisecondsInHour, millisecondsInMinute } from \"../constants/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\nimport toInteger from \"../_lib/toInteger/index.js\";\n/**\n * @name parseISO\n * @category Common Helpers\n * @summary Parse ISO string\n *\n * @description\n * Parse the given string in ISO 8601 format and return an instance of Date.\n *\n * Function accepts complete ISO 8601 formats as well as partial implementations.\n * ISO 8601: http://en.wikipedia.org/wiki/ISO_8601\n *\n * If the argument isn't a string, the function cannot parse the string or\n * the values are invalid, it returns Invalid Date.\n *\n * @param {String} argument - the value to convert\n * @param {Object} [options] - an object with options.\n * @param {0|1|2} [options.additionalDigits=2] - the additional number of digits in the extended year format\n * @returns {Date} the parsed date in the local time zone\n * @throws {TypeError} 1 argument required\n * @throws {RangeError} `options.additionalDigits` must be 0, 1 or 2\n *\n * @example\n * // Convert string '2014-02-11T11:30:30' to date:\n * const result = parseISO('2014-02-11T11:30:30')\n * //=> Tue Feb 11 2014 11:30:30\n *\n * @example\n * // Convert string '+02014101' to date,\n * // if the additional number of digits in the extended year format is 1:\n * const result = parseISO('+02014101', { additionalDigits: 1 })\n * //=> Fri Apr 11 2014 00:00:00\n */\nexport default function parseISO(argument, options) {\n var _options$additionalDi;\n requiredArgs(1, arguments);\n var additionalDigits = toInteger((_options$additionalDi = options === null || options === void 0 ? void 0 : options.additionalDigits) !== null && _options$additionalDi !== void 0 ? _options$additionalDi : 2);\n if (additionalDigits !== 2 && additionalDigits !== 1 && additionalDigits !== 0) {\n throw new RangeError('additionalDigits must be 0, 1 or 2');\n }\n if (!(typeof argument === 'string' || Object.prototype.toString.call(argument) === '[object String]')) {\n return new Date(NaN);\n }\n var dateStrings = splitDateString(argument);\n var date;\n if (dateStrings.date) {\n var parseYearResult = parseYear(dateStrings.date, additionalDigits);\n date = parseDate(parseYearResult.restDateString, parseYearResult.year);\n }\n if (!date || isNaN(date.getTime())) {\n return new Date(NaN);\n }\n var timestamp = date.getTime();\n var time = 0;\n var offset;\n if (dateStrings.time) {\n time = parseTime(dateStrings.time);\n if (isNaN(time)) {\n return new Date(NaN);\n }\n }\n if (dateStrings.timezone) {\n offset = parseTimezone(dateStrings.timezone);\n if (isNaN(offset)) {\n return new Date(NaN);\n }\n } else {\n var dirtyDate = new Date(timestamp + time);\n // js parsed string assuming it's in UTC timezone\n // but we need it to be parsed in our timezone\n // so we use utc values to build date in our timezone.\n // Year values from 0 to 99 map to the years 1900 to 1999\n // so set year explicitly with setFullYear.\n var result = new Date(0);\n result.setFullYear(dirtyDate.getUTCFullYear(), dirtyDate.getUTCMonth(), dirtyDate.getUTCDate());\n result.setHours(dirtyDate.getUTCHours(), dirtyDate.getUTCMinutes(), dirtyDate.getUTCSeconds(), dirtyDate.getUTCMilliseconds());\n return result;\n }\n return new Date(timestamp + time + offset);\n}\nvar patterns = {\n dateTimeDelimiter: /[T ]/,\n timeZoneDelimiter: /[Z ]/i,\n timezone: /([Z+-].*)$/\n};\nvar dateRegex = /^-?(?:(\\d{3})|(\\d{2})(?:-?(\\d{2}))?|W(\\d{2})(?:-?(\\d{1}))?|)$/;\nvar timeRegex = /^(\\d{2}(?:[.,]\\d*)?)(?::?(\\d{2}(?:[.,]\\d*)?))?(?::?(\\d{2}(?:[.,]\\d*)?))?$/;\nvar timezoneRegex = /^([+-])(\\d{2})(?::?(\\d{2}))?$/;\nfunction splitDateString(dateString) {\n var dateStrings = {};\n var array = dateString.split(patterns.dateTimeDelimiter);\n var timeString;\n\n // The regex match should only return at maximum two array elements.\n // [date], [time], or [date, time].\n if (array.length > 2) {\n return dateStrings;\n }\n if (/:/.test(array[0])) {\n timeString = array[0];\n } else {\n dateStrings.date = array[0];\n timeString = array[1];\n if (patterns.timeZoneDelimiter.test(dateStrings.date)) {\n dateStrings.date = dateString.split(patterns.timeZoneDelimiter)[0];\n timeString = dateString.substr(dateStrings.date.length, dateString.length);\n }\n }\n if (timeString) {\n var token = patterns.timezone.exec(timeString);\n if (token) {\n dateStrings.time = timeString.replace(token[1], '');\n dateStrings.timezone = token[1];\n } else {\n dateStrings.time = timeString;\n }\n }\n return dateStrings;\n}\nfunction parseYear(dateString, additionalDigits) {\n var regex = new RegExp('^(?:(\\\\d{4}|[+-]\\\\d{' + (4 + additionalDigits) + '})|(\\\\d{2}|[+-]\\\\d{' + (2 + additionalDigits) + '})$)');\n var captures = dateString.match(regex);\n // Invalid ISO-formatted year\n if (!captures) return {\n year: NaN,\n restDateString: ''\n };\n var year = captures[1] ? parseInt(captures[1]) : null;\n var century = captures[2] ? parseInt(captures[2]) : null;\n\n // either year or century is null, not both\n return {\n year: century === null ? year : century * 100,\n restDateString: dateString.slice((captures[1] || captures[2]).length)\n };\n}\nfunction parseDate(dateString, year) {\n // Invalid ISO-formatted year\n if (year === null) return new Date(NaN);\n var captures = dateString.match(dateRegex);\n // Invalid ISO-formatted string\n if (!captures) return new Date(NaN);\n var isWeekDate = !!captures[4];\n var dayOfYear = parseDateUnit(captures[1]);\n var month = parseDateUnit(captures[2]) - 1;\n var day = parseDateUnit(captures[3]);\n var week = parseDateUnit(captures[4]);\n var dayOfWeek = parseDateUnit(captures[5]) - 1;\n if (isWeekDate) {\n if (!validateWeekDate(year, week, dayOfWeek)) {\n return new Date(NaN);\n }\n return dayOfISOWeekYear(year, week, dayOfWeek);\n } else {\n var date = new Date(0);\n if (!validateDate(year, month, day) || !validateDayOfYearDate(year, dayOfYear)) {\n return new Date(NaN);\n }\n date.setUTCFullYear(year, month, Math.max(dayOfYear, day));\n return date;\n }\n}\nfunction parseDateUnit(value) {\n return value ? parseInt(value) : 1;\n}\nfunction parseTime(timeString) {\n var captures = timeString.match(timeRegex);\n if (!captures) return NaN; // Invalid ISO-formatted time\n\n var hours = parseTimeUnit(captures[1]);\n var minutes = parseTimeUnit(captures[2]);\n var seconds = parseTimeUnit(captures[3]);\n if (!validateTime(hours, minutes, seconds)) {\n return NaN;\n }\n return hours * millisecondsInHour + minutes * millisecondsInMinute + seconds * 1000;\n}\nfunction parseTimeUnit(value) {\n return value && parseFloat(value.replace(',', '.')) || 0;\n}\nfunction parseTimezone(timezoneString) {\n if (timezoneString === 'Z') return 0;\n var captures = timezoneString.match(timezoneRegex);\n if (!captures) return 0;\n var sign = captures[1] === '+' ? -1 : 1;\n var hours = parseInt(captures[2]);\n var minutes = captures[3] && parseInt(captures[3]) || 0;\n if (!validateTimezone(hours, minutes)) {\n return NaN;\n }\n return sign * (hours * millisecondsInHour + minutes * millisecondsInMinute);\n}\nfunction dayOfISOWeekYear(isoWeekYear, week, day) {\n var date = new Date(0);\n date.setUTCFullYear(isoWeekYear, 0, 4);\n var fourthOfJanuaryDay = date.getUTCDay() || 7;\n var diff = (week - 1) * 7 + day + 1 - fourthOfJanuaryDay;\n date.setUTCDate(date.getUTCDate() + diff);\n return date;\n}\n\n// Validation functions\n\n// February is null to handle the leap year (using ||)\nvar daysInMonths = [31, null, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];\nfunction isLeapYearIndex(year) {\n return year % 400 === 0 || year % 4 === 0 && year % 100 !== 0;\n}\nfunction validateDate(year, month, date) {\n return month >= 0 && month <= 11 && date >= 1 && date <= (daysInMonths[month] || (isLeapYearIndex(year) ? 29 : 28));\n}\nfunction validateDayOfYearDate(year, dayOfYear) {\n return dayOfYear >= 1 && dayOfYear <= (isLeapYearIndex(year) ? 366 : 365);\n}\nfunction validateWeekDate(_year, week, day) {\n return week >= 1 && week <= 53 && day >= 0 && day <= 6;\n}\nfunction validateTime(hours, minutes, seconds) {\n if (hours === 24) {\n return minutes === 0 && seconds === 0;\n }\n return seconds >= 0 && seconds < 60 && minutes >= 0 && minutes < 60 && hours >= 0 && hours < 25;\n}\nfunction validateTimezone(_hours, minutes) {\n return minutes >= 0 && minutes <= 59;\n}","import toDate from \"../toDate/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name getDaysInMonth\n * @category Month Helpers\n * @summary Get the number of days in a month of the given date.\n *\n * @description\n * Get the number of days in a month of the given date.\n *\n * @param {Date|Number} date - the given date\n * @returns {Number} the number of days in a month\n * @throws {TypeError} 1 argument required\n *\n * @example\n * // How many days are in February 2000?\n * const result = getDaysInMonth(new Date(2000, 1))\n * //=> 29\n */\nexport default function getDaysInMonth(dirtyDate) {\n requiredArgs(1, arguments);\n var date = toDate(dirtyDate);\n var year = date.getFullYear();\n var monthIndex = date.getMonth();\n var lastDayOfMonth = new Date(0);\n lastDayOfMonth.setFullYear(year, monthIndex + 1, 0);\n lastDayOfMonth.setHours(0, 0, 0, 0);\n return lastDayOfMonth.getDate();\n}","import toInteger from \"../_lib/toInteger/index.js\";\nimport toDate from \"../toDate/index.js\";\nimport getDaysInMonth from \"../getDaysInMonth/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name setMonth\n * @category Month Helpers\n * @summary Set the month to the given date.\n *\n * @description\n * Set the month to the given date.\n *\n * @param {Date|Number} date - the date to be changed\n * @param {Number} month - the month of the new date\n * @returns {Date} the new date with the month set\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // Set February to 1 September 2014:\n * const result = setMonth(new Date(2014, 8, 1), 1)\n * //=> Sat Feb 01 2014 00:00:00\n */\nexport default function setMonth(dirtyDate, dirtyMonth) {\n requiredArgs(2, arguments);\n var date = toDate(dirtyDate);\n var month = toInteger(dirtyMonth);\n var year = date.getFullYear();\n var day = date.getDate();\n var dateWithDesiredMonth = new Date(0);\n dateWithDesiredMonth.setFullYear(year, month, 15);\n dateWithDesiredMonth.setHours(0, 0, 0, 0);\n var daysInMonth = getDaysInMonth(dateWithDesiredMonth);\n // Set the last day of the new month\n // if the original date was the last day of the longer month\n date.setMonth(month, Math.min(day, daysInMonth));\n return date;\n}","import _typeof from \"@babel/runtime/helpers/esm/typeof\";\nimport toDate from \"../toDate/index.js\";\nimport setMonth from \"../setMonth/index.js\";\nimport toInteger from \"../_lib/toInteger/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name set\n * @category Common Helpers\n * @summary Set date values to a given date.\n *\n * @description\n * Set date values to a given date.\n *\n * Sets time values to date from object `values`.\n * A value is not set if it is undefined or null or doesn't exist in `values`.\n *\n * Note about bundle size: `set` does not internally use `setX` functions from date-fns but instead opts\n * to use native `Date#setX` methods. If you use this function, you may not want to include the\n * other `setX` functions that date-fns provides if you are concerned about the bundle size.\n *\n * @param {Date|Number} date - the date to be changed\n * @param {Object} values - an object with options\n * @param {Number} [values.year] - the number of years to be set\n * @param {Number} [values.month] - the number of months to be set\n * @param {Number} [values.date] - the number of days to be set\n * @param {Number} [values.hours] - the number of hours to be set\n * @param {Number} [values.minutes] - the number of minutes to be set\n * @param {Number} [values.seconds] - the number of seconds to be set\n * @param {Number} [values.milliseconds] - the number of milliseconds to be set\n * @returns {Date} the new date with options set\n * @throws {TypeError} 2 arguments required\n * @throws {RangeError} `values` must be an object\n *\n * @example\n * // Transform 1 September 2014 into 20 October 2015 in a single line:\n * const result = set(new Date(2014, 8, 20), { year: 2015, month: 9, date: 20 })\n * //=> Tue Oct 20 2015 00:00:00\n *\n * @example\n * // Set 12 PM to 1 September 2014 01:23:45 to 1 September 2014 12:00:00:\n * const result = set(new Date(2014, 8, 1, 1, 23, 45), { hours: 12 })\n * //=> Mon Sep 01 2014 12:23:45\n */\nexport default function set(dirtyDate, values) {\n requiredArgs(2, arguments);\n if (_typeof(values) !== 'object' || values === null) {\n throw new RangeError('values parameter must be an object');\n }\n var date = toDate(dirtyDate);\n\n // Check if date is Invalid Date because Date.prototype.setFullYear ignores the value of Invalid Date\n if (isNaN(date.getTime())) {\n return new Date(NaN);\n }\n if (values.year != null) {\n date.setFullYear(values.year);\n }\n if (values.month != null) {\n date = setMonth(date, values.month);\n }\n if (values.date != null) {\n date.setDate(toInteger(values.date));\n }\n if (values.hours != null) {\n date.setHours(toInteger(values.hours));\n }\n if (values.minutes != null) {\n date.setMinutes(toInteger(values.minutes));\n }\n if (values.seconds != null) {\n date.setSeconds(toInteger(values.seconds));\n }\n if (values.milliseconds != null) {\n date.setMilliseconds(toInteger(values.milliseconds));\n }\n return date;\n}","import addMilliseconds from \"../addMilliseconds/index.js\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\nimport toInteger from \"../_lib/toInteger/index.js\";\n/**\n * @name subMilliseconds\n * @category Millisecond Helpers\n * @summary Subtract the specified number of milliseconds from the given date.\n *\n * @description\n * Subtract the specified number of milliseconds from the given date.\n *\n * @param {Date|Number} date - the date to be changed\n * @param {Number} amount - the amount of milliseconds to be subtracted. Positive decimals will be rounded using `Math.floor`, decimals less than zero will be rounded using `Math.ceil`.\n * @returns {Date} the new date with the milliseconds subtracted\n * @throws {TypeError} 2 arguments required\n *\n * @example\n * // Subtract 750 milliseconds from 10 July 2014 12:45:30.000:\n * const result = subMilliseconds(new Date(2014, 6, 10, 12, 45, 30, 0), 750)\n * //=> Thu Jul 10 2014 12:45:29.250\n */\nexport default function subMilliseconds(dirtyDate, dirtyAmount) {\n requiredArgs(2, arguments);\n var amount = toInteger(dirtyAmount);\n return addMilliseconds(dirtyDate, -amount);\n}","import _typeof from \"@babel/runtime/helpers/esm/typeof\";\nimport requiredArgs from \"../_lib/requiredArgs/index.js\";\n/**\n * @name toDate\n * @category Common Helpers\n * @summary Convert the given argument to an instance of Date.\n *\n * @description\n * Convert the given argument to an instance of Date.\n *\n * If the argument is an instance of Date, the function returns its clone.\n *\n * If the argument is a number, it is treated as a timestamp.\n *\n * If the argument is none of the above, the function returns Invalid Date.\n *\n * **Note**: *all* Date arguments passed to any *date-fns* function is processed by `toDate`.\n *\n * @param {Date|Number} argument - the value to convert\n * @returns {Date} the parsed date in the local time zone\n * @throws {TypeError} 1 argument required\n *\n * @example\n * // Clone the date:\n * const result = toDate(new Date(2014, 1, 11, 11, 30, 30))\n * //=> Tue Feb 11 2014 11:30:30\n *\n * @example\n * // Convert the timestamp to date:\n * const result = toDate(1392098430000)\n * //=> Tue Feb 11 2014 11:30:30\n */\nexport default function toDate(argument) {\n requiredArgs(1, arguments);\n var argStr = Object.prototype.toString.call(argument);\n\n // Clone the date\n if (argument instanceof Date || _typeof(argument) === 'object' && argStr === '[object Date]') {\n // Prevent the date to lose the milliseconds when passed to new Date() in IE10\n return new Date(argument.getTime());\n } else if (typeof argument === 'number' || argStr === '[object Number]') {\n return new Date(argument);\n } else {\n if ((typeof argument === 'string' || argStr === '[object String]') && typeof console !== 'undefined') {\n // eslint-disable-next-line no-console\n console.warn(\"Starting with v2.0.0-beta.1 date-fns doesn't accept strings as date arguments. Please use `parseISO` to parse strings. See: https://github.com/date-fns/date-fns/blob/master/docs/upgradeGuide.md#string-arguments\");\n // eslint-disable-next-line no-console\n console.warn(new Error().stack);\n }\n return new Date(NaN);\n }\n}","/*! @license DOMPurify 2.5.8 | (c) Cure53 and other contributors | Released under the Apache license 2.0 and Mozilla Public License 2.0 | github.com/cure53/DOMPurify/blob/2.5.8/LICENSE */\n\n(function (global, factory) {\n typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :\n typeof define === 'function' && define.amd ? define(factory) :\n (global = typeof globalThis !== 'undefined' ? globalThis : global || self, global.DOMPurify = factory());\n})(this, (function () { 'use strict';\n\n function _typeof(obj) {\n \"@babel/helpers - typeof\";\n\n return _typeof = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (obj) {\n return typeof obj;\n } : function (obj) {\n return obj && \"function\" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj;\n }, _typeof(obj);\n }\n function _setPrototypeOf(o, p) {\n _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) {\n o.__proto__ = p;\n return o;\n };\n return _setPrototypeOf(o, p);\n }\n function _isNativeReflectConstruct() {\n if (typeof Reflect === \"undefined\" || !Reflect.construct) return false;\n if (Reflect.construct.sham) return false;\n if (typeof Proxy === \"function\") return true;\n try {\n Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {}));\n return true;\n } catch (e) {\n return false;\n }\n }\n function _construct(Parent, args, Class) {\n if (_isNativeReflectConstruct()) {\n _construct = Reflect.construct;\n } else {\n _construct = function _construct(Parent, args, Class) {\n var a = [null];\n a.push.apply(a, args);\n var Constructor = Function.bind.apply(Parent, a);\n var instance = new Constructor();\n if (Class) _setPrototypeOf(instance, Class.prototype);\n return instance;\n };\n }\n return _construct.apply(null, arguments);\n }\n function _toConsumableArray(arr) {\n return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread();\n }\n function _arrayWithoutHoles(arr) {\n if (Array.isArray(arr)) return _arrayLikeToArray(arr);\n }\n function _iterableToArray(iter) {\n if (typeof Symbol !== \"undefined\" && iter[Symbol.iterator] != null || iter[\"@@iterator\"] != null) return Array.from(iter);\n }\n function _unsupportedIterableToArray(o, minLen) {\n if (!o) return;\n if (typeof o === \"string\") return _arrayLikeToArray(o, minLen);\n var n = Object.prototype.toString.call(o).slice(8, -1);\n if (n === \"Object\" && o.constructor) n = o.constructor.name;\n if (n === \"Map\" || n === \"Set\") return Array.from(o);\n if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen);\n }\n function _arrayLikeToArray(arr, len) {\n if (len == null || len > arr.length) len = arr.length;\n for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i];\n return arr2;\n }\n function _nonIterableSpread() {\n throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\");\n }\n\n var hasOwnProperty = Object.hasOwnProperty,\n setPrototypeOf = Object.setPrototypeOf,\n isFrozen = Object.isFrozen,\n getPrototypeOf = Object.getPrototypeOf,\n getOwnPropertyDescriptor = Object.getOwnPropertyDescriptor;\n var freeze = Object.freeze,\n seal = Object.seal,\n create = Object.create; // eslint-disable-line import/no-mutable-exports\n var _ref = typeof Reflect !== 'undefined' && Reflect,\n apply = _ref.apply,\n construct = _ref.construct;\n if (!apply) {\n apply = function apply(fun, thisValue, args) {\n return fun.apply(thisValue, args);\n };\n }\n if (!freeze) {\n freeze = function freeze(x) {\n return x;\n };\n }\n if (!seal) {\n seal = function seal(x) {\n return x;\n };\n }\n if (!construct) {\n construct = function construct(Func, args) {\n return _construct(Func, _toConsumableArray(args));\n };\n }\n var arrayForEach = unapply(Array.prototype.forEach);\n var arrayPop = unapply(Array.prototype.pop);\n var arrayPush = unapply(Array.prototype.push);\n var stringToLowerCase = unapply(String.prototype.toLowerCase);\n var stringToString = unapply(String.prototype.toString);\n var stringMatch = unapply(String.prototype.match);\n var stringReplace = unapply(String.prototype.replace);\n var stringIndexOf = unapply(String.prototype.indexOf);\n var stringTrim = unapply(String.prototype.trim);\n var regExpTest = unapply(RegExp.prototype.test);\n var typeErrorCreate = unconstruct(TypeError);\n function unapply(func) {\n return function (thisArg) {\n for (var _len = arguments.length, args = new Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {\n args[_key - 1] = arguments[_key];\n }\n return apply(func, thisArg, args);\n };\n }\n function unconstruct(func) {\n return function () {\n for (var _len2 = arguments.length, args = new Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {\n args[_key2] = arguments[_key2];\n }\n return construct(func, args);\n };\n }\n\n /* Add properties to a lookup table */\n function addToSet(set, array, transformCaseFunc) {\n var _transformCaseFunc;\n transformCaseFunc = (_transformCaseFunc = transformCaseFunc) !== null && _transformCaseFunc !== void 0 ? _transformCaseFunc : stringToLowerCase;\n if (setPrototypeOf) {\n // Make 'in' and truthy checks like Boolean(set.constructor)\n // independent of any properties defined on Object.prototype.\n // Prevent prototype setters from intercepting set as a this value.\n setPrototypeOf(set, null);\n }\n var l = array.length;\n while (l--) {\n var element = array[l];\n if (typeof element === 'string') {\n var lcElement = transformCaseFunc(element);\n if (lcElement !== element) {\n // Config presets (e.g. tags.js, attrs.js) are immutable.\n if (!isFrozen(array)) {\n array[l] = lcElement;\n }\n element = lcElement;\n }\n }\n set[element] = true;\n }\n return set;\n }\n\n /* Shallow clone an object */\n function clone(object) {\n var newObject = create(null);\n var property;\n for (property in object) {\n if (apply(hasOwnProperty, object, [property]) === true) {\n newObject[property] = object[property];\n }\n }\n return newObject;\n }\n\n /* IE10 doesn't support __lookupGetter__ so lets'\n * simulate it. It also automatically checks\n * if the prop is function or getter and behaves\n * accordingly. */\n function lookupGetter(object, prop) {\n while (object !== null) {\n var desc = getOwnPropertyDescriptor(object, prop);\n if (desc) {\n if (desc.get) {\n return unapply(desc.get);\n }\n if (typeof desc.value === 'function') {\n return unapply(desc.value);\n }\n }\n object = getPrototypeOf(object);\n }\n function fallbackValue(element) {\n console.warn('fallback value for', element);\n return null;\n }\n return fallbackValue;\n }\n\n var html$1 = freeze(['a', 'abbr', 'acronym', 'address', 'area', 'article', 'aside', 'audio', 'b', 'bdi', 'bdo', 'big', 'blink', 'blockquote', 'body', 'br', 'button', 'canvas', 'caption', 'center', 'cite', 'code', 'col', 'colgroup', 'content', 'data', 'datalist', 'dd', 'decorator', 'del', 'details', 'dfn', 'dialog', 'dir', 'div', 'dl', 'dt', 'element', 'em', 'fieldset', 'figcaption', 'figure', 'font', 'footer', 'form', 'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'head', 'header', 'hgroup', 'hr', 'html', 'i', 'img', 'input', 'ins', 'kbd', 'label', 'legend', 'li', 'main', 'map', 'mark', 'marquee', 'menu', 'menuitem', 'meter', 'nav', 'nobr', 'ol', 'optgroup', 'option', 'output', 'p', 'picture', 'pre', 'progress', 'q', 'rp', 'rt', 'ruby', 's', 'samp', 'section', 'select', 'shadow', 'small', 'source', 'spacer', 'span', 'strike', 'strong', 'style', 'sub', 'summary', 'sup', 'table', 'tbody', 'td', 'template', 'textarea', 'tfoot', 'th', 'thead', 'time', 'tr', 'track', 'tt', 'u', 'ul', 'var', 'video', 'wbr']);\n\n // SVG\n var svg$1 = freeze(['svg', 'a', 'altglyph', 'altglyphdef', 'altglyphitem', 'animatecolor', 'animatemotion', 'animatetransform', 'circle', 'clippath', 'defs', 'desc', 'ellipse', 'filter', 'font', 'g', 'glyph', 'glyphref', 'hkern', 'image', 'line', 'lineargradient', 'marker', 'mask', 'metadata', 'mpath', 'path', 'pattern', 'polygon', 'polyline', 'radialgradient', 'rect', 'stop', 'style', 'switch', 'symbol', 'text', 'textpath', 'title', 'tref', 'tspan', 'view', 'vkern']);\n var svgFilters = freeze(['feBlend', 'feColorMatrix', 'feComponentTransfer', 'feComposite', 'feConvolveMatrix', 'feDiffuseLighting', 'feDisplacementMap', 'feDistantLight', 'feFlood', 'feFuncA', 'feFuncB', 'feFuncG', 'feFuncR', 'feGaussianBlur', 'feImage', 'feMerge', 'feMergeNode', 'feMorphology', 'feOffset', 'fePointLight', 'feSpecularLighting', 'feSpotLight', 'feTile', 'feTurbulence']);\n\n // List of SVG elements that are disallowed by default.\n // We still need to know them so that we can do namespace\n // checks properly in case one wants to add them to\n // allow-list.\n var svgDisallowed = freeze(['animate', 'color-profile', 'cursor', 'discard', 'fedropshadow', 'font-face', 'font-face-format', 'font-face-name', 'font-face-src', 'font-face-uri', 'foreignobject', 'hatch', 'hatchpath', 'mesh', 'meshgradient', 'meshpatch', 'meshrow', 'missing-glyph', 'script', 'set', 'solidcolor', 'unknown', 'use']);\n var mathMl$1 = freeze(['math', 'menclose', 'merror', 'mfenced', 'mfrac', 'mglyph', 'mi', 'mlabeledtr', 'mmultiscripts', 'mn', 'mo', 'mover', 'mpadded', 'mphantom', 'mroot', 'mrow', 'ms', 'mspace', 'msqrt', 'mstyle', 'msub', 'msup', 'msubsup', 'mtable', 'mtd', 'mtext', 'mtr', 'munder', 'munderover']);\n\n // Similarly to SVG, we want to know all MathML elements,\n // even those that we disallow by default.\n var mathMlDisallowed = freeze(['maction', 'maligngroup', 'malignmark', 'mlongdiv', 'mscarries', 'mscarry', 'msgroup', 'mstack', 'msline', 'msrow', 'semantics', 'annotation', 'annotation-xml', 'mprescripts', 'none']);\n var text = freeze(['#text']);\n\n var html = freeze(['accept', 'action', 'align', 'alt', 'autocapitalize', 'autocomplete', 'autopictureinpicture', 'autoplay', 'background', 'bgcolor', 'border', 'capture', 'cellpadding', 'cellspacing', 'checked', 'cite', 'class', 'clear', 'color', 'cols', 'colspan', 'controls', 'controlslist', 'coords', 'crossorigin', 'datetime', 'decoding', 'default', 'dir', 'disabled', 'disablepictureinpicture', 'disableremoteplayback', 'download', 'draggable', 'enctype', 'enterkeyhint', 'face', 'for', 'headers', 'height', 'hidden', 'high', 'href', 'hreflang', 'id', 'inputmode', 'integrity', 'ismap', 'kind', 'label', 'lang', 'list', 'loading', 'loop', 'low', 'max', 'maxlength', 'media', 'method', 'min', 'minlength', 'multiple', 'muted', 'name', 'nonce', 'noshade', 'novalidate', 'nowrap', 'open', 'optimum', 'pattern', 'placeholder', 'playsinline', 'poster', 'preload', 'pubdate', 'radiogroup', 'readonly', 'rel', 'required', 'rev', 'reversed', 'role', 'rows', 'rowspan', 'spellcheck', 'scope', 'selected', 'shape', 'size', 'sizes', 'span', 'srclang', 'start', 'src', 'srcset', 'step', 'style', 'summary', 'tabindex', 'title', 'translate', 'type', 'usemap', 'valign', 'value', 'width', 'xmlns', 'slot']);\n var svg = freeze(['accent-height', 'accumulate', 'additive', 'alignment-baseline', 'ascent', 'attributename', 'attributetype', 'azimuth', 'basefrequency', 'baseline-shift', 'begin', 'bias', 'by', 'class', 'clip', 'clippathunits', 'clip-path', 'clip-rule', 'color', 'color-interpolation', 'color-interpolation-filters', 'color-profile', 'color-rendering', 'cx', 'cy', 'd', 'dx', 'dy', 'diffuseconstant', 'direction', 'display', 'divisor', 'dur', 'edgemode', 'elevation', 'end', 'fill', 'fill-opacity', 'fill-rule', 'filter', 'filterunits', 'flood-color', 'flood-opacity', 'font-family', 'font-size', 'font-size-adjust', 'font-stretch', 'font-style', 'font-variant', 'font-weight', 'fx', 'fy', 'g1', 'g2', 'glyph-name', 'glyphref', 'gradientunits', 'gradienttransform', 'height', 'href', 'id', 'image-rendering', 'in', 'in2', 'k', 'k1', 'k2', 'k3', 'k4', 'kerning', 'keypoints', 'keysplines', 'keytimes', 'lang', 'lengthadjust', 'letter-spacing', 'kernelmatrix', 'kernelunitlength', 'lighting-color', 'local', 'marker-end', 'marker-mid', 'marker-start', 'markerheight', 'markerunits', 'markerwidth', 'maskcontentunits', 'maskunits', 'max', 'mask', 'media', 'method', 'mode', 'min', 'name', 'numoctaves', 'offset', 'operator', 'opacity', 'order', 'orient', 'orientation', 'origin', 'overflow', 'paint-order', 'path', 'pathlength', 'patterncontentunits', 'patterntransform', 'patternunits', 'points', 'preservealpha', 'preserveaspectratio', 'primitiveunits', 'r', 'rx', 'ry', 'radius', 'refx', 'refy', 'repeatcount', 'repeatdur', 'restart', 'result', 'rotate', 'scale', 'seed', 'shape-rendering', 'specularconstant', 'specularexponent', 'spreadmethod', 'startoffset', 'stddeviation', 'stitchtiles', 'stop-color', 'stop-opacity', 'stroke-dasharray', 'stroke-dashoffset', 'stroke-linecap', 'stroke-linejoin', 'stroke-miterlimit', 'stroke-opacity', 'stroke', 'stroke-width', 'style', 'surfacescale', 'systemlanguage', 'tabindex', 'targetx', 'targety', 'transform', 'transform-origin', 'text-anchor', 'text-decoration', 'text-rendering', 'textlength', 'type', 'u1', 'u2', 'unicode', 'values', 'viewbox', 'visibility', 'version', 'vert-adv-y', 'vert-origin-x', 'vert-origin-y', 'width', 'word-spacing', 'wrap', 'writing-mode', 'xchannelselector', 'ychannelselector', 'x', 'x1', 'x2', 'xmlns', 'y', 'y1', 'y2', 'z', 'zoomandpan']);\n var mathMl = freeze(['accent', 'accentunder', 'align', 'bevelled', 'close', 'columnsalign', 'columnlines', 'columnspan', 'denomalign', 'depth', 'dir', 'display', 'displaystyle', 'encoding', 'fence', 'frame', 'height', 'href', 'id', 'largeop', 'length', 'linethickness', 'lspace', 'lquote', 'mathbackground', 'mathcolor', 'mathsize', 'mathvariant', 'maxsize', 'minsize', 'movablelimits', 'notation', 'numalign', 'open', 'rowalign', 'rowlines', 'rowspacing', 'rowspan', 'rspace', 'rquote', 'scriptlevel', 'scriptminsize', 'scriptsizemultiplier', 'selection', 'separator', 'separators', 'stretchy', 'subscriptshift', 'supscriptshift', 'symmetric', 'voffset', 'width', 'xmlns']);\n var xml = freeze(['xlink:href', 'xml:id', 'xlink:title', 'xml:space', 'xmlns:xlink']);\n\n // eslint-disable-next-line unicorn/better-regex\n var MUSTACHE_EXPR = seal(/\\{\\{[\\w\\W]*|[\\w\\W]*\\}\\}/gm); // Specify template detection regex for SAFE_FOR_TEMPLATES mode\n var ERB_EXPR = seal(/<%[\\w\\W]*|[\\w\\W]*%>/gm);\n var TMPLIT_EXPR = seal(/\\${[\\w\\W]*}/gm);\n var DATA_ATTR = seal(/^data-[\\-\\w.\\u00B7-\\uFFFF]+$/); // eslint-disable-line no-useless-escape\n var ARIA_ATTR = seal(/^aria-[\\-\\w]+$/); // eslint-disable-line no-useless-escape\n var IS_ALLOWED_URI = seal(/^(?:(?:(?:f|ht)tps?|mailto|tel|callto|cid|xmpp):|[^a-z]|[a-z+.\\-]+(?:[^a-z+.\\-:]|$))/i // eslint-disable-line no-useless-escape\n );\n var IS_SCRIPT_OR_DATA = seal(/^(?:\\w+script|data):/i);\n var ATTR_WHITESPACE = seal(/[\\u0000-\\u0020\\u00A0\\u1680\\u180E\\u2000-\\u2029\\u205F\\u3000]/g // eslint-disable-line no-control-regex\n );\n var DOCTYPE_NAME = seal(/^html$/i);\n var CUSTOM_ELEMENT = seal(/^[a-z][.\\w]*(-[.\\w]+)+$/i);\n\n var getGlobal = function getGlobal() {\n return typeof window === 'undefined' ? null : window;\n };\n\n /**\n * Creates a no-op policy for internal use only.\n * Don't export this function outside this module!\n * @param {?TrustedTypePolicyFactory} trustedTypes The policy factory.\n * @param {Document} document The document object (to determine policy name suffix)\n * @return {?TrustedTypePolicy} The policy created (or null, if Trusted Types\n * are not supported).\n */\n var _createTrustedTypesPolicy = function _createTrustedTypesPolicy(trustedTypes, document) {\n if (_typeof(trustedTypes) !== 'object' || typeof trustedTypes.createPolicy !== 'function') {\n return null;\n }\n\n // Allow the callers to control the unique policy name\n // by adding a data-tt-policy-suffix to the script element with the DOMPurify.\n // Policy creation with duplicate names throws in Trusted Types.\n var suffix = null;\n var ATTR_NAME = 'data-tt-policy-suffix';\n if (document.currentScript && document.currentScript.hasAttribute(ATTR_NAME)) {\n suffix = document.currentScript.getAttribute(ATTR_NAME);\n }\n var policyName = 'dompurify' + (suffix ? '#' + suffix : '');\n try {\n return trustedTypes.createPolicy(policyName, {\n createHTML: function createHTML(html) {\n return html;\n },\n createScriptURL: function createScriptURL(scriptUrl) {\n return scriptUrl;\n }\n });\n } catch (_) {\n // Policy creation failed (most likely another DOMPurify script has\n // already run). Skip creating the policy, as this will only cause errors\n // if TT are enforced.\n console.warn('TrustedTypes policy ' + policyName + ' could not be created.');\n return null;\n }\n };\n function createDOMPurify() {\n var window = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : getGlobal();\n var DOMPurify = function DOMPurify(root) {\n return createDOMPurify(root);\n };\n\n /**\n * Version label, exposed for easier checks\n * if DOMPurify is up to date or not\n */\n DOMPurify.version = '2.5.8';\n\n /**\n * Array of elements that DOMPurify removed during sanitation.\n * Empty if nothing was removed.\n */\n DOMPurify.removed = [];\n if (!window || !window.document || window.document.nodeType !== 9) {\n // Not running in a browser, provide a factory function\n // so that you can pass your own Window\n DOMPurify.isSupported = false;\n return DOMPurify;\n }\n var originalDocument = window.document;\n var document = window.document;\n var DocumentFragment = window.DocumentFragment,\n HTMLTemplateElement = window.HTMLTemplateElement,\n Node = window.Node,\n Element = window.Element,\n NodeFilter = window.NodeFilter,\n _window$NamedNodeMap = window.NamedNodeMap,\n NamedNodeMap = _window$NamedNodeMap === void 0 ? window.NamedNodeMap || window.MozNamedAttrMap : _window$NamedNodeMap,\n HTMLFormElement = window.HTMLFormElement,\n DOMParser = window.DOMParser,\n trustedTypes = window.trustedTypes;\n var ElementPrototype = Element.prototype;\n var cloneNode = lookupGetter(ElementPrototype, 'cloneNode');\n var getNextSibling = lookupGetter(ElementPrototype, 'nextSibling');\n var getChildNodes = lookupGetter(ElementPrototype, 'childNodes');\n var getParentNode = lookupGetter(ElementPrototype, 'parentNode');\n\n // As per issue #47, the web-components registry is inherited by a\n // new document created via createHTMLDocument. As per the spec\n // (http://w3c.github.io/webcomponents/spec/custom/#creating-and-passing-registries)\n // a new empty registry is used when creating a template contents owner\n // document, so we use that as our parent document to ensure nothing\n // is inherited.\n if (typeof HTMLTemplateElement === 'function') {\n var template = document.createElement('template');\n if (template.content && template.content.ownerDocument) {\n document = template.content.ownerDocument;\n }\n }\n var trustedTypesPolicy = _createTrustedTypesPolicy(trustedTypes, originalDocument);\n var emptyHTML = trustedTypesPolicy ? trustedTypesPolicy.createHTML('') : '';\n var _document = document,\n implementation = _document.implementation,\n createNodeIterator = _document.createNodeIterator,\n createDocumentFragment = _document.createDocumentFragment,\n getElementsByTagName = _document.getElementsByTagName;\n var importNode = originalDocument.importNode;\n var documentMode = {};\n try {\n documentMode = clone(document).documentMode ? document.documentMode : {};\n } catch (_) {}\n var hooks = {};\n\n /**\n * Expose whether this browser supports running the full DOMPurify.\n */\n DOMPurify.isSupported = typeof getParentNode === 'function' && implementation && implementation.createHTMLDocument !== undefined && documentMode !== 9;\n var MUSTACHE_EXPR$1 = MUSTACHE_EXPR,\n ERB_EXPR$1 = ERB_EXPR,\n TMPLIT_EXPR$1 = TMPLIT_EXPR,\n DATA_ATTR$1 = DATA_ATTR,\n ARIA_ATTR$1 = ARIA_ATTR,\n IS_SCRIPT_OR_DATA$1 = IS_SCRIPT_OR_DATA,\n ATTR_WHITESPACE$1 = ATTR_WHITESPACE,\n CUSTOM_ELEMENT$1 = CUSTOM_ELEMENT;\n var IS_ALLOWED_URI$1 = IS_ALLOWED_URI;\n\n /**\n * We consider the elements and attributes below to be safe. Ideally\n * don't add any new ones but feel free to remove unwanted ones.\n */\n\n /* allowed element names */\n var ALLOWED_TAGS = null;\n var DEFAULT_ALLOWED_TAGS = addToSet({}, [].concat(_toConsumableArray(html$1), _toConsumableArray(svg$1), _toConsumableArray(svgFilters), _toConsumableArray(mathMl$1), _toConsumableArray(text)));\n\n /* Allowed attribute names */\n var ALLOWED_ATTR = null;\n var DEFAULT_ALLOWED_ATTR = addToSet({}, [].concat(_toConsumableArray(html), _toConsumableArray(svg), _toConsumableArray(mathMl), _toConsumableArray(xml)));\n\n /*\n * Configure how DOMPUrify should handle custom elements and their attributes as well as customized built-in elements.\n * @property {RegExp|Function|null} tagNameCheck one of [null, regexPattern, predicate]. Default: `null` (disallow any custom elements)\n * @property {RegExp|Function|null} attributeNameCheck one of [null, regexPattern, predicate]. Default: `null` (disallow any attributes not on the allow list)\n * @property {boolean} allowCustomizedBuiltInElements allow custom elements derived from built-ins if they pass CUSTOM_ELEMENT_HANDLING.tagNameCheck. Default: `false`.\n */\n var CUSTOM_ELEMENT_HANDLING = Object.seal(Object.create(null, {\n tagNameCheck: {\n writable: true,\n configurable: false,\n enumerable: true,\n value: null\n },\n attributeNameCheck: {\n writable: true,\n configurable: false,\n enumerable: true,\n value: null\n },\n allowCustomizedBuiltInElements: {\n writable: true,\n configurable: false,\n enumerable: true,\n value: false\n }\n }));\n\n /* Explicitly forbidden tags (overrides ALLOWED_TAGS/ADD_TAGS) */\n var FORBID_TAGS = null;\n\n /* Explicitly forbidden attributes (overrides ALLOWED_ATTR/ADD_ATTR) */\n var FORBID_ATTR = null;\n\n /* Decide if ARIA attributes are okay */\n var ALLOW_ARIA_ATTR = true;\n\n /* Decide if custom data attributes are okay */\n var ALLOW_DATA_ATTR = true;\n\n /* Decide if unknown protocols are okay */\n var ALLOW_UNKNOWN_PROTOCOLS = false;\n\n /* Decide if self-closing tags in attributes are allowed.\n * Usually removed due to a mXSS issue in jQuery 3.0 */\n var ALLOW_SELF_CLOSE_IN_ATTR = true;\n\n /* Output should be safe for common template engines.\n * This means, DOMPurify removes data attributes, mustaches and ERB\n */\n var SAFE_FOR_TEMPLATES = false;\n\n /* Output should be safe even for XML used within HTML and alike.\n * This means, DOMPurify removes comments when containing risky content.\n */\n var SAFE_FOR_XML = true;\n\n /* Decide if document with ... should be returned */\n var WHOLE_DOCUMENT = false;\n\n /* Track whether config is already set on this instance of DOMPurify. */\n var SET_CONFIG = false;\n\n /* Decide if all elements (e.g. style, script) must be children of\n * document.body. By default, browsers might move them to document.head */\n var FORCE_BODY = false;\n\n /* Decide if a DOM `HTMLBodyElement` should be returned, instead of a html\n * string (or a TrustedHTML object if Trusted Types are supported).\n * If `WHOLE_DOCUMENT` is enabled a `HTMLHtmlElement` will be returned instead\n */\n var RETURN_DOM = false;\n\n /* Decide if a DOM `DocumentFragment` should be returned, instead of a html\n * string (or a TrustedHTML object if Trusted Types are supported) */\n var RETURN_DOM_FRAGMENT = false;\n\n /* Try to return a Trusted Type object instead of a string, return a string in\n * case Trusted Types are not supported */\n var RETURN_TRUSTED_TYPE = false;\n\n /* Output should be free from DOM clobbering attacks?\n * This sanitizes markups named with colliding, clobberable built-in DOM APIs.\n */\n var SANITIZE_DOM = true;\n\n /* Achieve full DOM Clobbering protection by isolating the namespace of named\n * properties and JS variables, mitigating attacks that abuse the HTML/DOM spec rules.\n *\n * HTML/DOM spec rules that enable DOM Clobbering:\n * - Named Access on Window (§7.3.3)\n * - DOM Tree Accessors (§3.1.5)\n * - Form Element Parent-Child Relations (§4.10.3)\n * - Iframe srcdoc / Nested WindowProxies (§4.8.5)\n * - HTMLCollection (§4.2.10.2)\n *\n * Namespace isolation is implemented by prefixing `id` and `name` attributes\n * with a constant string, i.e., `user-content-`\n */\n var SANITIZE_NAMED_PROPS = false;\n var SANITIZE_NAMED_PROPS_PREFIX = 'user-content-';\n\n /* Keep element content when removing element? */\n var KEEP_CONTENT = true;\n\n /* If a `Node` is passed to sanitize(), then performs sanitization in-place instead\n * of importing it into a new Document and returning a sanitized copy */\n var IN_PLACE = false;\n\n /* Allow usage of profiles like html, svg and mathMl */\n var USE_PROFILES = {};\n\n /* Tags to ignore content of when KEEP_CONTENT is true */\n var FORBID_CONTENTS = null;\n var DEFAULT_FORBID_CONTENTS = addToSet({}, ['annotation-xml', 'audio', 'colgroup', 'desc', 'foreignobject', 'head', 'iframe', 'math', 'mi', 'mn', 'mo', 'ms', 'mtext', 'noembed', 'noframes', 'noscript', 'plaintext', 'script', 'style', 'svg', 'template', 'thead', 'title', 'video', 'xmp']);\n\n /* Tags that are safe for data: URIs */\n var DATA_URI_TAGS = null;\n var DEFAULT_DATA_URI_TAGS = addToSet({}, ['audio', 'video', 'img', 'source', 'image', 'track']);\n\n /* Attributes safe for values like \"javascript:\" */\n var URI_SAFE_ATTRIBUTES = null;\n var DEFAULT_URI_SAFE_ATTRIBUTES = addToSet({}, ['alt', 'class', 'for', 'id', 'label', 'name', 'pattern', 'placeholder', 'role', 'summary', 'title', 'value', 'style', 'xmlns']);\n var MATHML_NAMESPACE = 'http://www.w3.org/1998/Math/MathML';\n var SVG_NAMESPACE = 'http://www.w3.org/2000/svg';\n var HTML_NAMESPACE = 'http://www.w3.org/1999/xhtml';\n /* Document namespace */\n var NAMESPACE = HTML_NAMESPACE;\n var IS_EMPTY_INPUT = false;\n\n /* Allowed XHTML+XML namespaces */\n var ALLOWED_NAMESPACES = null;\n var DEFAULT_ALLOWED_NAMESPACES = addToSet({}, [MATHML_NAMESPACE, SVG_NAMESPACE, HTML_NAMESPACE], stringToString);\n\n /* Parsing of strict XHTML documents */\n var PARSER_MEDIA_TYPE;\n var SUPPORTED_PARSER_MEDIA_TYPES = ['application/xhtml+xml', 'text/html'];\n var DEFAULT_PARSER_MEDIA_TYPE = 'text/html';\n var transformCaseFunc;\n\n /* Keep a reference to config to pass to hooks */\n var CONFIG = null;\n\n /* Ideally, do not touch anything below this line */\n /* ______________________________________________ */\n\n var formElement = document.createElement('form');\n var isRegexOrFunction = function isRegexOrFunction(testValue) {\n return testValue instanceof RegExp || testValue instanceof Function;\n };\n\n /**\n * _parseConfig\n *\n * @param {Object} cfg optional config literal\n */\n // eslint-disable-next-line complexity\n var _parseConfig = function _parseConfig(cfg) {\n if (CONFIG && CONFIG === cfg) {\n return;\n }\n\n /* Shield configuration object from tampering */\n if (!cfg || _typeof(cfg) !== 'object') {\n cfg = {};\n }\n\n /* Shield configuration object from prototype pollution */\n cfg = clone(cfg);\n PARSER_MEDIA_TYPE =\n // eslint-disable-next-line unicorn/prefer-includes\n SUPPORTED_PARSER_MEDIA_TYPES.indexOf(cfg.PARSER_MEDIA_TYPE) === -1 ? PARSER_MEDIA_TYPE = DEFAULT_PARSER_MEDIA_TYPE : PARSER_MEDIA_TYPE = cfg.PARSER_MEDIA_TYPE;\n\n // HTML tags and attributes are not case-sensitive, converting to lowercase. Keeping XHTML as is.\n transformCaseFunc = PARSER_MEDIA_TYPE === 'application/xhtml+xml' ? stringToString : stringToLowerCase;\n\n /* Set configuration parameters */\n ALLOWED_TAGS = 'ALLOWED_TAGS' in cfg ? addToSet({}, cfg.ALLOWED_TAGS, transformCaseFunc) : DEFAULT_ALLOWED_TAGS;\n ALLOWED_ATTR = 'ALLOWED_ATTR' in cfg ? addToSet({}, cfg.ALLOWED_ATTR, transformCaseFunc) : DEFAULT_ALLOWED_ATTR;\n ALLOWED_NAMESPACES = 'ALLOWED_NAMESPACES' in cfg ? addToSet({}, cfg.ALLOWED_NAMESPACES, stringToString) : DEFAULT_ALLOWED_NAMESPACES;\n URI_SAFE_ATTRIBUTES = 'ADD_URI_SAFE_ATTR' in cfg ? addToSet(clone(DEFAULT_URI_SAFE_ATTRIBUTES),\n // eslint-disable-line indent\n cfg.ADD_URI_SAFE_ATTR,\n // eslint-disable-line indent\n transformCaseFunc // eslint-disable-line indent\n ) // eslint-disable-line indent\n : DEFAULT_URI_SAFE_ATTRIBUTES;\n DATA_URI_TAGS = 'ADD_DATA_URI_TAGS' in cfg ? addToSet(clone(DEFAULT_DATA_URI_TAGS),\n // eslint-disable-line indent\n cfg.ADD_DATA_URI_TAGS,\n // eslint-disable-line indent\n transformCaseFunc // eslint-disable-line indent\n ) // eslint-disable-line indent\n : DEFAULT_DATA_URI_TAGS;\n FORBID_CONTENTS = 'FORBID_CONTENTS' in cfg ? addToSet({}, cfg.FORBID_CONTENTS, transformCaseFunc) : DEFAULT_FORBID_CONTENTS;\n FORBID_TAGS = 'FORBID_TAGS' in cfg ? addToSet({}, cfg.FORBID_TAGS, transformCaseFunc) : {};\n FORBID_ATTR = 'FORBID_ATTR' in cfg ? addToSet({}, cfg.FORBID_ATTR, transformCaseFunc) : {};\n USE_PROFILES = 'USE_PROFILES' in cfg ? cfg.USE_PROFILES : false;\n ALLOW_ARIA_ATTR = cfg.ALLOW_ARIA_ATTR !== false; // Default true\n ALLOW_DATA_ATTR = cfg.ALLOW_DATA_ATTR !== false; // Default true\n ALLOW_UNKNOWN_PROTOCOLS = cfg.ALLOW_UNKNOWN_PROTOCOLS || false; // Default false\n ALLOW_SELF_CLOSE_IN_ATTR = cfg.ALLOW_SELF_CLOSE_IN_ATTR !== false; // Default true\n SAFE_FOR_TEMPLATES = cfg.SAFE_FOR_TEMPLATES || false; // Default false\n SAFE_FOR_XML = cfg.SAFE_FOR_XML !== false; // Default true\n WHOLE_DOCUMENT = cfg.WHOLE_DOCUMENT || false; // Default false\n RETURN_DOM = cfg.RETURN_DOM || false; // Default false\n RETURN_DOM_FRAGMENT = cfg.RETURN_DOM_FRAGMENT || false; // Default false\n RETURN_TRUSTED_TYPE = cfg.RETURN_TRUSTED_TYPE || false; // Default false\n FORCE_BODY = cfg.FORCE_BODY || false; // Default false\n SANITIZE_DOM = cfg.SANITIZE_DOM !== false; // Default true\n SANITIZE_NAMED_PROPS = cfg.SANITIZE_NAMED_PROPS || false; // Default false\n KEEP_CONTENT = cfg.KEEP_CONTENT !== false; // Default true\n IN_PLACE = cfg.IN_PLACE || false; // Default false\n IS_ALLOWED_URI$1 = cfg.ALLOWED_URI_REGEXP || IS_ALLOWED_URI$1;\n NAMESPACE = cfg.NAMESPACE || HTML_NAMESPACE;\n CUSTOM_ELEMENT_HANDLING = cfg.CUSTOM_ELEMENT_HANDLING || {};\n if (cfg.CUSTOM_ELEMENT_HANDLING && isRegexOrFunction(cfg.CUSTOM_ELEMENT_HANDLING.tagNameCheck)) {\n CUSTOM_ELEMENT_HANDLING.tagNameCheck = cfg.CUSTOM_ELEMENT_HANDLING.tagNameCheck;\n }\n if (cfg.CUSTOM_ELEMENT_HANDLING && isRegexOrFunction(cfg.CUSTOM_ELEMENT_HANDLING.attributeNameCheck)) {\n CUSTOM_ELEMENT_HANDLING.attributeNameCheck = cfg.CUSTOM_ELEMENT_HANDLING.attributeNameCheck;\n }\n if (cfg.CUSTOM_ELEMENT_HANDLING && typeof cfg.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements === 'boolean') {\n CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements = cfg.CUSTOM_ELEMENT_HANDLING.allowCustomizedBuiltInElements;\n }\n if (SAFE_FOR_TEMPLATES) {\n ALLOW_DATA_ATTR = false;\n }\n if (RETURN_DOM_FRAGMENT) {\n RETURN_DOM = true;\n }\n\n /* Parse profile info */\n if (USE_PROFILES) {\n ALLOWED_TAGS = addToSet({}, _toConsumableArray(text));\n ALLOWED_ATTR = [];\n if (USE_PROFILES.html === true) {\n addToSet(ALLOWED_TAGS, html$1);\n addToSet(ALLOWED_ATTR, html);\n }\n if (USE_PROFILES.svg === true) {\n addToSet(ALLOWED_TAGS, svg$1);\n addToSet(ALLOWED_ATTR, svg);\n addToSet(ALLOWED_ATTR, xml);\n }\n if (USE_PROFILES.svgFilters === true) {\n addToSet(ALLOWED_TAGS, svgFilters);\n addToSet(ALLOWED_ATTR, svg);\n addToSet(ALLOWED_ATTR, xml);\n }\n if (USE_PROFILES.mathMl === true) {\n addToSet(ALLOWED_TAGS, mathMl$1);\n addToSet(ALLOWED_ATTR, mathMl);\n addToSet(ALLOWED_ATTR, xml);\n }\n }\n\n /* Merge configuration parameters */\n if (cfg.ADD_TAGS) {\n if (ALLOWED_TAGS === DEFAULT_ALLOWED_TAGS) {\n ALLOWED_TAGS = clone(ALLOWED_TAGS);\n }\n addToSet(ALLOWED_TAGS, cfg.ADD_TAGS, transformCaseFunc);\n }\n if (cfg.ADD_ATTR) {\n if (ALLOWED_ATTR === DEFAULT_ALLOWED_ATTR) {\n ALLOWED_ATTR = clone(ALLOWED_ATTR);\n }\n addToSet(ALLOWED_ATTR, cfg.ADD_ATTR, transformCaseFunc);\n }\n if (cfg.ADD_URI_SAFE_ATTR) {\n addToSet(URI_SAFE_ATTRIBUTES, cfg.ADD_URI_SAFE_ATTR, transformCaseFunc);\n }\n if (cfg.FORBID_CONTENTS) {\n if (FORBID_CONTENTS === DEFAULT_FORBID_CONTENTS) {\n FORBID_CONTENTS = clone(FORBID_CONTENTS);\n }\n addToSet(FORBID_CONTENTS, cfg.FORBID_CONTENTS, transformCaseFunc);\n }\n\n /* Add #text in case KEEP_CONTENT is set to true */\n if (KEEP_CONTENT) {\n ALLOWED_TAGS['#text'] = true;\n }\n\n /* Add html, head and body to ALLOWED_TAGS in case WHOLE_DOCUMENT is true */\n if (WHOLE_DOCUMENT) {\n addToSet(ALLOWED_TAGS, ['html', 'head', 'body']);\n }\n\n /* Add tbody to ALLOWED_TAGS in case tables are permitted, see #286, #365 */\n if (ALLOWED_TAGS.table) {\n addToSet(ALLOWED_TAGS, ['tbody']);\n delete FORBID_TAGS.tbody;\n }\n\n // Prevent further manipulation of configuration.\n // Not available in IE8, Safari 5, etc.\n if (freeze) {\n freeze(cfg);\n }\n CONFIG = cfg;\n };\n var MATHML_TEXT_INTEGRATION_POINTS = addToSet({}, ['mi', 'mo', 'mn', 'ms', 'mtext']);\n var HTML_INTEGRATION_POINTS = addToSet({}, ['annotation-xml']);\n\n // Certain elements are allowed in both SVG and HTML\n // namespace. We need to specify them explicitly\n // so that they don't get erroneously deleted from\n // HTML namespace.\n var COMMON_SVG_AND_HTML_ELEMENTS = addToSet({}, ['title', 'style', 'font', 'a', 'script']);\n\n /* Keep track of all possible SVG and MathML tags\n * so that we can perform the namespace checks\n * correctly. */\n var ALL_SVG_TAGS = addToSet({}, svg$1);\n addToSet(ALL_SVG_TAGS, svgFilters);\n addToSet(ALL_SVG_TAGS, svgDisallowed);\n var ALL_MATHML_TAGS = addToSet({}, mathMl$1);\n addToSet(ALL_MATHML_TAGS, mathMlDisallowed);\n\n /**\n *\n *\n * @param {Element} element a DOM element whose namespace is being checked\n * @returns {boolean} Return false if the element has a\n * namespace that a spec-compliant parser would never\n * return. Return true otherwise.\n */\n var _checkValidNamespace = function _checkValidNamespace(element) {\n var parent = getParentNode(element);\n\n // In JSDOM, if we're inside shadow DOM, then parentNode\n // can be null. We just simulate parent in this case.\n if (!parent || !parent.tagName) {\n parent = {\n namespaceURI: NAMESPACE,\n tagName: 'template'\n };\n }\n var tagName = stringToLowerCase(element.tagName);\n var parentTagName = stringToLowerCase(parent.tagName);\n if (!ALLOWED_NAMESPACES[element.namespaceURI]) {\n return false;\n }\n if (element.namespaceURI === SVG_NAMESPACE) {\n // The only way to switch from HTML namespace to SVG\n // is via . If it happens via any other tag, then\n // it should be killed.\n if (parent.namespaceURI === HTML_NAMESPACE) {\n return tagName === 'svg';\n }\n\n // The only way to switch from MathML to SVG is via`\n // svg if parent is either or MathML\n // text integration points.\n if (parent.namespaceURI === MATHML_NAMESPACE) {\n return tagName === 'svg' && (parentTagName === 'annotation-xml' || MATHML_TEXT_INTEGRATION_POINTS[parentTagName]);\n }\n\n // We only allow elements that are defined in SVG\n // spec. All others are disallowed in SVG namespace.\n return Boolean(ALL_SVG_TAGS[tagName]);\n }\n if (element.namespaceURI === MATHML_NAMESPACE) {\n // The only way to switch from HTML namespace to MathML\n // is via . If it happens via any other tag, then\n // it should be killed.\n if (parent.namespaceURI === HTML_NAMESPACE) {\n return tagName === 'math';\n }\n\n // The only way to switch from SVG to MathML is via\n // and HTML integration points\n if (parent.namespaceURI === SVG_NAMESPACE) {\n return tagName === 'math' && HTML_INTEGRATION_POINTS[parentTagName];\n }\n\n // We only allow elements that are defined in MathML\n // spec. All others are disallowed in MathML namespace.\n return Boolean(ALL_MATHML_TAGS[tagName]);\n }\n if (element.namespaceURI === HTML_NAMESPACE) {\n // The only way to switch from SVG to HTML is via\n // HTML integration points, and from MathML to HTML\n // is via MathML text integration points\n if (parent.namespaceURI === SVG_NAMESPACE && !HTML_INTEGRATION_POINTS[parentTagName]) {\n return false;\n }\n if (parent.namespaceURI === MATHML_NAMESPACE && !MATHML_TEXT_INTEGRATION_POINTS[parentTagName]) {\n return false;\n }\n\n // We disallow tags that are specific for MathML\n // or SVG and should never appear in HTML namespace\n return !ALL_MATHML_TAGS[tagName] && (COMMON_SVG_AND_HTML_ELEMENTS[tagName] || !ALL_SVG_TAGS[tagName]);\n }\n\n // For XHTML and XML documents that support custom namespaces\n if (PARSER_MEDIA_TYPE === 'application/xhtml+xml' && ALLOWED_NAMESPACES[element.namespaceURI]) {\n return true;\n }\n\n // The code should never reach this place (this means\n // that the element somehow got namespace that is not\n // HTML, SVG, MathML or allowed via ALLOWED_NAMESPACES).\n // Return false just in case.\n return false;\n };\n\n /**\n * _forceRemove\n *\n * @param {Node} node a DOM node\n */\n var _forceRemove = function _forceRemove(node) {\n arrayPush(DOMPurify.removed, {\n element: node\n });\n try {\n // eslint-disable-next-line unicorn/prefer-dom-node-remove\n node.parentNode.removeChild(node);\n } catch (_) {\n try {\n node.outerHTML = emptyHTML;\n } catch (_) {\n node.remove();\n }\n }\n };\n\n /**\n * _removeAttribute\n *\n * @param {String} name an Attribute name\n * @param {Node} node a DOM node\n */\n var _removeAttribute = function _removeAttribute(name, node) {\n try {\n arrayPush(DOMPurify.removed, {\n attribute: node.getAttributeNode(name),\n from: node\n });\n } catch (_) {\n arrayPush(DOMPurify.removed, {\n attribute: null,\n from: node\n });\n }\n node.removeAttribute(name);\n\n // We void attribute values for unremovable \"is\"\" attributes\n if (name === 'is' && !ALLOWED_ATTR[name]) {\n if (RETURN_DOM || RETURN_DOM_FRAGMENT) {\n try {\n _forceRemove(node);\n } catch (_) {}\n } else {\n try {\n node.setAttribute(name, '');\n } catch (_) {}\n }\n }\n };\n\n /**\n * _initDocument\n *\n * @param {String} dirty a string of dirty markup\n * @return {Document} a DOM, filled with the dirty markup\n */\n var _initDocument = function _initDocument(dirty) {\n /* Create a HTML document */\n var doc;\n var leadingWhitespace;\n if (FORCE_BODY) {\n dirty = '' + dirty;\n } else {\n /* If FORCE_BODY isn't used, leading whitespace needs to be preserved manually */\n var matches = stringMatch(dirty, /^[\\r\\n\\t ]+/);\n leadingWhitespace = matches && matches[0];\n }\n if (PARSER_MEDIA_TYPE === 'application/xhtml+xml' && NAMESPACE === HTML_NAMESPACE) {\n // Root of XHTML doc must contain xmlns declaration (see https://www.w3.org/TR/xhtml1/normative.html#strict)\n dirty = '' + dirty + '';\n }\n var dirtyPayload = trustedTypesPolicy ? trustedTypesPolicy.createHTML(dirty) : dirty;\n /*\n * Use the DOMParser API by default, fallback later if needs be\n * DOMParser not work for svg when has multiple root element.\n */\n if (NAMESPACE === HTML_NAMESPACE) {\n try {\n doc = new DOMParser().parseFromString(dirtyPayload, PARSER_MEDIA_TYPE);\n } catch (_) {}\n }\n\n /* Use createHTMLDocument in case DOMParser is not available */\n if (!doc || !doc.documentElement) {\n doc = implementation.createDocument(NAMESPACE, 'template', null);\n try {\n doc.documentElement.innerHTML = IS_EMPTY_INPUT ? emptyHTML : dirtyPayload;\n } catch (_) {\n // Syntax error if dirtyPayload is invalid xml\n }\n }\n var body = doc.body || doc.documentElement;\n if (dirty && leadingWhitespace) {\n body.insertBefore(document.createTextNode(leadingWhitespace), body.childNodes[0] || null);\n }\n\n /* Work on whole document or just its body */\n if (NAMESPACE === HTML_NAMESPACE) {\n return getElementsByTagName.call(doc, WHOLE_DOCUMENT ? 'html' : 'body')[0];\n }\n return WHOLE_DOCUMENT ? doc.documentElement : body;\n };\n\n /**\n * _createIterator\n *\n * @param {Document} root document/fragment to create iterator for\n * @return {Iterator} iterator instance\n */\n var _createIterator = function _createIterator(root) {\n return createNodeIterator.call(root.ownerDocument || root, root,\n // eslint-disable-next-line no-bitwise\n NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT | NodeFilter.SHOW_TEXT | NodeFilter.SHOW_PROCESSING_INSTRUCTION | NodeFilter.SHOW_CDATA_SECTION, null, false);\n };\n\n /**\n * _isClobbered\n *\n * @param {Node} elm element to check for clobbering attacks\n * @return {Boolean} true if clobbered, false if safe\n */\n var _isClobbered = function _isClobbered(elm) {\n return elm instanceof HTMLFormElement && (typeof elm.nodeName !== 'string' || typeof elm.textContent !== 'string' || typeof elm.removeChild !== 'function' || !(elm.attributes instanceof NamedNodeMap) || typeof elm.removeAttribute !== 'function' || typeof elm.setAttribute !== 'function' || typeof elm.namespaceURI !== 'string' || typeof elm.insertBefore !== 'function' || typeof elm.hasChildNodes !== 'function');\n };\n\n /**\n * _isNode\n *\n * @param {Node} obj object to check whether it's a DOM node\n * @return {Boolean} true is object is a DOM node\n */\n var _isNode = function _isNode(object) {\n return _typeof(Node) === 'object' ? object instanceof Node : object && _typeof(object) === 'object' && typeof object.nodeType === 'number' && typeof object.nodeName === 'string';\n };\n\n /**\n * _executeHook\n * Execute user configurable hooks\n *\n * @param {String} entryPoint Name of the hook's entry point\n * @param {Node} currentNode node to work on with the hook\n * @param {Object} data additional hook parameters\n */\n var _executeHook = function _executeHook(entryPoint, currentNode, data) {\n if (!hooks[entryPoint]) {\n return;\n }\n arrayForEach(hooks[entryPoint], function (hook) {\n hook.call(DOMPurify, currentNode, data, CONFIG);\n });\n };\n\n /**\n * _sanitizeElements\n *\n * @protect nodeName\n * @protect textContent\n * @protect removeChild\n *\n * @param {Node} currentNode to check for permission to exist\n * @return {Boolean} true if node was killed, false if left alive\n */\n var _sanitizeElements = function _sanitizeElements(currentNode) {\n var content;\n\n /* Execute a hook if present */\n _executeHook('beforeSanitizeElements', currentNode, null);\n\n /* Check if element is clobbered or can clobber */\n if (_isClobbered(currentNode)) {\n _forceRemove(currentNode);\n return true;\n }\n\n /* Check if tagname contains Unicode */\n if (regExpTest(/[\\u0080-\\uFFFF]/, currentNode.nodeName)) {\n _forceRemove(currentNode);\n return true;\n }\n\n /* Now let's check the element's type and name */\n var tagName = transformCaseFunc(currentNode.nodeName);\n\n /* Execute a hook if present */\n _executeHook('uponSanitizeElement', currentNode, {\n tagName: tagName,\n allowedTags: ALLOWED_TAGS\n });\n\n /* Detect mXSS attempts abusing namespace confusion */\n if (currentNode.hasChildNodes() && !_isNode(currentNode.firstElementChild) && (!_isNode(currentNode.content) || !_isNode(currentNode.content.firstElementChild)) && regExpTest(/<[/\\w]/g, currentNode.innerHTML) && regExpTest(/<[/\\w]/g, currentNode.textContent)) {\n _forceRemove(currentNode);\n return true;\n }\n\n /* Mitigate a problem with templates inside select */\n if (tagName === 'select' && regExpTest(/