valtech-components 2.0.404 → 2.0.407
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +71 -0
- package/esm2022/lib/components/atoms/avatar/avatar.component.mjs +4 -4
- package/esm2022/lib/components/atoms/avatar/types.mjs +1 -1
- package/esm2022/lib/components/atoms/box/box.component.mjs +4 -4
- package/esm2022/lib/components/atoms/box/types.mjs +1 -1
- package/esm2022/lib/components/atoms/button/button.component.mjs +6 -6
- package/esm2022/lib/components/atoms/button/factory.mjs +1 -1
- package/esm2022/lib/components/atoms/countdown/countdown.component.mjs +343 -0
- package/esm2022/lib/components/atoms/countdown/types.mjs +27 -0
- package/esm2022/lib/components/atoms/display/display.component.mjs +4 -4
- package/esm2022/lib/components/atoms/display/types.mjs +1 -1
- package/esm2022/lib/components/atoms/divider/divider.component.mjs +4 -4
- package/esm2022/lib/components/atoms/divider/types.mjs +1 -1
- package/esm2022/lib/components/atoms/fab/fab.component.mjs +152 -0
- package/esm2022/lib/components/atoms/fab/types.mjs +2 -0
- package/esm2022/lib/components/atoms/href/href.component.mjs +4 -4
- package/esm2022/lib/components/atoms/href/types.mjs +1 -1
- package/esm2022/lib/components/atoms/icon/icon.component.mjs +4 -4
- package/esm2022/lib/components/atoms/icon/types.mjs +1 -1
- package/esm2022/lib/components/atoms/image/image.component.mjs +6 -6
- package/esm2022/lib/components/atoms/image/types.mjs +1 -1
- package/esm2022/lib/components/atoms/price-tag/price-tag.component.mjs +245 -0
- package/esm2022/lib/components/atoms/price-tag/types.mjs +15 -0
- package/esm2022/lib/components/atoms/progress-bar/progress-bar.component.mjs +4 -4
- package/esm2022/lib/components/atoms/progress-bar/types.mjs +1 -1
- package/esm2022/lib/components/atoms/progress-ring/progress-ring.component.mjs +149 -0
- package/esm2022/lib/components/atoms/progress-ring/types.mjs +2 -0
- package/esm2022/lib/components/atoms/qr-code/qr-code.component.mjs +417 -0
- package/esm2022/lib/components/atoms/qr-code/types.mjs +2 -0
- package/esm2022/lib/components/atoms/skeleton/skeleton.component.mjs +193 -0
- package/esm2022/lib/components/atoms/skeleton/types.mjs +13 -0
- package/esm2022/lib/components/atoms/text/text.component.mjs +4 -4
- package/esm2022/lib/components/atoms/text/types.mjs +1 -1
- package/esm2022/lib/components/atoms/title/title.component.mjs +4 -4
- package/esm2022/lib/components/atoms/title/types.mjs +1 -1
- package/esm2022/lib/components/molecules/accordion/accordion.component.mjs +139 -0
- package/esm2022/lib/components/molecules/accordion/types.mjs +2 -0
- package/esm2022/lib/components/molecules/action-header/action-header.component.mjs +4 -4
- package/esm2022/lib/components/molecules/action-header/types.mjs +1 -1
- package/esm2022/lib/components/molecules/alert-box/alert-box.component.mjs +4 -4
- package/esm2022/lib/components/molecules/alert-box/types.mjs +1 -1
- package/esm2022/lib/components/molecules/breadcrumb/breadcrumb.component.mjs +137 -0
- package/esm2022/lib/components/molecules/breadcrumb/types.mjs +2 -0
- package/esm2022/lib/components/molecules/button-group/button-group.component.mjs +4 -4
- package/esm2022/lib/components/molecules/button-group/types.mjs +1 -1
- package/esm2022/lib/components/molecules/card/card.component.mjs +5 -5
- package/esm2022/lib/components/molecules/card/types.mjs +1 -1
- package/esm2022/lib/components/molecules/check-input/check-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/chip-group/chip-group.component.mjs +174 -0
- package/esm2022/lib/components/molecules/chip-group/types.mjs +2 -0
- package/esm2022/lib/components/molecules/code-display/code-display.component.mjs +4 -4
- package/esm2022/lib/components/molecules/code-display/types.mjs +1 -1
- package/esm2022/lib/components/molecules/command-display/command-display.component.mjs +4 -4
- package/esm2022/lib/components/molecules/command-display/types.mjs +1 -1
- package/esm2022/lib/components/molecules/comment/comment.component.mjs +634 -0
- package/esm2022/lib/components/molecules/comment/types.mjs +2 -0
- package/esm2022/lib/components/molecules/comment-input/comment-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/content-loader/content-loader.component.mjs +4 -4
- package/esm2022/lib/components/molecules/content-loader/types.mjs +1 -1
- package/esm2022/lib/components/molecules/currency-input/currency-input.component.mjs +368 -0
- package/esm2022/lib/components/molecules/currency-input/types.mjs +18 -0
- package/esm2022/lib/components/molecules/date-input/date-input.component.mjs +5 -5
- package/esm2022/lib/components/molecules/date-range-input/date-range-input.component.mjs +418 -0
- package/esm2022/lib/components/molecules/date-range-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/email-input/email-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/expandable-text/expandable-text.component.mjs +4 -4
- package/esm2022/lib/components/molecules/expandable-text/types.mjs +1 -1
- package/esm2022/lib/components/molecules/file-input/file-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/glow-card/glow-card.component.mjs +279 -0
- package/esm2022/lib/components/molecules/glow-card/types.mjs +11 -0
- package/esm2022/lib/components/molecules/hint/hint.component.mjs +4 -4
- package/esm2022/lib/components/molecules/hour-input/hour-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/info/info.component.mjs +4 -4
- package/esm2022/lib/components/molecules/info/types.mjs +1 -1
- package/esm2022/lib/components/molecules/language-selector/language-selector.component.mjs +4 -4
- package/esm2022/lib/components/molecules/language-selector/types.mjs +1 -1
- package/esm2022/lib/components/molecules/layered-card/layered-card.component.mjs +4 -4
- package/esm2022/lib/components/molecules/layered-card/types.mjs +1 -1
- package/esm2022/lib/components/molecules/link/link.component.mjs +4 -4
- package/esm2022/lib/components/molecules/link/types.mjs +1 -1
- package/esm2022/lib/components/molecules/links-cake/links-cake.component.mjs +4 -4
- package/esm2022/lib/components/molecules/links-cake/types.mjs +1 -1
- package/esm2022/lib/components/molecules/multi-select-search/multi-select-search.component.mjs +5 -5
- package/esm2022/lib/components/molecules/notes-box/notes-box.component.mjs +4 -4
- package/esm2022/lib/components/molecules/notes-box/types.mjs +1 -1
- package/esm2022/lib/components/molecules/number-from-to/number-from-to.component.mjs +4 -4
- package/esm2022/lib/components/molecules/number-input/number-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/number-stepper/number-stepper.component.mjs +377 -0
- package/esm2022/lib/components/molecules/number-stepper/types.mjs +2 -0
- package/esm2022/lib/components/molecules/pagination/pagination.component.mjs +253 -0
- package/esm2022/lib/components/molecules/pagination/types.mjs +2 -0
- package/esm2022/lib/components/molecules/participant-card/participant-card.component.mjs +486 -0
- package/esm2022/lib/components/molecules/participant-card/types.mjs +21 -0
- package/esm2022/lib/components/molecules/password-input/password-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/phone-input/phone-input.component.mjs +336 -0
- package/esm2022/lib/components/molecules/phone-input/types.mjs +19 -0
- package/esm2022/lib/components/molecules/pill/pill.component.mjs +4 -4
- package/esm2022/lib/components/molecules/pill/types.mjs +1 -1
- package/esm2022/lib/components/molecules/pin-input/pin-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/plain-code-box/plain-code-box.component.mjs +4 -4
- package/esm2022/lib/components/molecules/plain-code-box/types.mjs +1 -1
- package/esm2022/lib/components/molecules/popover-selector/popover-selector.component.mjs +5 -5
- package/esm2022/lib/components/molecules/popover-selector/types.mjs +1 -1
- package/esm2022/lib/components/molecules/progress-status/progress-status.component.mjs +4 -4
- package/esm2022/lib/components/molecules/progress-status/types.mjs +1 -1
- package/esm2022/lib/components/molecules/prompter/prompter.component.mjs +4 -4
- package/esm2022/lib/components/molecules/prompter/types.mjs +1 -1
- package/esm2022/lib/components/molecules/quote-box/quote-box.component.mjs +155 -0
- package/esm2022/lib/components/molecules/radio-input/radio-input.component.mjs +5 -5
- package/esm2022/lib/components/molecules/raffle-status-card/raffle-status-card.component.mjs +484 -0
- package/esm2022/lib/components/molecules/raffle-status-card/types.mjs +23 -0
- package/esm2022/lib/components/molecules/range-input/range-input.component.mjs +148 -0
- package/esm2022/lib/components/molecules/range-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/rating/rating.component.mjs +149 -0
- package/esm2022/lib/components/molecules/rating/types.mjs +2 -0
- package/esm2022/lib/components/molecules/searchbar/searchbar.component.mjs +4 -4
- package/esm2022/lib/components/molecules/segment-control/segment-control.component.mjs +145 -0
- package/esm2022/lib/components/molecules/segment-control/types.mjs +2 -0
- package/esm2022/lib/components/molecules/select-input/select-input.component.mjs +5 -5
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +5 -5
- package/esm2022/lib/components/molecules/share-buttons/share-buttons.component.mjs +277 -0
- package/esm2022/lib/components/molecules/share-buttons/types.mjs +88 -0
- package/esm2022/lib/components/molecules/stats-card/stats-card.component.mjs +165 -0
- package/esm2022/lib/components/molecules/stats-card/types.mjs +2 -0
- package/esm2022/lib/components/molecules/stepper/stepper.component.mjs +239 -0
- package/esm2022/lib/components/molecules/stepper/types.mjs +2 -0
- package/esm2022/lib/components/molecules/tabs/tabs.component.mjs +135 -0
- package/esm2022/lib/components/molecules/tabs/types.mjs +2 -0
- package/esm2022/lib/components/molecules/text-input/text-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/textarea-input/textarea-input.component.mjs +204 -0
- package/esm2022/lib/components/molecules/textarea-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/ticket-grid/ticket-grid.component.mjs +497 -0
- package/esm2022/lib/components/molecules/ticket-grid/types.mjs +11 -0
- package/esm2022/lib/components/molecules/timeline/timeline.component.mjs +140 -0
- package/esm2022/lib/components/molecules/timeline/types.mjs +2 -0
- package/esm2022/lib/components/molecules/title-block/title-block.component.mjs +4 -4
- package/esm2022/lib/components/molecules/title-block/types.mjs +1 -1
- package/esm2022/lib/components/molecules/toggle-input/toggle-input.component.mjs +89 -0
- package/esm2022/lib/components/molecules/toggle-input/types.mjs +2 -0
- package/esm2022/lib/components/molecules/winner-display/types.mjs +9 -0
- package/esm2022/lib/components/molecules/winner-display/winner-display.component.mjs +370 -0
- package/esm2022/lib/components/organisms/article/article.component.mjs +4 -4
- package/esm2022/lib/components/organisms/article/types.mjs +1 -1
- package/esm2022/lib/components/organisms/banner/banner.component.mjs +4 -4
- package/esm2022/lib/components/organisms/banner/types.mjs +1 -1
- package/esm2022/lib/components/organisms/comment-section/comment-section.component.mjs +578 -0
- package/esm2022/lib/components/organisms/comment-section/types.mjs +2 -0
- package/esm2022/lib/components/organisms/data-table/data-table.component.mjs +853 -0
- package/esm2022/lib/components/organisms/data-table/types.mjs +13 -0
- package/esm2022/lib/components/organisms/footer/footer.component.mjs +4 -4
- package/esm2022/lib/components/organisms/footer/types.mjs +1 -1
- package/esm2022/lib/components/organisms/form/factory.mjs +1 -1
- package/esm2022/lib/components/organisms/form/form-footer/form-footer.component.mjs +4 -4
- package/esm2022/lib/components/organisms/form/form.component.mjs +4 -4
- package/esm2022/lib/components/organisms/header/header.component.mjs +4 -4
- package/esm2022/lib/components/organisms/header/types.mjs +1 -1
- package/esm2022/lib/components/organisms/item-list/item-list.component.mjs +4 -4
- package/esm2022/lib/components/organisms/item-list/types.mjs +1 -1
- package/esm2022/lib/components/organisms/no-content/no-content.component.mjs +4 -4
- package/esm2022/lib/components/organisms/no-content/types.mjs +1 -1
- package/esm2022/lib/components/organisms/toolbar/toolbar.component.mjs +4 -4
- package/esm2022/lib/components/organisms/toolbar/types.mjs +1 -1
- package/esm2022/lib/components/organisms/wizard/types.mjs +1 -1
- package/esm2022/lib/components/organisms/wizard/wizard-footer/wizard-footer.component.mjs +4 -4
- package/esm2022/lib/components/organisms/wizard/wizard.component.mjs +4 -4
- package/esm2022/lib/components/templates/layout/layout.component.mjs +4 -4
- package/esm2022/lib/components/templates/simple/simple.component.mjs +5 -5
- package/esm2022/lib/components/templates/simple/types.mjs +1 -1
- package/esm2022/lib/components/types.mjs +22 -16
- package/esm2022/lib/services/confirmation-dialog/confirmation-dialog.service.mjs +180 -0
- package/esm2022/lib/services/confirmation-dialog/types.mjs +14 -0
- package/esm2022/lib/services/download.service.mjs +4 -4
- package/esm2022/lib/services/icons.service.mjs +4 -4
- package/esm2022/lib/services/in-app-browser.service.mjs +4 -4
- package/esm2022/lib/services/lang-provider/components/lang-settings.mjs +1 -1
- package/esm2022/lib/services/lang-provider/content.mjs +1 -1
- package/esm2022/lib/services/lang-provider/lang-provider.service.mjs +4 -4
- package/esm2022/lib/services/lang-provider/types.mjs +1 -1
- package/esm2022/lib/services/link-processor.service.mjs +4 -4
- package/esm2022/lib/services/local-storage.service.mjs +1 -1
- package/esm2022/lib/services/modal/modal.service.mjs +213 -0
- package/esm2022/lib/services/modal/simple-modal-content.component.mjs +133 -0
- package/esm2022/lib/services/modal/types.mjs +26 -0
- package/esm2022/lib/services/navigation.service.mjs +4 -4
- package/esm2022/lib/services/qr-generator/qr-generator.service.mjs +341 -0
- package/esm2022/lib/services/qr-generator/types.mjs +46 -0
- package/esm2022/lib/services/theme.service.mjs +4 -4
- package/esm2022/lib/services/toast.service.mjs +4 -4
- package/esm2022/lib/services/types.mjs +1 -1
- package/esm2022/lib/shared/constants/storage.mjs +1 -1
- package/esm2022/lib/shared/pipes/process-links.pipe.mjs +4 -4
- package/esm2022/lib/shared/utils/content.mjs +1 -1
- package/esm2022/lib/shared/utils/dom.mjs +1 -1
- package/esm2022/lib/shared/utils/form-defaults.mjs +1 -1
- package/esm2022/lib/shared/utils/simple-content.mjs +1 -1
- package/esm2022/lib/shared/utils/styles.mjs +1 -1
- package/esm2022/lib/shared/utils/text.mjs +1 -1
- package/esm2022/public-api.mjs +73 -3
- package/esm2022/valtech-components.mjs +1 -1
- package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs +136 -0
- package/fesm2022/valtech-components-simple-modal-content.component-DQhEgUmS.mjs.map +1 -0
- package/fesm2022/valtech-components.mjs +14848 -4293
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/atoms/countdown/countdown.component.d.ts +38 -0
- package/lib/components/atoms/countdown/types.d.ts +108 -0
- package/lib/components/atoms/fab/fab.component.d.ts +16 -0
- package/lib/components/atoms/fab/types.d.ts +45 -0
- package/lib/components/atoms/price-tag/price-tag.component.d.ts +16 -0
- package/lib/components/atoms/price-tag/types.d.ts +59 -0
- package/lib/components/atoms/progress-ring/progress-ring.component.d.ts +20 -0
- package/lib/components/atoms/progress-ring/types.d.ts +24 -0
- package/lib/components/atoms/qr-code/qr-code.component.d.ts +36 -0
- package/lib/components/atoms/qr-code/types.d.ts +124 -0
- package/lib/components/atoms/skeleton/skeleton.component.d.ts +12 -0
- package/lib/components/atoms/skeleton/types.d.ts +29 -0
- package/lib/components/molecules/accordion/accordion.component.d.ts +19 -0
- package/lib/components/molecules/accordion/types.d.ts +47 -0
- package/lib/components/molecules/breadcrumb/breadcrumb.component.d.ts +22 -0
- package/lib/components/molecules/breadcrumb/types.d.ts +45 -0
- package/lib/components/molecules/chip-group/chip-group.component.d.ts +22 -0
- package/lib/components/molecules/chip-group/types.d.ts +65 -0
- package/lib/components/molecules/comment/comment.component.d.ts +42 -0
- package/lib/components/molecules/comment/types.d.ts +171 -0
- package/lib/components/molecules/currency-input/currency-input.component.d.ts +43 -0
- package/lib/components/molecules/currency-input/types.d.ts +96 -0
- package/lib/components/molecules/date-range-input/date-range-input.component.d.ts +42 -0
- package/lib/components/molecules/date-range-input/types.d.ts +109 -0
- package/lib/components/molecules/glow-card/glow-card.component.d.ts +51 -0
- package/lib/components/molecules/glow-card/types.d.ts +92 -0
- package/lib/components/molecules/number-stepper/number-stepper.component.d.ts +34 -0
- package/lib/components/molecules/number-stepper/types.d.ts +88 -0
- package/lib/components/molecules/pagination/pagination.component.d.ts +15 -0
- package/lib/components/molecules/pagination/types.d.ts +41 -0
- package/lib/components/molecules/participant-card/participant-card.component.d.ts +26 -0
- package/lib/components/molecules/participant-card/types.d.ts +132 -0
- package/lib/components/molecules/phone-input/phone-input.component.d.ts +38 -0
- package/lib/components/molecules/phone-input/types.d.ts +98 -0
- package/lib/components/molecules/quote-box/quote-box.component.d.ts +26 -0
- package/lib/components/molecules/raffle-status-card/raffle-status-card.component.d.ts +22 -0
- package/lib/components/molecules/raffle-status-card/types.d.ts +108 -0
- package/lib/components/molecules/range-input/range-input.component.d.ts +30 -0
- package/lib/components/molecules/range-input/types.d.ts +59 -0
- package/lib/components/molecules/rating/rating.component.d.ts +17 -0
- package/lib/components/molecules/rating/types.d.ts +41 -0
- package/lib/components/molecules/segment-control/segment-control.component.d.ts +30 -0
- package/lib/components/molecules/segment-control/types.d.ts +46 -0
- package/lib/components/molecules/share-buttons/share-buttons.component.d.ts +22 -0
- package/lib/components/molecules/share-buttons/types.d.ts +108 -0
- package/lib/components/molecules/stats-card/stats-card.component.d.ts +14 -0
- package/lib/components/molecules/stats-card/types.d.ts +41 -0
- package/lib/components/molecules/stepper/stepper.component.d.ts +21 -0
- package/lib/components/molecules/stepper/types.d.ts +66 -0
- package/lib/components/molecules/tabs/tabs.component.d.ts +17 -0
- package/lib/components/molecules/tabs/types.d.ts +45 -0
- package/lib/components/molecules/textarea-input/textarea-input.component.d.ts +27 -0
- package/lib/components/molecules/textarea-input/types.d.ts +74 -0
- package/lib/components/molecules/ticket-grid/ticket-grid.component.d.ts +41 -0
- package/lib/components/molecules/ticket-grid/types.d.ts +122 -0
- package/lib/components/molecules/timeline/timeline.component.d.ts +14 -0
- package/lib/components/molecules/timeline/types.d.ts +39 -0
- package/lib/components/molecules/toggle-input/toggle-input.component.d.ts +24 -0
- package/lib/components/molecules/toggle-input/types.d.ts +30 -0
- package/lib/components/molecules/winner-display/types.d.ts +103 -0
- package/lib/components/molecules/winner-display/winner-display.component.d.ts +37 -0
- package/lib/components/organisms/article/article.component.d.ts +1 -1
- package/lib/components/organisms/comment-section/comment-section.component.d.ts +52 -0
- package/lib/components/organisms/comment-section/types.d.ts +144 -0
- package/lib/components/organisms/data-table/data-table.component.d.ts +46 -0
- package/lib/components/organisms/data-table/types.d.ts +205 -0
- package/lib/components/types.d.ts +21 -15
- package/lib/services/confirmation-dialog/confirmation-dialog.service.d.ts +71 -0
- package/lib/services/confirmation-dialog/types.d.ts +61 -0
- package/lib/services/modal/modal.service.d.ts +98 -0
- package/lib/services/modal/simple-modal-content.component.d.ts +19 -0
- package/lib/services/modal/types.d.ts +155 -0
- package/lib/services/qr-generator/qr-generator.service.d.ts +115 -0
- package/lib/services/qr-generator/types.d.ts +141 -0
- package/package.json +11 -4
- package/public-api.d.ts +72 -2
- package/LICENSE +0 -21
|
@@ -0,0 +1,341 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import QRCodeStyling from 'qr-code-styling';
|
|
3
|
+
import { QR_PRESETS, } from './types';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
/**
|
|
6
|
+
* QrGeneratorService
|
|
7
|
+
*
|
|
8
|
+
* A comprehensive service for generating, customizing, and manipulating QR codes.
|
|
9
|
+
* Uses qr-code-styling library for high-quality, customizable QR code generation.
|
|
10
|
+
*
|
|
11
|
+
* @example Basic usage
|
|
12
|
+
* ```typescript
|
|
13
|
+
* const qr = await this.qrService.generate({ data: 'https://example.com' });
|
|
14
|
+
* console.log(qr.dataUrl); // base64 image
|
|
15
|
+
* ```
|
|
16
|
+
*
|
|
17
|
+
* @example With customization
|
|
18
|
+
* ```typescript
|
|
19
|
+
* const qr = await this.qrService.generate({
|
|
20
|
+
* data: 'https://example.com',
|
|
21
|
+
* width: 400,
|
|
22
|
+
* dotsStyle: 'rounded',
|
|
23
|
+
* dotsColor: '#3880ff',
|
|
24
|
+
* logo: { src: 'logo.png', width: 60 }
|
|
25
|
+
* });
|
|
26
|
+
* ```
|
|
27
|
+
*
|
|
28
|
+
* @example Using presets
|
|
29
|
+
* ```typescript
|
|
30
|
+
* const qr = await this.qrService.generateFromPreset('modern', 'https://example.com');
|
|
31
|
+
* ```
|
|
32
|
+
*/
|
|
33
|
+
export class QrGeneratorService {
|
|
34
|
+
/**
|
|
35
|
+
* Generate a QR code with the given configuration.
|
|
36
|
+
*
|
|
37
|
+
* @param config - QR code configuration
|
|
38
|
+
* @returns Promise<QrResult> - Generated QR code result
|
|
39
|
+
*
|
|
40
|
+
* @example
|
|
41
|
+
* ```typescript
|
|
42
|
+
* const qr = await this.qrService.generate({
|
|
43
|
+
* data: 'https://myapp.com/product/123',
|
|
44
|
+
* width: 300,
|
|
45
|
+
* dotsStyle: 'rounded',
|
|
46
|
+
* dotsColor: '#000000',
|
|
47
|
+
* backgroundColor: '#ffffff'
|
|
48
|
+
* });
|
|
49
|
+
* ```
|
|
50
|
+
*/
|
|
51
|
+
async generate(config) {
|
|
52
|
+
// Merge with defaults
|
|
53
|
+
const finalConfig = {
|
|
54
|
+
width: 300,
|
|
55
|
+
height: config.height || config.width || 300,
|
|
56
|
+
margin: 0,
|
|
57
|
+
dotsStyle: 'square',
|
|
58
|
+
dotsColor: '#000000',
|
|
59
|
+
backgroundColor: '#ffffff',
|
|
60
|
+
errorCorrectionLevel: 'M',
|
|
61
|
+
format: 'png',
|
|
62
|
+
quality: 1,
|
|
63
|
+
...config,
|
|
64
|
+
};
|
|
65
|
+
// Build qr-code-styling options
|
|
66
|
+
const qrOptions = {
|
|
67
|
+
width: finalConfig.width,
|
|
68
|
+
height: finalConfig.height,
|
|
69
|
+
data: finalConfig.data,
|
|
70
|
+
margin: finalConfig.margin,
|
|
71
|
+
qrOptions: {
|
|
72
|
+
errorCorrectionLevel: finalConfig.errorCorrectionLevel,
|
|
73
|
+
},
|
|
74
|
+
dotsOptions: {
|
|
75
|
+
type: finalConfig.dotsStyle,
|
|
76
|
+
...this.buildColorOptions(finalConfig.dotsColor),
|
|
77
|
+
},
|
|
78
|
+
backgroundOptions: {
|
|
79
|
+
...this.buildColorOptions(finalConfig.backgroundColor),
|
|
80
|
+
},
|
|
81
|
+
cornersSquareOptions: finalConfig.cornersSquareColor || finalConfig.cornersSquareStyle
|
|
82
|
+
? {
|
|
83
|
+
type: finalConfig.cornersSquareStyle,
|
|
84
|
+
...this.buildColorOptions(finalConfig.cornersSquareColor || finalConfig.dotsColor),
|
|
85
|
+
}
|
|
86
|
+
: undefined,
|
|
87
|
+
cornersDotOptions: finalConfig.cornersDotColor || finalConfig.cornersDotStyle
|
|
88
|
+
? {
|
|
89
|
+
type: finalConfig.cornersDotStyle,
|
|
90
|
+
...this.buildColorOptions(finalConfig.cornersDotColor || finalConfig.dotsColor),
|
|
91
|
+
}
|
|
92
|
+
: undefined,
|
|
93
|
+
};
|
|
94
|
+
// Add logo if provided
|
|
95
|
+
if (finalConfig.logo) {
|
|
96
|
+
qrOptions.image = finalConfig.logo.src;
|
|
97
|
+
qrOptions.imageOptions = {
|
|
98
|
+
hideBackgroundDots: finalConfig.logo.hideBackgroundDots ?? true,
|
|
99
|
+
imageSize: 0.4,
|
|
100
|
+
margin: finalConfig.logo.margin ?? 5,
|
|
101
|
+
crossOrigin: 'anonymous',
|
|
102
|
+
};
|
|
103
|
+
}
|
|
104
|
+
// Create QR code instance
|
|
105
|
+
const qrCode = new QRCodeStyling(qrOptions);
|
|
106
|
+
// Generate blob
|
|
107
|
+
const format = finalConfig.format || 'png';
|
|
108
|
+
const mimeType = this.getMimeType(format);
|
|
109
|
+
const extension = format === 'jpeg' ? 'jpg' : format;
|
|
110
|
+
let blob;
|
|
111
|
+
let svg;
|
|
112
|
+
let dataUrl;
|
|
113
|
+
if (format === 'svg') {
|
|
114
|
+
// Get SVG data
|
|
115
|
+
const svgData = await qrCode.getRawData('svg');
|
|
116
|
+
if (!svgData) {
|
|
117
|
+
throw new Error('Failed to generate SVG');
|
|
118
|
+
}
|
|
119
|
+
blob = svgData;
|
|
120
|
+
svg = await blob.text();
|
|
121
|
+
dataUrl = `data:${mimeType};base64,${btoa(svg)}`;
|
|
122
|
+
}
|
|
123
|
+
else {
|
|
124
|
+
// Get image blob
|
|
125
|
+
const imageBlob = await qrCode.getRawData(format);
|
|
126
|
+
if (!imageBlob) {
|
|
127
|
+
throw new Error('Failed to generate image');
|
|
128
|
+
}
|
|
129
|
+
blob = imageBlob;
|
|
130
|
+
dataUrl = await this.blobToDataUrl(blob);
|
|
131
|
+
}
|
|
132
|
+
return {
|
|
133
|
+
config: finalConfig,
|
|
134
|
+
dataUrl,
|
|
135
|
+
blob,
|
|
136
|
+
svg,
|
|
137
|
+
mimeType,
|
|
138
|
+
extension,
|
|
139
|
+
generatedAt: new Date(),
|
|
140
|
+
};
|
|
141
|
+
}
|
|
142
|
+
/**
|
|
143
|
+
* Generate a QR code using a preset configuration.
|
|
144
|
+
*
|
|
145
|
+
* @param preset - Preset name ('default', 'rounded', 'dots', 'classy', 'modern')
|
|
146
|
+
* @param data - Data to encode
|
|
147
|
+
* @param overrides - Optional configuration overrides
|
|
148
|
+
* @returns Promise<QrResult>
|
|
149
|
+
*/
|
|
150
|
+
async generateFromPreset(preset, data, overrides) {
|
|
151
|
+
const presetConfig = QR_PRESETS[preset] || QR_PRESETS['default'];
|
|
152
|
+
return this.generate({
|
|
153
|
+
...presetConfig,
|
|
154
|
+
...overrides,
|
|
155
|
+
data,
|
|
156
|
+
});
|
|
157
|
+
}
|
|
158
|
+
/**
|
|
159
|
+
* Download the QR code as a file.
|
|
160
|
+
*
|
|
161
|
+
* @param qr - QR result to download
|
|
162
|
+
* @param options - Download options
|
|
163
|
+
*/
|
|
164
|
+
async download(qr, options) {
|
|
165
|
+
const filename = options?.filename || `qr-code-${Date.now()}`;
|
|
166
|
+
const format = options?.format || qr.config.format || 'png';
|
|
167
|
+
const extension = format === 'jpeg' ? 'jpg' : format;
|
|
168
|
+
let blob = qr.blob;
|
|
169
|
+
// Convert if different format requested
|
|
170
|
+
if (options?.format && options.format !== qr.config.format) {
|
|
171
|
+
const newQr = await this.generate({
|
|
172
|
+
...qr.config,
|
|
173
|
+
format: options.format,
|
|
174
|
+
});
|
|
175
|
+
blob = newQr.blob;
|
|
176
|
+
}
|
|
177
|
+
// Create download link
|
|
178
|
+
const url = URL.createObjectURL(blob);
|
|
179
|
+
const link = document.createElement('a');
|
|
180
|
+
link.href = url;
|
|
181
|
+
link.download = `${filename}.${extension}`;
|
|
182
|
+
document.body.appendChild(link);
|
|
183
|
+
link.click();
|
|
184
|
+
document.body.removeChild(link);
|
|
185
|
+
URL.revokeObjectURL(url);
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Copy the QR code to clipboard as an image.
|
|
189
|
+
*
|
|
190
|
+
* @param qr - QR result to copy
|
|
191
|
+
* @returns Promise<boolean> - Whether copy was successful
|
|
192
|
+
*/
|
|
193
|
+
async copyToClipboard(qr) {
|
|
194
|
+
try {
|
|
195
|
+
// Clipboard API requires PNG format
|
|
196
|
+
let pngBlob = qr.blob;
|
|
197
|
+
if (qr.mimeType !== 'image/png') {
|
|
198
|
+
const pngQr = await this.generate({
|
|
199
|
+
...qr.config,
|
|
200
|
+
format: 'png',
|
|
201
|
+
});
|
|
202
|
+
pngBlob = pngQr.blob;
|
|
203
|
+
}
|
|
204
|
+
await navigator.clipboard.write([
|
|
205
|
+
new ClipboardItem({
|
|
206
|
+
'image/png': pngBlob,
|
|
207
|
+
}),
|
|
208
|
+
]);
|
|
209
|
+
return true;
|
|
210
|
+
}
|
|
211
|
+
catch (error) {
|
|
212
|
+
console.error('Failed to copy QR to clipboard:', error);
|
|
213
|
+
return false;
|
|
214
|
+
}
|
|
215
|
+
}
|
|
216
|
+
/**
|
|
217
|
+
* Share the QR code using Web Share API.
|
|
218
|
+
*
|
|
219
|
+
* @param qr - QR result to share
|
|
220
|
+
* @param options - Share options
|
|
221
|
+
* @returns Promise<boolean> - Whether share was successful
|
|
222
|
+
*/
|
|
223
|
+
async share(qr, options) {
|
|
224
|
+
if (!navigator.share || !navigator.canShare) {
|
|
225
|
+
console.warn('Web Share API not available');
|
|
226
|
+
return false;
|
|
227
|
+
}
|
|
228
|
+
try {
|
|
229
|
+
const filename = options?.filename || `qr-code-${Date.now()}.${qr.extension}`;
|
|
230
|
+
const file = new File([qr.blob], filename, { type: qr.mimeType });
|
|
231
|
+
const shareData = {
|
|
232
|
+
title: options?.title || 'QR Code',
|
|
233
|
+
text: options?.text,
|
|
234
|
+
files: [file],
|
|
235
|
+
};
|
|
236
|
+
if (navigator.canShare(shareData)) {
|
|
237
|
+
await navigator.share(shareData);
|
|
238
|
+
return true;
|
|
239
|
+
}
|
|
240
|
+
// Fallback without file
|
|
241
|
+
await navigator.share({
|
|
242
|
+
title: options?.title || 'QR Code',
|
|
243
|
+
text: options?.text,
|
|
244
|
+
});
|
|
245
|
+
return true;
|
|
246
|
+
}
|
|
247
|
+
catch (error) {
|
|
248
|
+
if (error.name !== 'AbortError') {
|
|
249
|
+
console.error('Failed to share QR:', error);
|
|
250
|
+
}
|
|
251
|
+
return false;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
/**
|
|
255
|
+
* Get the QR code as a base64 string (without data URL prefix).
|
|
256
|
+
*
|
|
257
|
+
* @param qr - QR result
|
|
258
|
+
* @returns Base64 string
|
|
259
|
+
*/
|
|
260
|
+
toBase64(qr) {
|
|
261
|
+
const prefix = `data:${qr.mimeType};base64,`;
|
|
262
|
+
if (qr.dataUrl.startsWith(prefix)) {
|
|
263
|
+
return qr.dataUrl.substring(prefix.length);
|
|
264
|
+
}
|
|
265
|
+
return qr.dataUrl;
|
|
266
|
+
}
|
|
267
|
+
/**
|
|
268
|
+
* Get the QR code as a data URL.
|
|
269
|
+
*
|
|
270
|
+
* @param qr - QR result
|
|
271
|
+
* @returns Data URL string
|
|
272
|
+
*/
|
|
273
|
+
toDataUrl(qr) {
|
|
274
|
+
return qr.dataUrl;
|
|
275
|
+
}
|
|
276
|
+
/**
|
|
277
|
+
* Get the QR code as a Blob.
|
|
278
|
+
*
|
|
279
|
+
* @param qr - QR result
|
|
280
|
+
* @returns Blob
|
|
281
|
+
*/
|
|
282
|
+
toBlob(qr) {
|
|
283
|
+
return qr.blob;
|
|
284
|
+
}
|
|
285
|
+
/**
|
|
286
|
+
* Check if Web Share API is available.
|
|
287
|
+
*/
|
|
288
|
+
canShare() {
|
|
289
|
+
return !!(navigator.share && navigator.canShare);
|
|
290
|
+
}
|
|
291
|
+
/**
|
|
292
|
+
* Check if Clipboard API is available for images.
|
|
293
|
+
*/
|
|
294
|
+
canCopyToClipboard() {
|
|
295
|
+
return !!(navigator.clipboard && navigator.clipboard.write);
|
|
296
|
+
}
|
|
297
|
+
// === Private Helpers ===
|
|
298
|
+
buildColorOptions(color) {
|
|
299
|
+
if (!color) {
|
|
300
|
+
return {};
|
|
301
|
+
}
|
|
302
|
+
if (typeof color === 'string') {
|
|
303
|
+
return { color };
|
|
304
|
+
}
|
|
305
|
+
// Gradient
|
|
306
|
+
const gradient = color;
|
|
307
|
+
return {
|
|
308
|
+
gradient: {
|
|
309
|
+
type: gradient.type,
|
|
310
|
+
rotation: gradient.rotation || 0,
|
|
311
|
+
colorStops: gradient.colorStops,
|
|
312
|
+
},
|
|
313
|
+
};
|
|
314
|
+
}
|
|
315
|
+
getMimeType(format) {
|
|
316
|
+
const mimeTypes = {
|
|
317
|
+
png: 'image/png',
|
|
318
|
+
jpeg: 'image/jpeg',
|
|
319
|
+
webp: 'image/webp',
|
|
320
|
+
svg: 'image/svg+xml',
|
|
321
|
+
};
|
|
322
|
+
return mimeTypes[format] || 'image/png';
|
|
323
|
+
}
|
|
324
|
+
blobToDataUrl(blob) {
|
|
325
|
+
return new Promise((resolve, reject) => {
|
|
326
|
+
const reader = new FileReader();
|
|
327
|
+
reader.onloadend = () => resolve(reader.result);
|
|
328
|
+
reader.onerror = reject;
|
|
329
|
+
reader.readAsDataURL(blob);
|
|
330
|
+
});
|
|
331
|
+
}
|
|
332
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: QrGeneratorService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
333
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: QrGeneratorService, providedIn: 'root' }); }
|
|
334
|
+
}
|
|
335
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: QrGeneratorService, decorators: [{
|
|
336
|
+
type: Injectable,
|
|
337
|
+
args: [{
|
|
338
|
+
providedIn: 'root',
|
|
339
|
+
}]
|
|
340
|
+
}] });
|
|
341
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"qr-generator.service.js","sourceRoot":"","sources":["../../../../../../src/lib/services/qr-generator/qr-generator.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAC3C,OAAO,aAAa,MAAM,iBAAiB,CAAC;AAC5C,OAAO,EAOL,UAAU,GACX,MAAM,SAAS,CAAC;;AAEjB;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;AAIH,MAAM,OAAO,kBAAkB;IAC7B;;;;;;;;;;;;;;;;OAgBG;IACH,KAAK,CAAC,QAAQ,CAAC,MAAgB;QAE7B,sBAAsB;QACtB,MAAM,WAAW,GAAa;YAC5B,KAAK,EAAE,GAAG;YACV,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,MAAM,CAAC,KAAK,IAAI,GAAG;YAC5C,MAAM,EAAE,CAAC;YACT,SAAS,EAAE,QAAQ;YACnB,SAAS,EAAE,SAAS;YACpB,eAAe,EAAE,SAAS;YAC1B,oBAAoB,EAAE,GAAG;YACzB,MAAM,EAAE,KAAK;YACb,OAAO,EAAE,CAAC;YACV,GAAG,MAAM;SACV,CAAC;QAEF,gCAAgC;QAChC,MAAM,SAAS,GAAQ;YACrB,KAAK,EAAE,WAAW,CAAC,KAAK;YACxB,MAAM,EAAE,WAAW,CAAC,MAAM;YAC1B,IAAI,EAAE,WAAW,CAAC,IAAI;YACtB,MAAM,EAAE,WAAW,CAAC,MAAM;YAC1B,SAAS,EAAE;gBACT,oBAAoB,EAAE,WAAW,CAAC,oBAAoB;aACvD;YACD,WAAW,EAAE;gBACX,IAAI,EAAE,WAAW,CAAC,SAAS;gBAC3B,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,SAAS,CAAC;aACjD;YACD,iBAAiB,EAAE;gBACjB,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,eAAe,CAAC;aACvD;YACD,oBAAoB,EAAE,WAAW,CAAC,kBAAkB,IAAI,WAAW,CAAC,kBAAkB;gBACpF,CAAC,CAAC;oBACE,IAAI,EAAE,WAAW,CAAC,kBAAkB;oBACpC,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,kBAAkB,IAAI,WAAW,CAAC,SAAS,CAAC;iBACnF;gBACH,CAAC,CAAC,SAAS;YACb,iBAAiB,EAAE,WAAW,CAAC,eAAe,IAAI,WAAW,CAAC,eAAe;gBAC3E,CAAC,CAAC;oBACE,IAAI,EAAE,WAAW,CAAC,eAAe;oBACjC,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,eAAe,IAAI,WAAW,CAAC,SAAS,CAAC;iBAChF;gBACH,CAAC,CAAC,SAAS;SACd,CAAC;QAEF,uBAAuB;QACvB,IAAI,WAAW,CAAC,IAAI,EAAE,CAAC;YACrB,SAAS,CAAC,KAAK,GAAG,WAAW,CAAC,IAAI,CAAC,GAAG,CAAC;YACvC,SAAS,CAAC,YAAY,GAAG;gBACvB,kBAAkB,EAAE,WAAW,CAAC,IAAI,CAAC,kBAAkB,IAAI,IAAI;gBAC/D,SAAS,EAAE,GAAG;gBACd,MAAM,EAAE,WAAW,CAAC,IAAI,CAAC,MAAM,IAAI,CAAC;gBACpC,WAAW,EAAE,WAAW;aACzB,CAAC;QACJ,CAAC;QAED,0BAA0B;QAC1B,MAAM,MAAM,GAAG,IAAI,aAAa,CAAC,SAAS,CAAC,CAAC;QAE5C,gBAAgB;QAChB,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,IAAI,KAAK,CAAC;QAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,CAAC;QAC1C,MAAM,SAAS,GAAG,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC;QAErD,IAAI,IAAU,CAAC;QACf,IAAI,GAAuB,CAAC;QAC5B,IAAI,OAAe,CAAC;QAEpB,IAAI,MAAM,KAAK,KAAK,EAAE,CAAC;YACrB,eAAe;YACf,MAAM,OAAO,GAAG,MAAM,MAAM,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;YAC/C,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,MAAM,IAAI,KAAK,CAAC,wBAAwB,CAAC,CAAC;YAC5C,CAAC;YACD,IAAI,GAAG,OAAe,CAAC;YACvB,GAAG,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;YACxB,OAAO,GAAG,QAAQ,QAAQ,WAAW,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC;QACnD,CAAC;aAAM,CAAC;YACN,iBAAiB;YACjB,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,UAAU,CAAC,MAAiC,CAAC,CAAC;YAC7E,IAAI,CAAC,SAAS,EAAE,CAAC;gBACf,MAAM,IAAI,KAAK,CAAC,0BAA0B,CAAC,CAAC;YAC9C,CAAC;YACD,IAAI,GAAG,SAAiB,CAAC;YACzB,OAAO,GAAG,MAAM,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;QAC3C,CAAC;QAED,OAAO;YACL,MAAM,EAAE,WAAW;YACnB,OAAO;YACP,IAAI;YACJ,GAAG;YACH,QAAQ;YACR,SAAS;YACT,WAAW,EAAE,IAAI,IAAI,EAAE;SACxB,CAAC;IACJ,CAAC;IAED;;;;;;;OAOG;IACH,KAAK,CAAC,kBAAkB,CACtB,MAAwC,EACxC,IAAY,EACZ,SAA6B;QAE7B,MAAM,YAAY,GAAG,UAAU,CAAC,MAAM,CAAC,IAAI,UAAU,CAAC,SAAS,CAAC,CAAC;QACjE,OAAO,IAAI,CAAC,QAAQ,CAAC;YACnB,GAAG,YAAY;YACf,GAAG,SAAS;YACZ,IAAI;SACL,CAAC,CAAC;IACL,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,QAAQ,CAAC,EAAY,EAAE,OAA2B;QACtD,MAAM,QAAQ,GAAG,OAAO,EAAE,QAAQ,IAAI,WAAW,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;QAC9D,MAAM,MAAM,GAAG,OAAO,EAAE,MAAM,IAAI,EAAE,CAAC,MAAM,CAAC,MAAM,IAAI,KAAK,CAAC;QAC5D,MAAM,SAAS,GAAG,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC;QAErD,IAAI,IAAI,GAAG,EAAE,CAAC,IAAI,CAAC;QAEnB,wCAAwC;QACxC,IAAI,OAAO,EAAE,MAAM,IAAI,OAAO,CAAC,MAAM,KAAK,EAAE,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YAC3D,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC;gBAChC,GAAG,EAAE,CAAC,MAAM;gBACZ,MAAM,EAAE,OAAO,CAAC,MAAM;aACvB,CAAC,CAAC;YACH,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC;QACpB,CAAC;QAED,uBAAuB;QACvB,MAAM,GAAG,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;QACtC,MAAM,IAAI,GAAG,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;QACzC,IAAI,CAAC,IAAI,GAAG,GAAG,CAAC;QAChB,IAAI,CAAC,QAAQ,GAAG,GAAG,QAAQ,IAAI,SAAS,EAAE,CAAC;QAC3C,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAChC,IAAI,CAAC,KAAK,EAAE,CAAC;QACb,QAAQ,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;QAChC,GAAG,CAAC,eAAe,CAAC,GAAG,CAAC,CAAC;IAC3B,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,eAAe,CAAC,EAAY;QAChC,IAAI,CAAC;YACH,oCAAoC;YACpC,IAAI,OAAO,GAAG,EAAE,CAAC,IAAI,CAAC;YAEtB,IAAI,EAAE,CAAC,QAAQ,KAAK,WAAW,EAAE,CAAC;gBAChC,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC;oBAChC,GAAG,EAAE,CAAC,MAAM;oBACZ,MAAM,EAAE,KAAK;iBACd,CAAC,CAAC;gBACH,OAAO,GAAG,KAAK,CAAC,IAAI,CAAC;YACvB,CAAC;YAED,MAAM,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC;gBAC9B,IAAI,aAAa,CAAC;oBAChB,WAAW,EAAE,OAAO;iBACrB,CAAC;aACH,CAAC,CAAC;YAEH,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,OAAO,CAAC,KAAK,CAAC,iCAAiC,EAAE,KAAK,CAAC,CAAC;YACxD,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,KAAK,CAAC,EAAY,EAAE,OAAwB;QAChD,IAAI,CAAC,SAAS,CAAC,KAAK,IAAI,CAAC,SAAS,CAAC,QAAQ,EAAE,CAAC;YAC5C,OAAO,CAAC,IAAI,CAAC,6BAA6B,CAAC,CAAC;YAC5C,OAAO,KAAK,CAAC;QACf,CAAC;QAED,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,OAAO,EAAE,QAAQ,IAAI,WAAW,IAAI,CAAC,GAAG,EAAE,IAAI,EAAE,CAAC,SAAS,EAAE,CAAC;YAC9E,MAAM,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,QAAQ,EAAE,EAAE,IAAI,EAAE,EAAE,CAAC,QAAQ,EAAE,CAAC,CAAC;YAElE,MAAM,SAAS,GAAc;gBAC3B,KAAK,EAAE,OAAO,EAAE,KAAK,IAAI,SAAS;gBAClC,IAAI,EAAE,OAAO,EAAE,IAAI;gBACnB,KAAK,EAAE,CAAC,IAAI,CAAC;aACd,CAAC;YAEF,IAAI,SAAS,CAAC,QAAQ,CAAC,SAAS,CAAC,EAAE,CAAC;gBAClC,MAAM,SAAS,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;gBACjC,OAAO,IAAI,CAAC;YACd,CAAC;YAED,wBAAwB;YACxB,MAAM,SAAS,CAAC,KAAK,CAAC;gBACpB,KAAK,EAAE,OAAO,EAAE,KAAK,IAAI,SAAS;gBAClC,IAAI,EAAE,OAAO,EAAE,IAAI;aACpB,CAAC,CAAC;YAEH,OAAO,IAAI,CAAC;QACd,CAAC;QAAC,OAAO,KAAK,EAAE,CAAC;YACf,IAAK,KAAe,CAAC,IAAI,KAAK,YAAY,EAAE,CAAC;gBAC3C,OAAO,CAAC,KAAK,CAAC,qBAAqB,EAAE,KAAK,CAAC,CAAC;YAC9C,CAAC;YACD,OAAO,KAAK,CAAC;QACf,CAAC;IACH,CAAC;IAED;;;;;OAKG;IACH,QAAQ,CAAC,EAAY;QACnB,MAAM,MAAM,GAAG,QAAQ,EAAE,CAAC,QAAQ,UAAU,CAAC;QAC7C,IAAI,EAAE,CAAC,OAAO,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YAClC,OAAO,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QAC7C,CAAC;QACD,OAAO,EAAE,CAAC,OAAO,CAAC;IACpB,CAAC;IAED;;;;;OAKG;IACH,SAAS,CAAC,EAAY;QACpB,OAAO,EAAE,CAAC,OAAO,CAAC;IACpB,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,EAAY;QACjB,OAAO,EAAE,CAAC,IAAI,CAAC;IACjB,CAAC;IAED;;OAEG;IACH,QAAQ;QACN,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,IAAI,SAAS,CAAC,QAAQ,CAAC,CAAC;IACnD,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,SAAS,IAAI,SAAS,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;IAC9D,CAAC;IAED,0BAA0B;IAElB,iBAAiB,CAAC,KAAqB;QAC7C,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,EAAE,CAAC;QACZ,CAAC;QAED,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,OAAO,EAAE,KAAK,EAAE,CAAC;QACnB,CAAC;QAED,WAAW;QACX,MAAM,QAAQ,GAAG,KAAmB,CAAC;QACrC,OAAO;YACL,QAAQ,EAAE;gBACR,IAAI,EAAE,QAAQ,CAAC,IAAI;gBACnB,QAAQ,EAAE,QAAQ,CAAC,QAAQ,IAAI,CAAC;gBAChC,UAAU,EAAE,QAAQ,CAAC,UAAU;aAChC;SACF,CAAC;IACJ,CAAC;IAEO,WAAW,CAAC,MAAc;QAChC,MAAM,SAAS,GAA2B;YACxC,GAAG,EAAE,WAAW;YAChB,IAAI,EAAE,YAAY;YAClB,IAAI,EAAE,YAAY;YAClB,GAAG,EAAE,eAAe;SACrB,CAAC;QACF,OAAO,SAAS,CAAC,MAAM,CAAC,IAAI,WAAW,CAAC;IAC1C,CAAC;IAEO,aAAa,CAAC,IAAU;QAC9B,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;YAChC,MAAM,CAAC,SAAS,GAAG,GAAG,EAAE,CAAC,OAAO,CAAC,MAAM,CAAC,MAAgB,CAAC,CAAC;YAC1D,MAAM,CAAC,OAAO,GAAG,MAAM,CAAC;YACxB,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;QAC7B,CAAC,CAAC,CAAC;IACL,CAAC;+GA7UU,kBAAkB;mHAAlB,kBAAkB,cAFjB,MAAM;;4FAEP,kBAAkB;kBAH9B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport QRCodeStyling from 'qr-code-styling';\nimport {\n  QrConfig,\n  QrResult,\n  QrDownloadOptions,\n  QrShareOptions,\n  QrColorConfig,\n  QrGradient,\n  QR_PRESETS,\n} from './types';\n\n/**\n * QrGeneratorService\n *\n * A comprehensive service for generating, customizing, and manipulating QR codes.\n * Uses qr-code-styling library for high-quality, customizable QR code generation.\n *\n * @example Basic usage\n * ```typescript\n * const qr = await this.qrService.generate({ data: 'https://example.com' });\n * console.log(qr.dataUrl); // base64 image\n * ```\n *\n * @example With customization\n * ```typescript\n * const qr = await this.qrService.generate({\n *   data: 'https://example.com',\n *   width: 400,\n *   dotsStyle: 'rounded',\n *   dotsColor: '#3880ff',\n *   logo: { src: 'logo.png', width: 60 }\n * });\n * ```\n *\n * @example Using presets\n * ```typescript\n * const qr = await this.qrService.generateFromPreset('modern', 'https://example.com');\n * ```\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class QrGeneratorService {\n  /**\n   * Generate a QR code with the given configuration.\n   *\n   * @param config - QR code configuration\n   * @returns Promise<QrResult> - Generated QR code result\n   *\n   * @example\n   * ```typescript\n   * const qr = await this.qrService.generate({\n   *   data: 'https://myapp.com/product/123',\n   *   width: 300,\n   *   dotsStyle: 'rounded',\n   *   dotsColor: '#000000',\n   *   backgroundColor: '#ffffff'\n   * });\n   * ```\n   */\n  async generate(config: QrConfig): Promise<QrResult> {\n\n    // Merge with defaults\n    const finalConfig: QrConfig = {\n      width: 300,\n      height: config.height || config.width || 300,\n      margin: 0,\n      dotsStyle: 'square',\n      dotsColor: '#000000',\n      backgroundColor: '#ffffff',\n      errorCorrectionLevel: 'M',\n      format: 'png',\n      quality: 1,\n      ...config,\n    };\n\n    // Build qr-code-styling options\n    const qrOptions: any = {\n      width: finalConfig.width,\n      height: finalConfig.height,\n      data: finalConfig.data,\n      margin: finalConfig.margin,\n      qrOptions: {\n        errorCorrectionLevel: finalConfig.errorCorrectionLevel,\n      },\n      dotsOptions: {\n        type: finalConfig.dotsStyle,\n        ...this.buildColorOptions(finalConfig.dotsColor),\n      },\n      backgroundOptions: {\n        ...this.buildColorOptions(finalConfig.backgroundColor),\n      },\n      cornersSquareOptions: finalConfig.cornersSquareColor || finalConfig.cornersSquareStyle\n        ? {\n            type: finalConfig.cornersSquareStyle,\n            ...this.buildColorOptions(finalConfig.cornersSquareColor || finalConfig.dotsColor),\n          }\n        : undefined,\n      cornersDotOptions: finalConfig.cornersDotColor || finalConfig.cornersDotStyle\n        ? {\n            type: finalConfig.cornersDotStyle,\n            ...this.buildColorOptions(finalConfig.cornersDotColor || finalConfig.dotsColor),\n          }\n        : undefined,\n    };\n\n    // Add logo if provided\n    if (finalConfig.logo) {\n      qrOptions.image = finalConfig.logo.src;\n      qrOptions.imageOptions = {\n        hideBackgroundDots: finalConfig.logo.hideBackgroundDots ?? true,\n        imageSize: 0.4,\n        margin: finalConfig.logo.margin ?? 5,\n        crossOrigin: 'anonymous',\n      };\n    }\n\n    // Create QR code instance\n    const qrCode = new QRCodeStyling(qrOptions);\n\n    // Generate blob\n    const format = finalConfig.format || 'png';\n    const mimeType = this.getMimeType(format);\n    const extension = format === 'jpeg' ? 'jpg' : format;\n\n    let blob: Blob;\n    let svg: string | undefined;\n    let dataUrl: string;\n\n    if (format === 'svg') {\n      // Get SVG data\n      const svgData = await qrCode.getRawData('svg');\n      if (!svgData) {\n        throw new Error('Failed to generate SVG');\n      }\n      blob = svgData as Blob;\n      svg = await blob.text();\n      dataUrl = `data:${mimeType};base64,${btoa(svg)}`;\n    } else {\n      // Get image blob\n      const imageBlob = await qrCode.getRawData(format as 'png' | 'jpeg' | 'webp');\n      if (!imageBlob) {\n        throw new Error('Failed to generate image');\n      }\n      blob = imageBlob as Blob;\n      dataUrl = await this.blobToDataUrl(blob);\n    }\n\n    return {\n      config: finalConfig,\n      dataUrl,\n      blob,\n      svg,\n      mimeType,\n      extension,\n      generatedAt: new Date(),\n    };\n  }\n\n  /**\n   * Generate a QR code using a preset configuration.\n   *\n   * @param preset - Preset name ('default', 'rounded', 'dots', 'classy', 'modern')\n   * @param data - Data to encode\n   * @param overrides - Optional configuration overrides\n   * @returns Promise<QrResult>\n   */\n  async generateFromPreset(\n    preset: keyof typeof QR_PRESETS | string,\n    data: string,\n    overrides?: Partial<QrConfig>\n  ): Promise<QrResult> {\n    const presetConfig = QR_PRESETS[preset] || QR_PRESETS['default'];\n    return this.generate({\n      ...presetConfig,\n      ...overrides,\n      data,\n    });\n  }\n\n  /**\n   * Download the QR code as a file.\n   *\n   * @param qr - QR result to download\n   * @param options - Download options\n   */\n  async download(qr: QrResult, options?: QrDownloadOptions): Promise<void> {\n    const filename = options?.filename || `qr-code-${Date.now()}`;\n    const format = options?.format || qr.config.format || 'png';\n    const extension = format === 'jpeg' ? 'jpg' : format;\n\n    let blob = qr.blob;\n\n    // Convert if different format requested\n    if (options?.format && options.format !== qr.config.format) {\n      const newQr = await this.generate({\n        ...qr.config,\n        format: options.format,\n      });\n      blob = newQr.blob;\n    }\n\n    // Create download link\n    const url = URL.createObjectURL(blob);\n    const link = document.createElement('a');\n    link.href = url;\n    link.download = `${filename}.${extension}`;\n    document.body.appendChild(link);\n    link.click();\n    document.body.removeChild(link);\n    URL.revokeObjectURL(url);\n  }\n\n  /**\n   * Copy the QR code to clipboard as an image.\n   *\n   * @param qr - QR result to copy\n   * @returns Promise<boolean> - Whether copy was successful\n   */\n  async copyToClipboard(qr: QrResult): Promise<boolean> {\n    try {\n      // Clipboard API requires PNG format\n      let pngBlob = qr.blob;\n\n      if (qr.mimeType !== 'image/png') {\n        const pngQr = await this.generate({\n          ...qr.config,\n          format: 'png',\n        });\n        pngBlob = pngQr.blob;\n      }\n\n      await navigator.clipboard.write([\n        new ClipboardItem({\n          'image/png': pngBlob,\n        }),\n      ]);\n\n      return true;\n    } catch (error) {\n      console.error('Failed to copy QR to clipboard:', error);\n      return false;\n    }\n  }\n\n  /**\n   * Share the QR code using Web Share API.\n   *\n   * @param qr - QR result to share\n   * @param options - Share options\n   * @returns Promise<boolean> - Whether share was successful\n   */\n  async share(qr: QrResult, options?: QrShareOptions): Promise<boolean> {\n    if (!navigator.share || !navigator.canShare) {\n      console.warn('Web Share API not available');\n      return false;\n    }\n\n    try {\n      const filename = options?.filename || `qr-code-${Date.now()}.${qr.extension}`;\n      const file = new File([qr.blob], filename, { type: qr.mimeType });\n\n      const shareData: ShareData = {\n        title: options?.title || 'QR Code',\n        text: options?.text,\n        files: [file],\n      };\n\n      if (navigator.canShare(shareData)) {\n        await navigator.share(shareData);\n        return true;\n      }\n\n      // Fallback without file\n      await navigator.share({\n        title: options?.title || 'QR Code',\n        text: options?.text,\n      });\n\n      return true;\n    } catch (error) {\n      if ((error as Error).name !== 'AbortError') {\n        console.error('Failed to share QR:', error);\n      }\n      return false;\n    }\n  }\n\n  /**\n   * Get the QR code as a base64 string (without data URL prefix).\n   *\n   * @param qr - QR result\n   * @returns Base64 string\n   */\n  toBase64(qr: QrResult): string {\n    const prefix = `data:${qr.mimeType};base64,`;\n    if (qr.dataUrl.startsWith(prefix)) {\n      return qr.dataUrl.substring(prefix.length);\n    }\n    return qr.dataUrl;\n  }\n\n  /**\n   * Get the QR code as a data URL.\n   *\n   * @param qr - QR result\n   * @returns Data URL string\n   */\n  toDataUrl(qr: QrResult): string {\n    return qr.dataUrl;\n  }\n\n  /**\n   * Get the QR code as a Blob.\n   *\n   * @param qr - QR result\n   * @returns Blob\n   */\n  toBlob(qr: QrResult): Blob {\n    return qr.blob;\n  }\n\n  /**\n   * Check if Web Share API is available.\n   */\n  canShare(): boolean {\n    return !!(navigator.share && navigator.canShare);\n  }\n\n  /**\n   * Check if Clipboard API is available for images.\n   */\n  canCopyToClipboard(): boolean {\n    return !!(navigator.clipboard && navigator.clipboard.write);\n  }\n\n  // === Private Helpers ===\n\n  private buildColorOptions(color?: QrColorConfig): any {\n    if (!color) {\n      return {};\n    }\n\n    if (typeof color === 'string') {\n      return { color };\n    }\n\n    // Gradient\n    const gradient = color as QrGradient;\n    return {\n      gradient: {\n        type: gradient.type,\n        rotation: gradient.rotation || 0,\n        colorStops: gradient.colorStops,\n      },\n    };\n  }\n\n  private getMimeType(format: string): string {\n    const mimeTypes: Record<string, string> = {\n      png: 'image/png',\n      jpeg: 'image/jpeg',\n      webp: 'image/webp',\n      svg: 'image/svg+xml',\n    };\n    return mimeTypes[format] || 'image/png';\n  }\n\n  private blobToDataUrl(blob: Blob): Promise<string> {\n    return new Promise((resolve, reject) => {\n      const reader = new FileReader();\n      reader.onloadend = () => resolve(reader.result as string);\n      reader.onerror = reject;\n      reader.readAsDataURL(blob);\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Default presets for quick styling.
|
|
3
|
+
*/
|
|
4
|
+
export const QR_PRESETS = {
|
|
5
|
+
default: {
|
|
6
|
+
width: 300,
|
|
7
|
+
dotsStyle: 'square',
|
|
8
|
+
dotsColor: '#000000',
|
|
9
|
+
backgroundColor: '#ffffff',
|
|
10
|
+
errorCorrectionLevel: 'M',
|
|
11
|
+
},
|
|
12
|
+
rounded: {
|
|
13
|
+
width: 300,
|
|
14
|
+
dotsStyle: 'rounded',
|
|
15
|
+
cornersSquareStyle: 'extra-rounded',
|
|
16
|
+
cornersDotStyle: 'dot',
|
|
17
|
+
dotsColor: '#000000',
|
|
18
|
+
backgroundColor: '#ffffff',
|
|
19
|
+
},
|
|
20
|
+
dots: {
|
|
21
|
+
width: 300,
|
|
22
|
+
dotsStyle: 'dots',
|
|
23
|
+
cornersSquareStyle: 'dot',
|
|
24
|
+
cornersDotStyle: 'dot',
|
|
25
|
+
dotsColor: '#000000',
|
|
26
|
+
backgroundColor: '#ffffff',
|
|
27
|
+
},
|
|
28
|
+
classy: {
|
|
29
|
+
width: 300,
|
|
30
|
+
dotsStyle: 'classy',
|
|
31
|
+
cornersSquareStyle: 'extra-rounded',
|
|
32
|
+
cornersDotStyle: 'dot',
|
|
33
|
+
dotsColor: '#000000',
|
|
34
|
+
backgroundColor: '#ffffff',
|
|
35
|
+
},
|
|
36
|
+
modern: {
|
|
37
|
+
width: 300,
|
|
38
|
+
dotsStyle: 'extra-rounded',
|
|
39
|
+
cornersSquareStyle: 'extra-rounded',
|
|
40
|
+
cornersDotStyle: 'dot',
|
|
41
|
+
dotsColor: '#1a1a1a',
|
|
42
|
+
backgroundColor: '#ffffff',
|
|
43
|
+
margin: 2,
|
|
44
|
+
},
|
|
45
|
+
};
|
|
46
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../../src/lib/services/qr-generator/types.ts"],"names":[],"mappings":"AAkKA;;GAEG;AACH,MAAM,CAAC,MAAM,UAAU,GAAsC;IAC3D,OAAO,EAAE;QACP,KAAK,EAAE,GAAG;QACV,SAAS,EAAE,QAAQ;QACnB,SAAS,EAAE,SAAS;QACpB,eAAe,EAAE,SAAS;QAC1B,oBAAoB,EAAE,GAAG;KAC1B;IACD,OAAO,EAAE;QACP,KAAK,EAAE,GAAG;QACV,SAAS,EAAE,SAAS;QACpB,kBAAkB,EAAE,eAAe;QACnC,eAAe,EAAE,KAAK;QACtB,SAAS,EAAE,SAAS;QACpB,eAAe,EAAE,SAAS;KAC3B;IACD,IAAI,EAAE;QACJ,KAAK,EAAE,GAAG;QACV,SAAS,EAAE,MAAM;QACjB,kBAAkB,EAAE,KAAK;QACzB,eAAe,EAAE,KAAK;QACtB,SAAS,EAAE,SAAS;QACpB,eAAe,EAAE,SAAS;KAC3B;IACD,MAAM,EAAE;QACN,KAAK,EAAE,GAAG;QACV,SAAS,EAAE,QAAQ;QACnB,kBAAkB,EAAE,eAAe;QACnC,eAAe,EAAE,KAAK;QACtB,SAAS,EAAE,SAAS;QACpB,eAAe,EAAE,SAAS;KAC3B;IACD,MAAM,EAAE;QACN,KAAK,EAAE,GAAG;QACV,SAAS,EAAE,eAAe;QAC1B,kBAAkB,EAAE,eAAe;QACnC,eAAe,EAAE,KAAK;QACtB,SAAS,EAAE,SAAS;QACpB,eAAe,EAAE,SAAS;QAC1B,MAAM,EAAE,CAAC;KACV;CACF,CAAC","sourcesContent":["/**\n * Style options for QR code dots (data modules).\n */\nexport type QrDotsStyle = 'square' | 'dots' | 'rounded' | 'extra-rounded' | 'classy' | 'classy-rounded';\n\n/**\n * Style options for QR code corner squares.\n */\nexport type QrCornersSquareStyle = 'square' | 'dot' | 'extra-rounded';\n\n/**\n * Style options for QR code corner dots.\n */\nexport type QrCornersDotStyle = 'square' | 'dot';\n\n/**\n * Output format for the generated QR code.\n */\nexport type QrOutputFormat = 'svg' | 'png' | 'jpeg' | 'webp';\n\n/**\n * Error correction level for QR codes.\n * Higher levels allow more damage tolerance but reduce data capacity.\n */\nexport type QrErrorCorrectionLevel = 'L' | 'M' | 'Q' | 'H';\n\n/**\n * Logo/image configuration for the QR code center.\n */\nexport interface QrLogoConfig {\n  /** Image source URL or base64 */\n  src: string;\n  /** Logo width (default: 25% of QR size) */\n  width?: number;\n  /** Logo height (default: same as width) */\n  height?: number;\n  /** Margin around the logo */\n  margin?: number;\n  /** Border radius for rounded corners */\n  borderRadius?: number;\n  /** Whether to hide dots behind the logo */\n  hideBackgroundDots?: boolean;\n}\n\n/**\n * Gradient configuration for QR elements.\n */\nexport interface QrGradient {\n  /** Gradient type */\n  type: 'linear' | 'radial';\n  /** Rotation angle in degrees (for linear) */\n  rotation?: number;\n  /** Color stops */\n  colorStops: Array<{\n    offset: number;\n    color: string;\n  }>;\n}\n\n/**\n * Color configuration - can be solid color or gradient.\n */\nexport type QrColorConfig = string | QrGradient;\n\n/**\n * Complete configuration for QR code generation.\n */\nexport interface QrConfig {\n  /** The data to encode (URL, text, etc.) */\n  data: string;\n\n  // === Size ===\n  /** QR code width in pixels (default: 300) */\n  width?: number;\n  /** QR code height in pixels (default: same as width) */\n  height?: number;\n  /** Quiet zone margin in modules (default: 0) */\n  margin?: number;\n\n  // === Colors ===\n  /** Color/gradient for data dots */\n  dotsColor?: QrColorConfig;\n  /** Background color (default: white) */\n  backgroundColor?: QrColorConfig;\n  /** Color for corner squares */\n  cornersSquareColor?: QrColorConfig;\n  /** Color for corner dots */\n  cornersDotColor?: QrColorConfig;\n\n  // === Styles ===\n  /** Style of data dots (default: 'square') */\n  dotsStyle?: QrDotsStyle;\n  /** Style of corner squares (default: 'square') */\n  cornersSquareStyle?: QrCornersSquareStyle;\n  /** Style of corner dots (default: 'square') */\n  cornersDotStyle?: QrCornersDotStyle;\n\n  // === Logo ===\n  /** Logo/image configuration */\n  logo?: QrLogoConfig;\n\n  // === Technical ===\n  /** Error correction level (default: 'M') */\n  errorCorrectionLevel?: QrErrorCorrectionLevel;\n\n  // === Output ===\n  /** Output format (default: 'png') */\n  format?: QrOutputFormat;\n  /** Quality for jpeg/webp (0-1, default: 1) */\n  quality?: number;\n}\n\n/**\n * Result from QR code generation.\n */\nexport interface QrResult {\n  /** Original configuration used */\n  config: QrConfig;\n  /** Generated data as base64 data URL */\n  dataUrl: string;\n  /** Generated data as Blob */\n  blob: Blob;\n  /** SVG string (only when format is 'svg') */\n  svg?: string;\n  /** MIME type of the generated image */\n  mimeType: string;\n  /** File extension */\n  extension: string;\n  /** Timestamp of generation */\n  generatedAt: Date;\n}\n\n/**\n * Options for downloading the QR code.\n */\nexport interface QrDownloadOptions {\n  /** Custom filename (without extension) */\n  filename?: string;\n  /** Override format for download */\n  format?: QrOutputFormat;\n}\n\n/**\n * Options for sharing the QR code.\n */\nexport interface QrShareOptions {\n  /** Share title */\n  title?: string;\n  /** Share text/description */\n  text?: string;\n  /** Custom filename for the shared file */\n  filename?: string;\n}\n\n/**\n * Preset configurations for common use cases.\n */\nexport interface QrPreset {\n  name: string;\n  config: Partial<QrConfig>;\n}\n\n/**\n * Default presets for quick styling.\n */\nexport const QR_PRESETS: Record<string, Partial<QrConfig>> = {\n  default: {\n    width: 300,\n    dotsStyle: 'square',\n    dotsColor: '#000000',\n    backgroundColor: '#ffffff',\n    errorCorrectionLevel: 'M',\n  },\n  rounded: {\n    width: 300,\n    dotsStyle: 'rounded',\n    cornersSquareStyle: 'extra-rounded',\n    cornersDotStyle: 'dot',\n    dotsColor: '#000000',\n    backgroundColor: '#ffffff',\n  },\n  dots: {\n    width: 300,\n    dotsStyle: 'dots',\n    cornersSquareStyle: 'dot',\n    cornersDotStyle: 'dot',\n    dotsColor: '#000000',\n    backgroundColor: '#ffffff',\n  },\n  classy: {\n    width: 300,\n    dotsStyle: 'classy',\n    cornersSquareStyle: 'extra-rounded',\n    cornersDotStyle: 'dot',\n    dotsColor: '#000000',\n    backgroundColor: '#ffffff',\n  },\n  modern: {\n    width: 300,\n    dotsStyle: 'extra-rounded',\n    cornersSquareStyle: 'extra-rounded',\n    cornersDotStyle: 'dot',\n    dotsColor: '#1a1a1a',\n    backgroundColor: '#ffffff',\n    margin: 2,\n  },\n};\n"]}
|
|
@@ -132,13 +132,13 @@ export class ThemeService {
|
|
|
132
132
|
break;
|
|
133
133
|
}
|
|
134
134
|
}
|
|
135
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
136
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "
|
|
135
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ThemeService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
136
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ThemeService, providedIn: 'root' }); }
|
|
137
137
|
}
|
|
138
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
138
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ThemeService, decorators: [{
|
|
139
139
|
type: Injectable,
|
|
140
140
|
args: [{
|
|
141
141
|
providedIn: 'root',
|
|
142
142
|
}]
|
|
143
143
|
}], ctorParameters: () => [] });
|
|
144
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme.service.js","sourceRoot":"","sources":["../../../../../projects/valtech-components/src/lib/services/theme.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Service for managing application themes (light, dark, auto).\n * Handles user preferences, system theme detection, and theme toggling.\n * Uses localStorage to persist the selected theme.\n */\n\nimport { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { THEME } from '../shared/constants/storage';\nimport { LocalStorageService } from './local-storage.service';\n\nexport enum ThemeOption {\n  LIGHT = 'light',\n  DARK = 'dark',\n  AUTO = 'auto',\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class ThemeService {\n  /**\n   * Indicates if the light theme is active.\n   */\n  lightToggle = false;\n  /**\n   * Indicates if the dark theme is active.\n   */\n  darkToggle = false;\n  /**\n   * Indicates if the auto theme is active.\n   */\n  autoToggle = false;\n  /**\n   * The current selected theme option.\n   */\n  currentOption = ThemeOption.AUTO;\n  /**\n   * Enum with all available theme options.\n   */\n  themeOptions = ThemeOption;\n  /**\n   * Whether the system prefers dark mode.\n   */\n  prefersDark = false;\n  /**\n   * The default theme option.\n   */\n  default = ThemeOption.AUTO;\n  /**\n   * Observable for the current theme.\n   */\n  theme: BehaviorSubject<ThemeOption>;\n\n  constructor() {\n    const current = LocalStorageService.get<ThemeOption>(THEME);\n    console.log('💡 ThemeConfig current::: ', current);\n    this.theme = new BehaviorSubject<ThemeOption>(current || this.default);\n    this.currentOption = this.Theme;\n    console.log('💡 ThemeConfig this.currentOption::: ', this.currentOption);\n    this.toggleUserPreference(this.currentOption);\n\n    const prefersDarkQuery = window.matchMedia('(prefers-color-scheme: dark)');\n    this.prefersDark = prefersDarkQuery.matches;\n    this.handleAutoConfiguration();\n\n    prefersDarkQuery.addEventListener('change', mediaQuery => {\n      console.log('💡 ThemeConfig addEventListener change::: ', mediaQuery);\n      this.prefersDark = mediaQuery.matches;\n      this.handleAutoConfiguration();\n    });\n  }\n\n  private handleAutoConfiguration() {\n    if (this.prefersDark && this.currentOption === ThemeOption.AUTO) {\n      console.log('💡 ThemeConfig prefersDark::: ', this.prefersDark);\n      this.toggleUserPreference(ThemeOption.AUTO);\n    }\n    if (!this.prefersDark && this.currentOption === ThemeOption.AUTO) {\n      console.log('💡 ThemeConfig prefersDark::: ', this.prefersDark);\n      this.toggleUserPreference(ThemeOption.AUTO);\n    }\n  }\n\n  private handleDarkPreference() {\n    this.toggleTheme(ThemeOption.DARK, true);\n    this.toggleTheme(ThemeOption.LIGHT, false);\n  }\n\n  private handleLightPreference() {\n    this.toggleTheme(ThemeOption.LIGHT, true);\n    this.toggleTheme(ThemeOption.DARK, false);\n  }\n\n  /**\n   * Returns true if the system prefers dark mode.\n   */\n  get IsDark(): boolean {\n    return this.prefersDark;\n  }\n\n  /**\n   * Gets the current theme option.\n   */\n  get Theme(): ThemeOption {\n    return this.theme.value;\n  }\n\n  /**\n   * Sets the current theme option and persists it.\n   */\n  set Theme(theme: ThemeOption) {\n    this.theme.next(theme);\n    LocalStorageService.set<ThemeOption>(THEME, theme);\n  }\n\n  /**\n   * Toggles a theme class on the document body.\n   * @param name The theme name\n   * @param shouldAdd Whether to add or remove the class\n   */\n  toggleTheme(name: string, shouldAdd: boolean) {\n    console.log('toggleTheme::: ', name, shouldAdd);\n    document.body.classList.toggle(name, shouldAdd);\n  }\n\n  /**\n   * Sets the user theme preference and updates toggles and theme classes.\n   * @param option The selected theme option\n   */\n  toggleUserPreference(option: ThemeOption) {\n    this.currentOption = option;\n    this.Theme = option;\n    this.lightToggle = option === ThemeOption.LIGHT;\n    this.darkToggle = option === ThemeOption.DARK;\n    this.autoToggle = option === ThemeOption.AUTO;\n\n    switch (option) {\n      case ThemeOption.LIGHT:\n        this.handleLightPreference();\n        break;\n      case ThemeOption.DARK:\n        this.handleDarkPreference();\n        break;\n      case ThemeOption.AUTO:\n        if (this.prefersDark) {\n          this.handleDarkPreference();\n        } else {\n          this.handleLightPreference();\n        }\n        break;\n    }\n  }\n}\n"]}
|
|
144
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"theme.service.js","sourceRoot":"","sources":["../../../../../src/lib/services/theme.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Service for managing application themes (light, dark, auto).\n * Handles user preferences, system theme detection, and theme toggling.\n * Uses localStorage to persist the selected theme.\n */\n\nimport { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { THEME } from '../shared/constants/storage';\nimport { LocalStorageService } from './local-storage.service';\n\nexport enum ThemeOption {\n  LIGHT = 'light',\n  DARK = 'dark',\n  AUTO = 'auto',\n}\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class ThemeService {\n  /**\n   * Indicates if the light theme is active.\n   */\n  lightToggle = false;\n  /**\n   * Indicates if the dark theme is active.\n   */\n  darkToggle = false;\n  /**\n   * Indicates if the auto theme is active.\n   */\n  autoToggle = false;\n  /**\n   * The current selected theme option.\n   */\n  currentOption = ThemeOption.AUTO;\n  /**\n   * Enum with all available theme options.\n   */\n  themeOptions = ThemeOption;\n  /**\n   * Whether the system prefers dark mode.\n   */\n  prefersDark = false;\n  /**\n   * The default theme option.\n   */\n  default = ThemeOption.AUTO;\n  /**\n   * Observable for the current theme.\n   */\n  theme: BehaviorSubject<ThemeOption>;\n\n  constructor() {\n    const current = LocalStorageService.get<ThemeOption>(THEME);\n    console.log('💡 ThemeConfig current::: ', current);\n    this.theme = new BehaviorSubject<ThemeOption>(current || this.default);\n    this.currentOption = this.Theme;\n    console.log('💡 ThemeConfig this.currentOption::: ', this.currentOption);\n    this.toggleUserPreference(this.currentOption);\n\n    const prefersDarkQuery = window.matchMedia('(prefers-color-scheme: dark)');\n    this.prefersDark = prefersDarkQuery.matches;\n    this.handleAutoConfiguration();\n\n    prefersDarkQuery.addEventListener('change', mediaQuery => {\n      console.log('💡 ThemeConfig addEventListener change::: ', mediaQuery);\n      this.prefersDark = mediaQuery.matches;\n      this.handleAutoConfiguration();\n    });\n  }\n\n  private handleAutoConfiguration() {\n    if (this.prefersDark && this.currentOption === ThemeOption.AUTO) {\n      console.log('💡 ThemeConfig prefersDark::: ', this.prefersDark);\n      this.toggleUserPreference(ThemeOption.AUTO);\n    }\n    if (!this.prefersDark && this.currentOption === ThemeOption.AUTO) {\n      console.log('💡 ThemeConfig prefersDark::: ', this.prefersDark);\n      this.toggleUserPreference(ThemeOption.AUTO);\n    }\n  }\n\n  private handleDarkPreference() {\n    this.toggleTheme(ThemeOption.DARK, true);\n    this.toggleTheme(ThemeOption.LIGHT, false);\n  }\n\n  private handleLightPreference() {\n    this.toggleTheme(ThemeOption.LIGHT, true);\n    this.toggleTheme(ThemeOption.DARK, false);\n  }\n\n  /**\n   * Returns true if the system prefers dark mode.\n   */\n  get IsDark(): boolean {\n    return this.prefersDark;\n  }\n\n  /**\n   * Gets the current theme option.\n   */\n  get Theme(): ThemeOption {\n    return this.theme.value;\n  }\n\n  /**\n   * Sets the current theme option and persists it.\n   */\n  set Theme(theme: ThemeOption) {\n    this.theme.next(theme);\n    LocalStorageService.set<ThemeOption>(THEME, theme);\n  }\n\n  /**\n   * Toggles a theme class on the document body.\n   * @param name The theme name\n   * @param shouldAdd Whether to add or remove the class\n   */\n  toggleTheme(name: string, shouldAdd: boolean) {\n    console.log('toggleTheme::: ', name, shouldAdd);\n    document.body.classList.toggle(name, shouldAdd);\n  }\n\n  /**\n   * Sets the user theme preference and updates toggles and theme classes.\n   * @param option The selected theme option\n   */\n  toggleUserPreference(option: ThemeOption) {\n    this.currentOption = option;\n    this.Theme = option;\n    this.lightToggle = option === ThemeOption.LIGHT;\n    this.darkToggle = option === ThemeOption.DARK;\n    this.autoToggle = option === ThemeOption.AUTO;\n\n    switch (option) {\n      case ThemeOption.LIGHT:\n        this.handleLightPreference();\n        break;\n      case ThemeOption.DARK:\n        this.handleDarkPreference();\n        break;\n      case ThemeOption.AUTO:\n        if (this.prefersDark) {\n          this.handleDarkPreference();\n        } else {\n          this.handleLightPreference();\n        }\n        break;\n    }\n  }\n}\n"]}
|
|
@@ -35,13 +35,13 @@ export class ToastService {
|
|
|
35
35
|
console.error(JSON.stringify(error));
|
|
36
36
|
});
|
|
37
37
|
}
|
|
38
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
39
|
-
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "
|
|
38
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToastService, deps: [{ token: i1.ToastController }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
39
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToastService, providedIn: 'root' }); }
|
|
40
40
|
}
|
|
41
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
41
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ToastService, decorators: [{
|
|
42
42
|
type: Injectable,
|
|
43
43
|
args: [{
|
|
44
44
|
providedIn: 'root',
|
|
45
45
|
}]
|
|
46
46
|
}], ctorParameters: () => [{ type: i1.ToastController }] });
|
|
47
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
47
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidG9hc3Quc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2VydmljZXMvdG9hc3Quc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOzs7QUFHM0M7OztHQUdHO0FBSUgsTUFBTSxPQUFPLFlBQVk7SUFDdkIsWUFBb0IsZUFBZ0M7UUFBaEMsb0JBQWUsR0FBZixlQUFlLENBQWlCO0lBQUcsQ0FBQztJQUV4RDs7O09BR0c7SUFDSCxLQUFLLENBQUMsWUFBWSxDQUFDLE9BQXFCO1FBQ3RDLE1BQU0sS0FBSyxHQUFHLE1BQU0sSUFBSSxDQUFDLGVBQWUsQ0FBQyxNQUFNLENBQUM7WUFDOUMsT0FBTyxFQUFFLE9BQU8sQ0FBQyxPQUFPO1lBQ3hCLFFBQVEsRUFBRSxPQUFPLENBQUMsUUFBUTtZQUMxQixRQUFRLEVBQUUsT0FBTyxDQUFDLFFBQVE7WUFDMUIsS0FBSyxFQUFFLE9BQU8sQ0FBQyxLQUFLO1NBQ3JCLENBQUMsQ0FBQztRQUVILE1BQU0sS0FBSyxDQUFDLE9BQU8sRUFBRSxDQUFDO0lBQ3hCLENBQUM7SUFFRDs7O09BR0c7SUFDSCxJQUFJLENBQUMsT0FBcUI7UUFDeEIsSUFBSSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUM7YUFDdkIsSUFBSSxDQUFDLEdBQUcsRUFBRTtZQUNULE9BQU8sQ0FBQyxJQUFJLENBQUMsZUFBZSxDQUFDLENBQUM7UUFDaEMsQ0FBQyxDQUFDO2FBQ0QsS0FBSyxDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ2IsT0FBTyxDQUFDLEtBQUssQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLEtBQUssQ0FBQyxDQUFDLENBQUM7UUFDdkMsQ0FBQyxDQUFDLENBQUM7SUFDUCxDQUFDOytHQTlCVSxZQUFZO21IQUFaLFlBQVksY0FGWCxNQUFNOzs0RkFFUCxZQUFZO2tCQUh4QixVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFRvYXN0Q29udHJvbGxlciwgVG9hc3RPcHRpb25zIH0gZnJvbSAnQGlvbmljL2FuZ3VsYXInO1xuXG4vKipcbiAqIFNlcnZpY2UgZm9yIGRpc3BsYXlpbmcgdG9hc3Qgbm90aWZpY2F0aW9ucyB1c2luZyBJb25pYydzIFRvYXN0Q29udHJvbGxlci5cbiAqIFByb3ZpZGVzIG1ldGhvZHMgdG8gc2hvdyBhbmQgcHJlc2VudCB0b2FzdHMgd2l0aCBjdXN0b20gb3B0aW9ucy5cbiAqL1xuQEluamVjdGFibGUoe1xuICBwcm92aWRlZEluOiAncm9vdCcsXG59KVxuZXhwb3J0IGNsYXNzIFRvYXN0U2VydmljZSB7XG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgdG9hc3RDb250cm9sbGVyOiBUb2FzdENvbnRyb2xsZXIpIHt9XG5cbiAgLyoqXG4gICAqIFByZXNlbnRzIGEgdG9hc3Qgbm90aWZpY2F0aW9uIHdpdGggdGhlIGdpdmVuIG9wdGlvbnMuXG4gICAqIEBwYXJhbSByZXF1ZXN0IFRvYXN0IG9wdGlvbnMgKG1lc3NhZ2UsIGR1cmF0aW9uLCBwb3NpdGlvbiwgY29sb3IsIGV0Yy4pXG4gICAqL1xuICBhc3luYyBwcmVzZW50VG9hc3QocmVxdWVzdDogVG9hc3RPcHRpb25zKSB7XG4gICAgY29uc3QgdG9hc3QgPSBhd2FpdCB0aGlzLnRvYXN0Q29udHJvbGxlci5jcmVhdGUoe1xuICAgICAgbWVzc2FnZTogcmVxdWVzdC5tZXNzYWdlLFxuICAgICAgZHVyYXRpb246IHJlcXVlc3QuZHVyYXRpb24sXG4gICAgICBwb3NpdGlvbjogcmVxdWVzdC5wb3NpdGlvbixcbiAgICAgIGNvbG9yOiByZXF1ZXN0LmNvbG9yLFxuICAgIH0pO1xuXG4gICAgYXdhaXQgdG9hc3QucHJlc2VudCgpO1xuICB9XG5cbiAgLyoqXG4gICAqIFNob3dzIGEgdG9hc3Qgbm90aWZpY2F0aW9uIGFuZCBsb2dzIHRoZSByZXN1bHQuXG4gICAqIEBwYXJhbSByZXF1ZXN0IFRvYXN0IG9wdGlvbnMgKG1lc3NhZ2UsIGR1cmF0aW9uLCBwb3NpdGlvbiwgY29sb3IsIGV0Yy4pXG4gICAqL1xuICBzaG93KHJlcXVlc3Q6IFRvYXN0T3B0aW9ucykge1xuICAgIHRoaXMucHJlc2VudFRvYXN0KHJlcXVlc3QpXG4gICAgICAudGhlbigoKSA9PiB7XG4gICAgICAgIGNvbnNvbGUuaW5mbygnVG9hc3QgY3JlYXRlZCcpO1xuICAgICAgfSlcbiAgICAgIC5jYXRjaChlcnJvciA9PiB7XG4gICAgICAgIGNvbnNvbGUuZXJyb3IoSlNPTi5zdHJpbmdpZnkoZXJyb3IpKTtcbiAgICAgIH0pO1xuICB9XG59XG4iXX0=
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
import { InjectionToken } from '@angular/core';
|
|
2
2
|
export const ValtechConfigService = new InjectionToken('ValtechConfig');
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9zcmMvbGliL3NlcnZpY2VzL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFPL0MsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsSUFBSSxjQUFjLENBQWdCLGVBQWUsQ0FBQyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0aW9uVG9rZW4gfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFByb3ZpZGVyIH0gZnJvbSAnLi9sYW5nLXByb3ZpZGVyL2NvbnRlbnQnO1xuXG5leHBvcnQgaW50ZXJmYWNlIFZhbHRlY2hDb25maWcge1xuICBjb250ZW50OiBQcm92aWRlcjtcbn1cblxuZXhwb3J0IGNvbnN0IFZhbHRlY2hDb25maWdTZXJ2aWNlID0gbmV3IEluamVjdGlvblRva2VuPFZhbHRlY2hDb25maWc+KCdWYWx0ZWNoQ29uZmlnJyk7XG4iXX0=
|
|
@@ -1,3 +1,3 @@
|
|
|
1
1
|
export const LANG = 'LANG';
|
|
2
2
|
export const THEME = 'THEME';
|
|
3
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
3
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RvcmFnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3NyYy9saWIvc2hhcmVkL2NvbnN0YW50cy9zdG9yYWdlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE1BQU0sQ0FBQyxNQUFNLElBQUksR0FBRyxNQUFNLENBQUM7QUFDM0IsTUFBTSxDQUFDLE1BQU0sS0FBSyxHQUFHLE9BQU8sQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCBjb25zdCBMQU5HID0gJ0xBTkcnO1xuZXhwb3J0IGNvbnN0IFRIRU1FID0gJ1RIRU1FJztcbiJdfQ==
|
|
@@ -56,14 +56,14 @@ export class ProcessLinksPipe {
|
|
|
56
56
|
}
|
|
57
57
|
return this.linkProcessor.processLinks(value, config);
|
|
58
58
|
}
|
|
59
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
60
|
-
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "
|
|
59
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ProcessLinksPipe, deps: [], target: i0.ɵɵFactoryTarget.Pipe }); }
|
|
60
|
+
static { this.ɵpipe = i0.ɵɵngDeclarePipe({ minVersion: "14.0.0", version: "18.2.14", ngImport: i0, type: ProcessLinksPipe, isStandalone: true, name: "processLinks" }); }
|
|
61
61
|
}
|
|
62
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
62
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: ProcessLinksPipe, decorators: [{
|
|
63
63
|
type: Pipe,
|
|
64
64
|
args: [{
|
|
65
65
|
name: 'processLinks',
|
|
66
66
|
standalone: true,
|
|
67
67
|
}]
|
|
68
68
|
}] });
|
|
69
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
69
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvY2Vzcy1saW5rcy5waXBlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9zaGFyZWQvcGlwZXMvcHJvY2Vzcy1saW5rcy5waXBlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxJQUFJLEVBQWlCLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUU1RCxPQUFPLEVBQXVCLG9CQUFvQixFQUFFLE1BQU0sdUNBQXVDLENBQUM7O0FBRWxHOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBNEJHO0FBS0gsTUFBTSxPQUFPLGdCQUFnQjtJQUo3QjtRQUtVLGtCQUFhLEdBQUcsTUFBTSxDQUFDLG9CQUFvQixDQUFDLENBQUM7S0F5QnREO0lBdkJDOzs7Ozs7Ozs7Ozs7Ozs7T0FlRztJQUNILFNBQVMsQ0FBQyxLQUFhLEVBQUUsTUFBNEI7UUFDbkQsSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ1gsT0FBTyxFQUFFLENBQUM7UUFDWixDQUFDO1FBRUQsT0FBTyxJQUFJLENBQUMsYUFBYSxDQUFDLFlBQVksQ0FBQyxLQUFLLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDeEQsQ0FBQzsrR0F6QlUsZ0JBQWdCOzZHQUFoQixnQkFBZ0I7OzRGQUFoQixnQkFBZ0I7a0JBSjVCLElBQUk7bUJBQUM7b0JBQ0osSUFBSSxFQUFFLGNBQWM7b0JBQ3BCLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IFBpcGUsIFBpcGVUcmFuc2Zvcm0sIGluamVjdCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgU2FmZUh0bWwgfSBmcm9tICdAYW5ndWxhci9wbGF0Zm9ybS1icm93c2VyJztcbmltcG9ydCB7IExpbmtQcm9jZXNzb3JDb25maWcsIExpbmtQcm9jZXNzb3JTZXJ2aWNlIH0gZnJvbSAnLi4vLi4vc2VydmljZXMvbGluay1wcm9jZXNzb3Iuc2VydmljZSc7XG5cbi8qKlxuICogUHJvY2Vzc0xpbmtzUGlwZSAtIFBpcGUgcGFyYSBwcm9jZXNhciB0ZXh0byB5IGNvbnZlcnRpciBVUkxzIGVuIGVubGFjZXMgY2xpY2tlYWJsZXMuXG4gKlxuICogRXN0ZSBwaXBlIHN0YW5kYWxvbmUgZGV0ZWN0YSBhdXRvbcOhdGljYW1lbnRlIFVSTHMgZXh0ZXJuYXMgKGh0dHAvaHR0cHMpIHkgcnV0YXMgaW50ZXJuYXNcbiAqIChxdWUgZW1waWV6YW4gY29uIC8pIHkgbGFzIGNvbnZpZXJ0ZSBlbiBlbGVtZW50b3MgSFRNTCBhbmNob3IgY29uIGxvcyBhdHJpYnV0b3MgYXByb3BpYWRvcy5cbiAqXG4gKiBAZXhhbXBsZSBVc28gYsOhc2ljbzpcbiAqIGBgYGh0bWxcbiAqIDxkaXYgW2lubmVySFRNTF09XCInVmlzaXQgaHR0cHM6Ly9leGFtcGxlLmNvbSBvciBnbyB0byAvcHJvZmlsZScgfCBwcm9jZXNzTGlua3NcIj48L2Rpdj5cbiAqIGBgYFxuICpcbiAqIEBleGFtcGxlIENvbiBjb25maWd1cmFjacOzbiBwZXJzb25hbGl6YWRhOlxuICogYGBgaHRtbFxuICogPGRpdiBbaW5uZXJIVE1MXT1cInRleHQgfCBwcm9jZXNzTGlua3M6bGlua0NvbmZpZ1wiPjwvZGl2PlxuICogYGBgXG4gKlxuICogQGV4YW1wbGUgRW4gVHlwZVNjcmlwdDpcbiAqIGBgYHR5cGVzY3JpcHRcbiAqIGV4cG9ydCBjbGFzcyBNeUNvbXBvbmVudCB7XG4gKiAgIGxpbmtDb25maWc6IExpbmtQcm9jZXNzb3JDb25maWcgPSB7XG4gKiAgICAgb3BlbkV4dGVybmFsSW5OZXdUYWI6IHRydWUsXG4gKiAgICAgb3BlbkludGVybmFsSW5OZXdUYWI6IGZhbHNlLFxuICogICAgIGxpbmtDbGFzczogJ215LWxpbmsnLFxuICogICAgIGV4dGVybmFsTGlua0NsYXNzOiAnZXh0ZXJuYWwnLFxuICogICAgIGludGVybmFsTGlua0NsYXNzOiAnaW50ZXJuYWwnXG4gKiAgIH07XG4gKiB9XG4gKiBgYGBcbiAqL1xuQFBpcGUoe1xuICBuYW1lOiAncHJvY2Vzc0xpbmtzJyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbn0pXG5leHBvcnQgY2xhc3MgUHJvY2Vzc0xpbmtzUGlwZSBpbXBsZW1lbnRzIFBpcGVUcmFuc2Zvcm0ge1xuICBwcml2YXRlIGxpbmtQcm9jZXNzb3IgPSBpbmplY3QoTGlua1Byb2Nlc3NvclNlcnZpY2UpO1xuXG4gIC8qKlxuICAgKiBUcmFuc2Zvcm1hIHRleHRvIHByb2Nlc2FuZG8gVVJMcyB5IHJ1dGFzIGludGVybmFzIHBhcmEgY29udmVydGlybGFzIGVuIGVubGFjZXMuXG4gICAqXG4gICAqIEBwYXJhbSB2YWx1ZSAtIEVsIHRleHRvIGEgcHJvY2VzYXJcbiAgICogQHBhcmFtIGNvbmZpZyAtIENvbmZpZ3VyYWNpw7NuIG9wY2lvbmFsIHBhcmEgZWwgcHJvY2VzYW1pZW50byBkZSBlbmxhY2VzXG4gICAqIEByZXR1cm5zIFNhZmVIdG1sIGNvbiBlbmxhY2VzIHByb2Nlc2Fkb3MgbyBzdHJpbmcgb3JpZ2luYWwgc2kgbm8gaGF5IGVubGFjZXNcbiAgICpcbiAgICogQGV4YW1wbGVcbiAgICogYGBgaHRtbFxuICAgKiA8IS0tIFVzbyBiw6FzaWNvIC0tPlxuICAgKiA8cCBbaW5uZXJIVE1MXT1cIidDaGVjayBvdXQgaHR0cHM6Ly9hbmd1bGFyLmlvJyB8IHByb2Nlc3NMaW5rc1wiPjwvcD5cbiAgICpcbiAgICogPCEtLSBDb24gY29uZmlndXJhY2nDs24gLS0+XG4gICAqIDxwIFtpbm5lckhUTUxdPVwibWVzc2FnZSB8IHByb2Nlc3NMaW5rczp7IG9wZW5FeHRlcm5hbEluTmV3VGFiOiBmYWxzZSB9XCI+PC9wPlxuICAgKiBgYGBcbiAgICovXG4gIHRyYW5zZm9ybSh2YWx1ZTogc3RyaW5nLCBjb25maWc/OiBMaW5rUHJvY2Vzc29yQ29uZmlnKTogU2FmZUh0bWwgfCBzdHJpbmcge1xuICAgIGlmICghdmFsdWUpIHtcbiAgICAgIHJldHVybiAnJztcbiAgICB9XG5cbiAgICByZXR1cm4gdGhpcy5saW5rUHJvY2Vzc29yLnByb2Nlc3NMaW5rcyh2YWx1ZSwgY29uZmlnKTtcbiAgfVxufVxuIl19
|