Compare commits
	
		
			18 Commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 70c29c778c | |||
| 0fc302699e | |||
| dcb7ca2df3 | |||
| ccbb0415e4 | |||
| 496f54cedd | |||
| 83b5ecebeb | |||
| 53b5cbed07 | |||
| 352fe79791 | |||
| a95d5a96a0 | |||
| ece7bb9a94 | |||
| d42859b7b2 | |||
| f5655ad20b | |||
| d3463f009b | |||
| bb883ce341 | |||
| d9703d3ce3 | |||
| 7b5ba74d8b | |||
| a61f57db13 | |||
| c33ad2e405 | 
							
								
								
									
										20
									
								
								changelog.md
									
									
									
									
									
								
							
							
						
						
									
										20
									
								
								changelog.md
									
									
									
									
									
								
							| @@ -1,5 +1,25 @@ | ||||
| # Changelog | ||||
|  | ||||
| ## 2025-10-23 - 1.12.6 - fix(dependencies) | ||||
| Bump FontAwesome to ^7.1.0 and add local claude settings | ||||
|  | ||||
| - Updated @fortawesome packages (@fortawesome/fontawesome-svg-core, @fortawesome/free-brands-svg-icons, @fortawesome/free-regular-svg-icons, @fortawesome/free-solid-svg-icons) to ^7.1.0 in package.json | ||||
| - Added .claude/settings.local.json to configure local Claude/tooling permissions for repository operations | ||||
|  | ||||
| ## 2025-09-23 - 1.12.5 - fix(ci) | ||||
| Add local permissions settings for development | ||||
|  | ||||
| - Adds a new local settings file: .claude/settings.local.json | ||||
| - Provides explicit permission entries for development tasks (allow running pnpm scripts, reading files, searching/replacing patterns, activating project, and helper tooling) | ||||
| - Intended for local dev environment to enable tool automation without changing repository code | ||||
|  | ||||
| ## 2025-09-20 - 1.12.4 - fix(ci) | ||||
| Add local assistant settings to enable permitted dev tooling commands | ||||
|  | ||||
| - Add a local assistant settings file to configure allowed development tooling commands. | ||||
| - Allows running pnpm scripts, file read/search/replace operations and other local project helper actions. | ||||
| - Local configuration only — does not change library code or public API. | ||||
|  | ||||
| ## 2025-09-19 - 1.12.3 - fix(dees-input-fileupload) | ||||
| Show selected files inside dropzone and improve file upload UX | ||||
|  | ||||
|   | ||||
							
								
								
									
										11
									
								
								package.json
									
									
									
									
									
								
							
							
						
						
									
										11
									
								
								package.json
									
									
									
									
									
								
							| @@ -1,6 +1,6 @@ | ||||
| { | ||||
|   "name": "@design.estate/dees-catalog", | ||||
|   "version": "1.12.3", | ||||
|   "version": "1.12.6", | ||||
|   "private": false, | ||||
|   "description": "A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.", | ||||
|   "main": "dist_ts_web/index.js", | ||||
| @@ -19,10 +19,10 @@ | ||||
|     "@design.estate/dees-domtools": "^2.3.3", | ||||
|     "@design.estate/dees-element": "^2.1.2", | ||||
|     "@design.estate/dees-wcctools": "^1.2.0", | ||||
|     "@fortawesome/fontawesome-svg-core": "^7.0.1", | ||||
|     "@fortawesome/free-brands-svg-icons": "^7.0.1", | ||||
|     "@fortawesome/free-regular-svg-icons": "^7.0.1", | ||||
|     "@fortawesome/free-solid-svg-icons": "^7.0.1", | ||||
|     "@fortawesome/fontawesome-svg-core": "^7.1.0", | ||||
|     "@fortawesome/free-brands-svg-icons": "^7.1.0", | ||||
|     "@fortawesome/free-regular-svg-icons": "^7.1.0", | ||||
|     "@fortawesome/free-solid-svg-icons": "^7.1.0", | ||||
|     "@push.rocks/smarti18n": "^1.0.4", | ||||
|     "@push.rocks/smartpromise": "^4.2.0", | ||||
|     "@push.rocks/smartstring": "^4.1.0", | ||||
| @@ -37,6 +37,7 @@ | ||||
|     "apexcharts": "^5.3.5", | ||||
|     "highlight.js": "11.11.1", | ||||
|     "ibantools": "^4.5.1", | ||||
|     "lit": "^3.3.1", | ||||
|     "lucide": "^0.544.0", | ||||
|     "monaco-editor": "0.52.2", | ||||
|     "pdfjs-dist": "^4.10.38", | ||||
|   | ||||
							
								
								
									
										95
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										95
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							| @@ -18,17 +18,17 @@ importers: | ||||
|         specifier: ^1.2.0 | ||||
|         version: 1.2.0 | ||||
|       '@fortawesome/fontawesome-svg-core': | ||||
|         specifier: ^7.0.1 | ||||
|         version: 7.0.1 | ||||
|         specifier: ^7.1.0 | ||||
|         version: 7.1.0 | ||||
|       '@fortawesome/free-brands-svg-icons': | ||||
|         specifier: ^7.0.1 | ||||
|         version: 7.0.1 | ||||
|         specifier: ^7.1.0 | ||||
|         version: 7.1.0 | ||||
|       '@fortawesome/free-regular-svg-icons': | ||||
|         specifier: ^7.0.1 | ||||
|         version: 7.0.1 | ||||
|         specifier: ^7.1.0 | ||||
|         version: 7.1.0 | ||||
|       '@fortawesome/free-solid-svg-icons': | ||||
|         specifier: ^7.0.1 | ||||
|         version: 7.0.1 | ||||
|         specifier: ^7.1.0 | ||||
|         version: 7.1.0 | ||||
|       '@push.rocks/smarti18n': | ||||
|         specifier: ^1.0.4 | ||||
|         version: 1.0.4 | ||||
| @@ -71,6 +71,9 @@ importers: | ||||
|       ibantools: | ||||
|         specifier: ^4.5.1 | ||||
|         version: 4.5.1 | ||||
|       lit: | ||||
|         specifier: ^3.3.1 | ||||
|         version: 3.3.1 | ||||
|       lucide: | ||||
|         specifier: ^0.544.0 | ||||
|         version: 0.544.0 | ||||
| @@ -771,24 +774,24 @@ packages: | ||||
|   '@esm-bundle/chai@4.3.4-fix.0': | ||||
|     resolution: {integrity: sha512-26SKdM4uvDWlY8/OOOxSB1AqQWeBosCX3wRYUZO7enTAj03CtVxIiCimYVG2WpULcyV51qapK4qTovwkUr5Mlw==} | ||||
|  | ||||
|   '@fortawesome/fontawesome-common-types@7.0.1': | ||||
|     resolution: {integrity: sha512-0VpNtO5cNe1/HQWMkl4OdncYK/mv9hnBte0Ew0n6DMzmo3Q3WzDFABHm6LeNTipt5zAyhQ6Ugjiu8aLaEjh1gg==} | ||||
|   '@fortawesome/fontawesome-common-types@7.1.0': | ||||
|     resolution: {integrity: sha512-l/BQM7fYntsCI//du+6sEnHOP6a74UixFyOYUyz2DLMXKx+6DEhfR3F2NYGE45XH1JJuIamacb4IZs9S0ZOWLA==} | ||||
|     engines: {node: '>=6'} | ||||
|  | ||||
|   '@fortawesome/fontawesome-svg-core@7.0.1': | ||||
|     resolution: {integrity: sha512-x0cR55ILVqFpUioSMf6ebpRCMXMcheGN743P05W2RB5uCNpJUqWIqW66Lap8PfL/lngvjTbZj0BNSUweIr/fHQ==} | ||||
|   '@fortawesome/fontawesome-svg-core@7.1.0': | ||||
|     resolution: {integrity: sha512-fNxRUk1KhjSbnbuBxlWSnBLKLBNun52ZBTcs22H/xEEzM6Ap81ZFTQ4bZBxVQGQgVY0xugKGoRcCbaKjLQ3XZA==} | ||||
|     engines: {node: '>=6'} | ||||
|  | ||||
|   '@fortawesome/free-brands-svg-icons@7.0.1': | ||||
|     resolution: {integrity: sha512-6xPmn5SrND/GM0+W33E77x05+aDn6RpR02eWd8eLdN0IxY0vXa5yU/ugaAKloOVxiG9w2330TSRsbJYL6c57Ow==} | ||||
|   '@fortawesome/free-brands-svg-icons@7.1.0': | ||||
|     resolution: {integrity: sha512-9byUd9bgNfthsZAjBl6GxOu1VPHgBuRUP9juI7ZoM98h8xNPTCTagfwUFyYscdZq4Hr7gD1azMfM9s5tIWKZZA==} | ||||
|     engines: {node: '>=6'} | ||||
|  | ||||
|   '@fortawesome/free-regular-svg-icons@7.0.1': | ||||
|     resolution: {integrity: sha512-4V9fHbHjcx9Qu4O99AM5B4zuEDfB4zajk1I77hEzOxPN00f8g3484Aeq6WpfFcmookvjLE3Pr71Dhf/lqw7tbA==} | ||||
|   '@fortawesome/free-regular-svg-icons@7.1.0': | ||||
|     resolution: {integrity: sha512-0e2fdEyB4AR+e6kU4yxwA/MonnYcw/CsMEP9lH82ORFi9svA6/RhDyhxIv5mlJaldmaHLLYVTb+3iEr+PDSZuQ==} | ||||
|     engines: {node: '>=6'} | ||||
|  | ||||
|   '@fortawesome/free-solid-svg-icons@7.0.1': | ||||
|     resolution: {integrity: sha512-esKuSrl1WMOTMDLNt38i16VfLe/gRZt2ZAJ3Yw7slfs7sj583MKqNFqO57zmhknk1Sya6f9Wys89aCzIJkcqlg==} | ||||
|   '@fortawesome/free-solid-svg-icons@7.1.0': | ||||
|     resolution: {integrity: sha512-Udu3K7SzAo9N013qt7qmm22/wo2hADdheXtBfxFTecp+ogsc0caQNRKEb7pkvvagUGOpG9wJC1ViH6WXs8oXIA==} | ||||
|     engines: {node: '>=6'} | ||||
|  | ||||
|   '@git.zone/tsbuild@2.6.8': | ||||
| @@ -853,15 +856,9 @@ packages: | ||||
|   '@leichtgewicht/ip-codec@2.0.5': | ||||
|     resolution: {integrity: sha512-Vo+PSpZG2/fmgmiNzYK9qWRh8h/CHrwD0mo1h1DzL4yzHNSfWYujGTYsWGreD000gcgmZ7K4Ys6Tx9TxtsKdDw==} | ||||
|  | ||||
|   '@lit-labs/ssr-dom-shim@1.3.0': | ||||
|     resolution: {integrity: sha512-nQIWonJ6eFAvUUrSlwyHDm/aE8PBDu5kRpL0vHMg6K8fK3Diq1xdPjTnsJSwxABhaZ+5eBi1btQB5ShUTKo4nQ==} | ||||
|  | ||||
|   '@lit-labs/ssr-dom-shim@1.4.0': | ||||
|     resolution: {integrity: sha512-ficsEARKnmmW5njugNYKipTm4SFnbik7CXtoencDZzmzo/dQ+2Q0bgkzJuoJP20Aj0F+izzJjOqsnkd6F/o1bw==} | ||||
|  | ||||
|   '@lit/reactive-element@2.1.0': | ||||
|     resolution: {integrity: sha512-L2qyoZSQClcBmq0qajBVbhYEcG6iK0XfLn66ifLe/RfC0/ihpc+pl0Wdn8bJ8o+hj38cG0fGXRgSS20MuXn7qA==} | ||||
|  | ||||
|   '@lit/reactive-element@2.1.1': | ||||
|     resolution: {integrity: sha512-N+dm5PAYdQ8e6UlywyyrgI2t++wFGXfHx+dSJ1oBrg6FAxUj40jId++EaRm80MKX5JnlH1sBsyZ5h0bcZKemCg==} | ||||
|  | ||||
| @@ -3850,9 +3847,6 @@ packages: | ||||
|   linkifyjs@4.3.1: | ||||
|     resolution: {integrity: sha512-DRSlB9DKVW04c4SUdGvKK5FR6be45lTU9M76JnngqPeeGDqPwYc0zdUErtsNVMtxPXgUWV4HbXbnC4sNyBxkYg==} | ||||
|  | ||||
|   lit-element@4.2.0: | ||||
|     resolution: {integrity: sha512-MGrXJVAI5x+Bfth/pU9Kst1iWID6GHDLEzFEnyULB/sFiRLgkd8NPK/PeeXxktA3T6EIIaq8U3KcbTU5XFcP2Q==} | ||||
|  | ||||
|   lit-element@4.2.1: | ||||
|     resolution: {integrity: sha512-WGAWRGzirAgyphK2urmYOV72tlvnxw7YfyLDgQ+OZnM9vQQBQnumQ7jUJe6unEzwGU3ahFOjuz1iz1jjrpCPuw==} | ||||
|  | ||||
| @@ -3862,9 +3856,6 @@ packages: | ||||
|   lit-html@3.3.1: | ||||
|     resolution: {integrity: sha512-S9hbyDu/vs1qNrithiNyeyv64c9yqiW9l+DBgI18fL+MTvOtWoFR0FWiyq1TxaYef5wNlpEmzlXoBlZEO+WjoA==} | ||||
|  | ||||
|   lit@3.3.0: | ||||
|     resolution: {integrity: sha512-DGVsqsOIHBww2DqnuZzW7QsuCdahp50ojuDaBPC7jUDRpYoH0z7kHBBYZewRzer75FwtrkmkKk7iOAwSaWdBmw==} | ||||
|  | ||||
|   lit@3.3.1: | ||||
|     resolution: {integrity: sha512-Ksr/8L3PTapbdXJCk+EJVB78jDodUMaP54gD24W186zGRARvwrsPfS60wae/SSCTCNZVPd1chXqio1qHQmu4NA==} | ||||
|  | ||||
| @@ -6466,7 +6457,7 @@ snapshots: | ||||
|       '@push.rocks/websetup': 3.0.19 | ||||
|       '@push.rocks/webstore': 2.0.20 | ||||
|       lenis: 1.3.4 | ||||
|       lit: 3.3.0 | ||||
|       lit: 3.3.1 | ||||
|       sweet-scroll: 4.0.0 | ||||
|     transitivePeerDependencies: | ||||
|       - '@nuxt/kit' | ||||
| @@ -6671,23 +6662,23 @@ snapshots: | ||||
|     dependencies: | ||||
|       '@types/chai': 4.3.20 | ||||
|  | ||||
|   '@fortawesome/fontawesome-common-types@7.0.1': {} | ||||
|   '@fortawesome/fontawesome-common-types@7.1.0': {} | ||||
|  | ||||
|   '@fortawesome/fontawesome-svg-core@7.0.1': | ||||
|   '@fortawesome/fontawesome-svg-core@7.1.0': | ||||
|     dependencies: | ||||
|       '@fortawesome/fontawesome-common-types': 7.0.1 | ||||
|       '@fortawesome/fontawesome-common-types': 7.1.0 | ||||
|  | ||||
|   '@fortawesome/free-brands-svg-icons@7.0.1': | ||||
|   '@fortawesome/free-brands-svg-icons@7.1.0': | ||||
|     dependencies: | ||||
|       '@fortawesome/fontawesome-common-types': 7.0.1 | ||||
|       '@fortawesome/fontawesome-common-types': 7.1.0 | ||||
|  | ||||
|   '@fortawesome/free-regular-svg-icons@7.0.1': | ||||
|   '@fortawesome/free-regular-svg-icons@7.1.0': | ||||
|     dependencies: | ||||
|       '@fortawesome/fontawesome-common-types': 7.0.1 | ||||
|       '@fortawesome/fontawesome-common-types': 7.1.0 | ||||
|  | ||||
|   '@fortawesome/free-solid-svg-icons@7.0.1': | ||||
|   '@fortawesome/free-solid-svg-icons@7.1.0': | ||||
|     dependencies: | ||||
|       '@fortawesome/fontawesome-common-types': 7.0.1 | ||||
|       '@fortawesome/fontawesome-common-types': 7.1.0 | ||||
|  | ||||
|   '@git.zone/tsbuild@2.6.8': | ||||
|     dependencies: | ||||
| @@ -6810,10 +6801,8 @@ snapshots: | ||||
|     transitivePeerDependencies: | ||||
|       - '@nuxt/kit' | ||||
|       - '@swc/helpers' | ||||
|       - bufferutil | ||||
|       - react | ||||
|       - supports-color | ||||
|       - utf-8-validate | ||||
|       - vue | ||||
|  | ||||
|   '@hapi/bourne@3.0.0': {} | ||||
| @@ -6866,14 +6855,8 @@ snapshots: | ||||
|  | ||||
|   '@leichtgewicht/ip-codec@2.0.5': {} | ||||
|  | ||||
|   '@lit-labs/ssr-dom-shim@1.3.0': {} | ||||
|  | ||||
|   '@lit-labs/ssr-dom-shim@1.4.0': {} | ||||
|  | ||||
|   '@lit/reactive-element@2.1.0': | ||||
|     dependencies: | ||||
|       '@lit-labs/ssr-dom-shim': 1.3.0 | ||||
|  | ||||
|   '@lit/reactive-element@2.1.1': | ||||
|     dependencies: | ||||
|       '@lit-labs/ssr-dom-shim': 1.4.0 | ||||
| @@ -6994,7 +6977,7 @@ snapshots: | ||||
|   '@open-wc/scoped-elements@3.0.5': | ||||
|     dependencies: | ||||
|       '@open-wc/dedupe-mixin': 1.4.0 | ||||
|       lit: 3.3.0 | ||||
|       lit: 3.3.1 | ||||
|  | ||||
|   '@open-wc/semantic-dom-diff@0.20.1': | ||||
|     dependencies: | ||||
| @@ -7008,7 +6991,7 @@ snapshots: | ||||
|   '@open-wc/testing-helpers@3.0.1': | ||||
|     dependencies: | ||||
|       '@open-wc/scoped-elements': 3.0.5 | ||||
|       lit: 3.3.0 | ||||
|       lit: 3.3.1 | ||||
|       lit-html: 3.3.0 | ||||
|  | ||||
|   '@open-wc/testing@4.0.0': | ||||
| @@ -10989,12 +10972,6 @@ snapshots: | ||||
|  | ||||
|   linkifyjs@4.3.1: {} | ||||
|  | ||||
|   lit-element@4.2.0: | ||||
|     dependencies: | ||||
|       '@lit-labs/ssr-dom-shim': 1.3.0 | ||||
|       '@lit/reactive-element': 2.1.0 | ||||
|       lit-html: 3.3.0 | ||||
|  | ||||
|   lit-element@4.2.1: | ||||
|     dependencies: | ||||
|       '@lit-labs/ssr-dom-shim': 1.4.0 | ||||
| @@ -11009,12 +10986,6 @@ snapshots: | ||||
|     dependencies: | ||||
|       '@types/trusted-types': 2.0.7 | ||||
|  | ||||
|   lit@3.3.0: | ||||
|     dependencies: | ||||
|       '@lit/reactive-element': 2.1.0 | ||||
|       lit-element: 4.2.0 | ||||
|       lit-html: 3.3.0 | ||||
|  | ||||
|   lit@3.3.1: | ||||
|     dependencies: | ||||
|       '@lit/reactive-element': 2.1.1 | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								readme.plan.md
									
									
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								readme.plan.md
									
									
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							| @@ -3,6 +3,6 @@ | ||||
|  */ | ||||
| export const commitinfo = { | ||||
|   name: '@design.estate/dees-catalog', | ||||
|   version: '1.12.3', | ||||
|   version: '1.12.6', | ||||
|   description: 'A comprehensive library that provides dynamic web components for building sophisticated and modern web applications using JavaScript and TypeScript.' | ||||
| } | ||||
|   | ||||
| @@ -2,8 +2,8 @@ import { css, cssManager } from '@design.estate/dees-element'; | ||||
| import { DeesInputBase } from '../dees-input-base.js'; | ||||
|  | ||||
| export const fileuploadStyles = [ | ||||
|   ...DeesInputBase.baseStyles, | ||||
|   cssManager.defaultStyles, | ||||
|   ...DeesInputBase.baseStyles, | ||||
|   css` | ||||
|     :host { | ||||
|       position: relative; | ||||
|   | ||||
							
								
								
									
										537
									
								
								ts_web/elements/dees-pdf-preview/component.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										537
									
								
								ts_web/elements/dees-pdf-preview/component.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,537 @@ | ||||
| import { DeesElement, property, html, customElement, type TemplateResult } from '@design.estate/dees-element'; | ||||
| import { PdfManager } from '../dees-pdf-shared/PdfManager.js'; | ||||
| import { CanvasPool, type PooledCanvas } from '../dees-pdf-shared/CanvasPool.js'; | ||||
| import { PerformanceMonitor, throttle } from '../dees-pdf-shared/utils.js'; | ||||
| import { previewStyles } from './styles.js'; | ||||
| import { demo as demoFunc } from './demo.js'; | ||||
| import '../dees-icon.js'; | ||||
|  | ||||
| declare global { | ||||
|   interface HTMLElementTagNameMap { | ||||
|     'dees-pdf-preview': DeesPdfPreview; | ||||
|   } | ||||
| } | ||||
|  | ||||
| @customElement('dees-pdf-preview') | ||||
| export class DeesPdfPreview extends DeesElement { | ||||
|   public static demo = demoFunc; | ||||
|   public static styles = previewStyles; | ||||
|  | ||||
|   @property({ type: String }) | ||||
|   public pdfUrl: string = ''; | ||||
|  | ||||
|   @property({ type: Number }) | ||||
|   public currentPreviewPage: number = 1; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   public clickable: boolean = true; | ||||
|  | ||||
|   @property({ type: Number }) | ||||
|   private pageCount: number = 0; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   private loading: boolean = false; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   private rendered: boolean = false; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   private error: boolean = false; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   private isHovering: boolean = false; | ||||
|  | ||||
|   @property({ type: Boolean }) | ||||
|   private isA4Format: boolean = true; | ||||
|  | ||||
|   private renderPagesTask: Promise<void> | null = null; | ||||
|   private renderPagesQueued: boolean = false; | ||||
|  | ||||
|   private observer: IntersectionObserver; | ||||
|   private pdfDocument: any; | ||||
|   private canvases: PooledCanvas[] = []; | ||||
|   private resizeObserver?: ResizeObserver; | ||||
|   private previewContainer: HTMLElement | null = null; | ||||
|   private stackElement: HTMLElement | null = null; | ||||
|   private loadedPdfUrl: string | null = null; | ||||
|  | ||||
|   constructor() { | ||||
|     super(); | ||||
|   } | ||||
|  | ||||
|   public render(): TemplateResult { | ||||
|     return html` | ||||
|       <div | ||||
|         class="preview-container ${this.loading ? 'loading' : ''} ${this.error ? 'error' : ''} ${this.clickable ? 'clickable' : ''}" | ||||
|         @click=${this.handleClick} | ||||
|         @mouseenter=${this.handleMouseEnter} | ||||
|         @mouseleave=${this.handleMouseLeave} | ||||
|         @mousemove=${this.handleMouseMove} | ||||
|       > | ||||
|         ${this.loading ? html` | ||||
|           <div class="preview-loading"> | ||||
|             <div class="preview-spinner"></div> | ||||
|             <div class="preview-text">Loading preview...</div> | ||||
|           </div> | ||||
|         ` : ''} | ||||
|  | ||||
|         ${this.error ? html` | ||||
|           <div class="preview-error"> | ||||
|             <dees-icon icon="lucide:FileX"></dees-icon> | ||||
|             <div class="preview-text">Failed to load PDF</div> | ||||
|           </div> | ||||
|         ` : ''} | ||||
|  | ||||
|         ${!this.loading && !this.error ? html` | ||||
|           <div class="preview-stack ${!this.isA4Format ? 'non-a4' : ''}"> | ||||
|             <canvas | ||||
|               class="preview-canvas" | ||||
|               data-page="${this.currentPreviewPage}" | ||||
|             ></canvas> | ||||
|           </div> | ||||
|  | ||||
|           ${this.pageCount > 1 && this.isHovering ? html` | ||||
|             <div class="preview-page-indicator"> | ||||
|               Page ${this.currentPreviewPage} of ${this.pageCount} | ||||
|             </div> | ||||
|           ` : ''} | ||||
|  | ||||
|           ${this.pageCount > 0 && !this.isHovering ? html` | ||||
|             <div class="preview-info"> | ||||
|               <dees-icon icon="lucide:FileText"></dees-icon> | ||||
|               <span class="preview-pages">${this.pageCount} page${this.pageCount > 1 ? 's' : ''}</span> | ||||
|             </div> | ||||
|           ` : ''} | ||||
|  | ||||
|           ${this.clickable ? html` | ||||
|             <div class="preview-overlay"> | ||||
|               <dees-icon icon="lucide:Eye"></dees-icon> | ||||
|               <span>View PDF</span> | ||||
|             </div> | ||||
|           ` : ''} | ||||
|         ` : ''} | ||||
|       </div> | ||||
|     `; | ||||
|   } | ||||
|  | ||||
|   private handleMouseEnter() { | ||||
|     this.isHovering = true; | ||||
|   } | ||||
|  | ||||
|   private handleMouseLeave() { | ||||
|     this.isHovering = false; | ||||
|     // Reset to first page when not hovering | ||||
|     if (this.currentPreviewPage !== 1) { | ||||
|       this.currentPreviewPage = 1; | ||||
|       void this.scheduleRenderPages(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private handleMouseMove(e: MouseEvent) { | ||||
|     if (!this.isHovering || this.pageCount <= 1) return; | ||||
|  | ||||
|     const rect = this.getBoundingClientRect(); | ||||
|     const x = e.clientX - rect.left; | ||||
|     const width = rect.width; | ||||
|  | ||||
|     // Calculate which page to show based on horizontal position | ||||
|     const percentage = Math.max(0, Math.min(1, x / width)); | ||||
|     const newPage = Math.ceil(percentage * this.pageCount) || 1; | ||||
|  | ||||
|     if (newPage !== this.currentPreviewPage) { | ||||
|       this.currentPreviewPage = newPage; | ||||
|       void this.scheduleRenderPages(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public async connectedCallback() { | ||||
|     await super.connectedCallback(); | ||||
|     this.setupIntersectionObserver(); | ||||
|     await this.updateComplete; | ||||
|     this.cacheElements(); | ||||
|     this.setupResizeObserver(); | ||||
|   } | ||||
|  | ||||
|   public async disconnectedCallback() { | ||||
|     await super.disconnectedCallback(); | ||||
|     this.cleanup(); | ||||
|     if (this.observer) { | ||||
|       this.observer.disconnect(); | ||||
|     } | ||||
|     this.resizeObserver?.disconnect(); | ||||
|     this.resizeObserver = undefined; | ||||
|   } | ||||
|  | ||||
|   private setupIntersectionObserver() { | ||||
|     const options = { | ||||
|       root: null, | ||||
|       rootMargin: '200px', | ||||
|       threshold: 0.01, | ||||
|     }; | ||||
|  | ||||
|     this.observer = new IntersectionObserver( | ||||
|       throttle((entries) => { | ||||
|         for (const entry of entries) { | ||||
|           if (entry.isIntersecting && !this.rendered && this.pdfUrl) { | ||||
|             this.loadAndRenderPreview(); | ||||
|           } else if (!entry.isIntersecting && this.rendered) { | ||||
|             // Optional: Clear canvases when out of view for memory optimization | ||||
|             // this.clearCanvases(); | ||||
|           } | ||||
|         } | ||||
|       }, 100), | ||||
|       options | ||||
|     ); | ||||
|  | ||||
|     this.observer.observe(this); | ||||
|   } | ||||
|  | ||||
|   private async loadAndRenderPreview() { | ||||
|     if (this.rendered || this.loading) return; | ||||
|  | ||||
|     this.loading = true; | ||||
|     this.error = false; | ||||
|     PerformanceMonitor.mark(`preview-load-${this.pdfUrl}`); | ||||
|  | ||||
|     try { | ||||
|       this.pdfDocument = await PdfManager.loadDocument(this.pdfUrl); | ||||
|       this.pageCount = this.pdfDocument.numPages; | ||||
|       this.currentPreviewPage = 1; | ||||
|       this.loadedPdfUrl = this.pdfUrl; | ||||
|  | ||||
|       // Force an update to ensure the canvas element is in the DOM | ||||
|       this.loading = false; | ||||
|       await this.updateComplete; | ||||
|       this.cacheElements(); | ||||
|  | ||||
|       // Now render the first page | ||||
|       await this.scheduleRenderPages(); | ||||
|  | ||||
|       this.rendered = true; | ||||
|  | ||||
|       const duration = PerformanceMonitor.measure(`preview-render-${this.pdfUrl}`, `preview-load-${this.pdfUrl}`); | ||||
|       console.log(`PDF preview rendered in ${duration}ms`); | ||||
|     } catch (error) { | ||||
|       console.error('Failed to load PDF preview:', error); | ||||
|       this.error = true; | ||||
|       this.loading = false; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private scheduleRenderPages(): Promise<void> { | ||||
|     if (!this.pdfDocument) { | ||||
|       return Promise.resolve(); | ||||
|     } | ||||
|  | ||||
|     if (this.renderPagesTask) { | ||||
|       this.renderPagesQueued = true; | ||||
|       return this.renderPagesTask; | ||||
|     } | ||||
|  | ||||
|     this.renderPagesTask = (async () => { | ||||
|       try { | ||||
|         await this.performRenderPages(); | ||||
|       } catch (error) { | ||||
|         console.error('Failed to render PDF preview pages:', error); | ||||
|       } | ||||
|     })().finally(() => { | ||||
|       this.renderPagesTask = null; | ||||
|       if (this.renderPagesQueued) { | ||||
|         this.renderPagesQueued = false; | ||||
|         void this.scheduleRenderPages(); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     return this.renderPagesTask; | ||||
|   } | ||||
|  | ||||
|   private async performRenderPages() { | ||||
|     if (!this.pdfDocument) return; | ||||
|  | ||||
|     // Wait a frame to ensure DOM is ready | ||||
|     await new Promise(resolve => requestAnimationFrame(resolve)); | ||||
|  | ||||
|     const canvas = this.shadowRoot?.querySelector('.preview-canvas') as HTMLCanvasElement; | ||||
|     if (!canvas) { | ||||
|       console.warn('Preview canvas not found in DOM'); | ||||
|       return; | ||||
|     } | ||||
|  | ||||
|     // Release old canvases | ||||
|     this.clearCanvases(); | ||||
|  | ||||
|     this.cacheElements(); | ||||
|  | ||||
|     // Get available size for the preview | ||||
|     const { availableWidth, availableHeight } = this.getAvailableSize(); | ||||
|  | ||||
|     try { | ||||
|       // Get the page to render | ||||
|       const pageNum = this.currentPreviewPage; | ||||
|       const page = await this.pdfDocument.getPage(pageNum); | ||||
|  | ||||
|       // Calculate scale to fit within available area while keeping aspect ratio | ||||
|       // Use higher scale for sharper rendering | ||||
|       const initialViewport = page.getViewport({ scale: 1 }); | ||||
|  | ||||
|       // Check if this is standard paper format (A4 or US Letter) | ||||
|       const aspectRatio = initialViewport.height / initialViewport.width; | ||||
|  | ||||
|       // Common paper format ratios | ||||
|       const a4PortraitRatio = 1.414; // 297mm / 210mm | ||||
|       const a4LandscapeRatio = 0.707; // 210mm / 297mm | ||||
|       const letterPortraitRatio = 1.294; // 11" / 8.5" | ||||
|       const letterLandscapeRatio = 0.773; // 8.5" / 11" | ||||
|  | ||||
|       // Check for standard formats with 5% tolerance | ||||
|       const tolerance = 0.05; | ||||
|       const isA4Portrait = Math.abs(aspectRatio - a4PortraitRatio) < (a4PortraitRatio * tolerance); | ||||
|       const isA4Landscape = Math.abs(aspectRatio - a4LandscapeRatio) < (a4LandscapeRatio * tolerance); | ||||
|       const isLetterPortrait = Math.abs(aspectRatio - letterPortraitRatio) < (letterPortraitRatio * tolerance); | ||||
|       const isLetterLandscape = Math.abs(aspectRatio - letterLandscapeRatio) < (letterLandscapeRatio * tolerance); | ||||
|  | ||||
|       // Consider it standard format if it matches A4 or US Letter | ||||
|       this.isA4Format = isA4Portrait || isA4Landscape || isLetterPortrait || isLetterLandscape; | ||||
|  | ||||
|       // Debug logging | ||||
|       console.log(`PDF aspect ratio: ${aspectRatio.toFixed(3)}, standard format: ${this.isA4Format}`) | ||||
|  | ||||
|       // Adjust available size for non-A4 documents (account for padding) | ||||
|       const adjustedWidth = this.isA4Format ? availableWidth : availableWidth - 24; | ||||
|       const adjustedHeight = this.isA4Format ? availableHeight : availableHeight - 24; | ||||
|  | ||||
|       const scaleX = adjustedWidth > 0 ? adjustedWidth / initialViewport.width : 0; | ||||
|       const scaleY = adjustedHeight > 0 ? adjustedHeight / initialViewport.height : 0; | ||||
|       // Increase scale by 2x for sharper rendering, but limit to 3.0 max | ||||
|       const baseScale = Math.min(scaleX || 0.5, scaleY || scaleX || 0.5); | ||||
|       const renderScale = Math.min(baseScale * 2, 3.0); | ||||
|  | ||||
|       if (!Number.isFinite(renderScale) || renderScale <= 0) { | ||||
|         page.cleanup?.(); | ||||
|         return; | ||||
|       } | ||||
|  | ||||
|       const viewport = page.getViewport({ scale: renderScale }); | ||||
|  | ||||
|       // Acquire canvas from pool | ||||
|       const pooledCanvas = CanvasPool.acquire(viewport.width, viewport.height); | ||||
|       this.canvases.push(pooledCanvas); | ||||
|  | ||||
|       // Render to pooled canvas first | ||||
|       const renderContext = { | ||||
|         canvasContext: pooledCanvas.ctx, | ||||
|         viewport: viewport, | ||||
|       }; | ||||
|  | ||||
|       await page.render(renderContext).promise; | ||||
|  | ||||
|       // Transfer to display canvas | ||||
|       // Set actual canvas resolution for sharpness | ||||
|       canvas.width = viewport.width; | ||||
|       canvas.height = viewport.height; | ||||
|  | ||||
|       // Scale down display size to fit the container while keeping high resolution | ||||
|       // For A4, fill the container; for non-A4, respect padding | ||||
|       const displayWidth = adjustedWidth; | ||||
|       const displayHeight = (viewport.height / viewport.width) * adjustedWidth; | ||||
|  | ||||
|       // If it fits height-wise better, scale by height instead | ||||
|       if (displayHeight > adjustedHeight) { | ||||
|         const altDisplayHeight = adjustedHeight; | ||||
|         const altDisplayWidth = (viewport.width / viewport.height) * adjustedHeight; | ||||
|         canvas.style.width = `${altDisplayWidth}px`; | ||||
|         canvas.style.height = `${altDisplayHeight}px`; | ||||
|       } else { | ||||
|         canvas.style.width = `${displayWidth}px`; | ||||
|         canvas.style.height = `${displayHeight}px`; | ||||
|       } | ||||
|  | ||||
|       const ctx = canvas.getContext('2d'); | ||||
|       if (ctx) { | ||||
|         // Enable image smoothing for better quality | ||||
|         ctx.imageSmoothingEnabled = true; | ||||
|         ctx.imageSmoothingQuality = 'high'; | ||||
|         ctx.drawImage(pooledCanvas.canvas, 0, 0); | ||||
|       } | ||||
|  | ||||
|       // Release page to free memory | ||||
|       page.cleanup(); | ||||
|     } catch (error) { | ||||
|       console.error(`Failed to render page ${this.currentPreviewPage}:`, error); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private clearCanvases() { | ||||
|     // Release pooled canvases | ||||
|     for (const pooledCanvas of this.canvases) { | ||||
|       CanvasPool.release(pooledCanvas); | ||||
|     } | ||||
|     this.canvases = []; | ||||
|   } | ||||
|  | ||||
|   private cleanup() { | ||||
|     this.clearCanvases(); | ||||
|  | ||||
|     if (this.pdfDocument) { | ||||
|       PdfManager.releaseDocument(this.loadedPdfUrl ?? this.pdfUrl); | ||||
|       this.pdfDocument = null; | ||||
|     } | ||||
|  | ||||
|     this.renderPagesQueued = false; | ||||
|  | ||||
|     this.pageCount = 0; | ||||
|     this.currentPreviewPage = 1; | ||||
|     this.isHovering = false; | ||||
|     this.isA4Format = true; | ||||
|     this.previewContainer = null; | ||||
|     this.stackElement = null; | ||||
|     this.loadedPdfUrl = null; | ||||
|     this.rendered = false; | ||||
|     this.loading = false; | ||||
|     this.error = false; | ||||
|   } | ||||
|  | ||||
|   private handleClick() { | ||||
|     if (!this.clickable) return; | ||||
|  | ||||
|     // Dispatch custom event for parent to handle | ||||
|     this.dispatchEvent(new CustomEvent('pdf-preview-click', { | ||||
|       detail: { | ||||
|         pdfUrl: this.pdfUrl, | ||||
|         pageCount: this.pageCount, | ||||
|       }, | ||||
|       bubbles: true, | ||||
|       composed: true, | ||||
|     })); | ||||
|   } | ||||
|  | ||||
|   public async updated(changedProperties: Map<PropertyKey, unknown>) { | ||||
|     super.updated(changedProperties); | ||||
|  | ||||
|     if (changedProperties.has('pdfUrl') && this.pdfUrl) { | ||||
|       const previousUrl = changedProperties.get('pdfUrl') as string | undefined; | ||||
|       if (previousUrl) { | ||||
|         PdfManager.releaseDocument(previousUrl); | ||||
|       } | ||||
|       this.cleanup(); | ||||
|       this.rendered = false; | ||||
|       this.currentPreviewPage = 1; | ||||
|  | ||||
|       // Check if in viewport and render if so | ||||
|       if (this.observer) { | ||||
|         const rect = this.getBoundingClientRect(); | ||||
|         if (rect.top < window.innerHeight && rect.bottom > 0) { | ||||
|           this.loadAndRenderPreview(); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     if (changedProperties.has('currentPreviewPage') && this.rendered) { | ||||
|       await this.scheduleRenderPages(); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   /** | ||||
|    * Provide context menu items for right-click functionality | ||||
|    */ | ||||
|   public getContextMenuItems() { | ||||
|     const items: any[] = []; | ||||
|  | ||||
|     // If clickable, add option to view the PDF | ||||
|     if (this.clickable) { | ||||
|       items.push({ | ||||
|         name: 'View PDF', | ||||
|         iconName: 'lucide:Eye', | ||||
|         action: async () => { | ||||
|           this.handleClick(); | ||||
|         } | ||||
|       }); | ||||
|       items.push({ divider: true }); | ||||
|     } | ||||
|  | ||||
|     items.push( | ||||
|       { | ||||
|         name: 'Open PDF in New Tab', | ||||
|         iconName: 'lucide:ExternalLink', | ||||
|         action: async () => { | ||||
|           window.open(this.pdfUrl, '_blank'); | ||||
|         } | ||||
|       }, | ||||
|       { divider: true }, | ||||
|       { | ||||
|         name: 'Copy PDF URL', | ||||
|         iconName: 'lucide:Copy', | ||||
|         action: async () => { | ||||
|           await navigator.clipboard.writeText(this.pdfUrl); | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         name: 'Download PDF', | ||||
|         iconName: 'lucide:Download', | ||||
|         action: async () => { | ||||
|           const link = document.createElement('a'); | ||||
|           link.href = this.pdfUrl; | ||||
|           link.download = this.pdfUrl.split('/').pop() || 'document.pdf'; | ||||
|           link.click(); | ||||
|         } | ||||
|       } | ||||
|     ); | ||||
|  | ||||
|     // Add page count info as a disabled item | ||||
|     if (this.pageCount > 0) { | ||||
|       items.push( | ||||
|         { divider: true }, | ||||
|         { | ||||
|           name: `${this.pageCount} page${this.pageCount > 1 ? 's' : ''}`, | ||||
|           iconName: 'lucide:FileText', | ||||
|           disabled: true, | ||||
|           action: async () => {} | ||||
|         } | ||||
|       ); | ||||
|     } | ||||
|  | ||||
|     return items; | ||||
|   } | ||||
|  | ||||
|   private cacheElements() { | ||||
|     if (!this.previewContainer) { | ||||
|       this.previewContainer = this.shadowRoot?.querySelector('.preview-container') as HTMLElement; | ||||
|     } | ||||
|     if (!this.stackElement) { | ||||
|       this.stackElement = this.shadowRoot?.querySelector('.preview-stack') as HTMLElement; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   private setupResizeObserver() { | ||||
|     if (!this.previewContainer || this.resizeObserver) return; | ||||
|  | ||||
|     this.resizeObserver = new ResizeObserver(() => { | ||||
|       if (this.rendered && this.pdfDocument && !this.loading) { | ||||
|         void this.scheduleRenderPages(); | ||||
|       } | ||||
|     }); | ||||
|  | ||||
|     this.resizeObserver.observe(this); | ||||
|   } | ||||
|  | ||||
|   private getAvailableSize() { | ||||
|     if (!this.stackElement) { | ||||
|       // Try to get the stack element if it's not cached | ||||
|       this.stackElement = this.shadowRoot?.querySelector('.preview-stack') as HTMLElement; | ||||
|     } | ||||
|  | ||||
|     if (!this.stackElement) { | ||||
|       // Fallback to default size if element not found | ||||
|       return { | ||||
|         availableWidth: 200,  // Full container width | ||||
|         availableHeight: 260, // Full container height | ||||
|       }; | ||||
|     } | ||||
|  | ||||
|     const rect = this.stackElement.getBoundingClientRect(); | ||||
|     const availableWidth = Math.max(rect.width, 0) || 200; | ||||
|     const availableHeight = Math.max(rect.height, 0) || 260; | ||||
|  | ||||
|     return { availableWidth, availableHeight }; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										189
									
								
								ts_web/elements/dees-pdf-preview/demo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										189
									
								
								ts_web/elements/dees-pdf-preview/demo.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,189 @@ | ||||
| import { html } from '@design.estate/dees-element'; | ||||
|  | ||||
| export const demo = () => { | ||||
|   const samplePdfs = [ | ||||
|     'https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf', | ||||
|     'https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/web/compressed.tracemonkey-pldi-09.pdf', | ||||
|   ]; | ||||
|  | ||||
|   const generateGridItems = (count: number) => { | ||||
|     const items = []; | ||||
|     for (let i = 0; i < count; i++) { | ||||
|       const pdfUrl = samplePdfs[i % samplePdfs.length]; | ||||
|       items.push(html` | ||||
|         <dees-pdf-preview | ||||
|           pdfUrl="${pdfUrl}" | ||||
|           maxPages="3" | ||||
|           stackOffset="6" | ||||
|           clickable="true" | ||||
|           grid-mode | ||||
|           @pdf-preview-click=${(e: CustomEvent) => { | ||||
|             console.log('PDF Preview clicked:', e.detail); | ||||
|             alert(`PDF clicked: ${e.detail.pageCount} pages`); | ||||
|           }} | ||||
|         ></dees-pdf-preview> | ||||
|       `); | ||||
|     } | ||||
|     return items; | ||||
|   }; | ||||
|  | ||||
|   return html` | ||||
|     <style> | ||||
|       .demo-container { | ||||
|         padding: 40px; | ||||
|         background: #f5f5f5; | ||||
|       } | ||||
|  | ||||
|       .demo-section { | ||||
|         margin-bottom: 60px; | ||||
|       } | ||||
|  | ||||
|       h3 { | ||||
|         margin-bottom: 20px; | ||||
|         font-size: 18px; | ||||
|         font-weight: 600; | ||||
|       } | ||||
|  | ||||
|       .preview-grid { | ||||
|         display: grid; | ||||
|         grid-template-columns: repeat(auto-fill, minmax(200px, 1fr)); | ||||
|         gap: 24px; | ||||
|       } | ||||
|  | ||||
|       .preview-row { | ||||
|         display: flex; | ||||
|         gap: 24px; | ||||
|         align-items: center; | ||||
|         margin-bottom: 20px; | ||||
|       } | ||||
|  | ||||
|       .preview-label { | ||||
|         font-size: 14px; | ||||
|         font-weight: 500; | ||||
|         min-width: 100px; | ||||
|       } | ||||
|  | ||||
|       .performance-stats { | ||||
|         margin-top: 20px; | ||||
|         padding: 16px; | ||||
|         background: white; | ||||
|         border-radius: 8px; | ||||
|         font-size: 14px; | ||||
|       } | ||||
|  | ||||
|       .stats-grid { | ||||
|         display: grid; | ||||
|         grid-template-columns: repeat(auto-fill, minmax(150px, 1fr)); | ||||
|         gap: 12px; | ||||
|         margin-top: 12px; | ||||
|       } | ||||
|  | ||||
|       .stat-item { | ||||
|         display: flex; | ||||
|         flex-direction: column; | ||||
|         gap: 4px; | ||||
|       } | ||||
|  | ||||
|       .stat-label { | ||||
|         font-size: 12px; | ||||
|         color: #666; | ||||
|       } | ||||
|  | ||||
|       .stat-value { | ||||
|         font-size: 16px; | ||||
|         font-weight: 600; | ||||
|       } | ||||
|     </style> | ||||
|  | ||||
|     <div class="demo-container"> | ||||
|       <div class="demo-section"> | ||||
|         <h3>Single PDF Preview with Stacked Pages</h3> | ||||
|         <dees-pdf-preview | ||||
|           pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/web/compressed.tracemonkey-pldi-09.pdf" | ||||
|           maxPages="3" | ||||
|           stackOffset="8" | ||||
|           clickable="true" | ||||
|         ></dees-pdf-preview> | ||||
|       </div> | ||||
|  | ||||
|       <div class="demo-section"> | ||||
|         <h3>Different Sizes</h3> | ||||
|         <div class="preview-row"> | ||||
|           <div class="preview-label">Small:</div> | ||||
|           <dees-pdf-preview | ||||
|             size="small" | ||||
|             pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf" | ||||
|             maxPages="2" | ||||
|             stackOffset="6" | ||||
|             clickable="true" | ||||
|           ></dees-pdf-preview> | ||||
|         </div> | ||||
|  | ||||
|         <div class="preview-row"> | ||||
|           <div class="preview-label">Default:</div> | ||||
|           <dees-pdf-preview | ||||
|             pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf" | ||||
|             maxPages="3" | ||||
|             stackOffset="8" | ||||
|             clickable="true" | ||||
|           ></dees-pdf-preview> | ||||
|         </div> | ||||
|  | ||||
|         <div class="preview-row"> | ||||
|           <div class="preview-label">Large:</div> | ||||
|           <dees-pdf-preview | ||||
|             size="large" | ||||
|             pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf" | ||||
|             maxPages="4" | ||||
|             stackOffset="10" | ||||
|             clickable="true" | ||||
|           ></dees-pdf-preview> | ||||
|         </div> | ||||
|       </div> | ||||
|  | ||||
|       <div class="demo-section"> | ||||
|         <h3>Non-Clickable Preview</h3> | ||||
|         <dees-pdf-preview | ||||
|           pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf" | ||||
|           maxPages="3" | ||||
|           stackOffset="8" | ||||
|           clickable="false" | ||||
|         ></dees-pdf-preview> | ||||
|       </div> | ||||
|  | ||||
|       <div class="demo-section"> | ||||
|         <h3>Performance Grid - 50 PDFs with Lazy Loading</h3> | ||||
|         <p style="margin-bottom: 20px; font-size: 14px; color: #666;"> | ||||
|           This grid demonstrates the performance optimizations with 50 PDF previews. | ||||
|           Scroll to see lazy loading in action - previews render only when visible. | ||||
|         </p> | ||||
|  | ||||
|         <div class="preview-grid"> | ||||
|           ${generateGridItems(50)} | ||||
|         </div> | ||||
|  | ||||
|         <div class="performance-stats"> | ||||
|           <h4>Performance Features</h4> | ||||
|           <div class="stats-grid"> | ||||
|             <div class="stat-item"> | ||||
|               <span class="stat-label">Lazy Loading</span> | ||||
|               <span class="stat-value">✓ Enabled</span> | ||||
|             </div> | ||||
|             <div class="stat-item"> | ||||
|               <span class="stat-label">Canvas Pooling</span> | ||||
|               <span class="stat-value">✓ Active</span> | ||||
|             </div> | ||||
|             <div class="stat-item"> | ||||
|               <span class="stat-label">Memory Management</span> | ||||
|               <span class="stat-value">✓ Optimized</span> | ||||
|             </div> | ||||
|             <div class="stat-item"> | ||||
|               <span class="stat-label">Intersection Observer</span> | ||||
|               <span class="stat-value">200px margin</span> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   `; | ||||
| }; | ||||
							
								
								
									
										1
									
								
								ts_web/elements/dees-pdf-preview/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								ts_web/elements/dees-pdf-preview/index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| export * from './component.js'; | ||||
							
								
								
									
										223
									
								
								ts_web/elements/dees-pdf-preview/styles.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										223
									
								
								ts_web/elements/dees-pdf-preview/styles.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,223 @@ | ||||
| import { css, cssManager } from '@design.estate/dees-element'; | ||||
|  | ||||
| export const previewStyles = [ | ||||
|   cssManager.defaultStyles, | ||||
|   css` | ||||
|     :host { | ||||
|       display: inline-block; | ||||
|       position: relative; | ||||
|     } | ||||
|  | ||||
|     .preview-container { | ||||
|       position: relative; | ||||
|       width: 200px; | ||||
|       height: 260px; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 98%)', 'hsl(215 20% 14%)')}; | ||||
|       border-radius: 4px; | ||||
|       overflow: hidden; | ||||
|       transition: transform 0.2s ease, box-shadow 0.2s ease; | ||||
|       box-shadow: 0 1px 3px ${cssManager.bdTheme('rgba(0, 0, 0, 0.12)', 'rgba(0, 0, 0, 0.24)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-container.clickable { | ||||
|       cursor: pointer; | ||||
|     } | ||||
|  | ||||
|     .preview-container.clickable:hover { | ||||
|       transform: translateY(-2px); | ||||
|       box-shadow: 0 8px 24px ${cssManager.bdTheme('rgba(0, 0, 0, 0.12)', 'rgba(0, 0, 0, 0.3)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-container.clickable:hover .preview-overlay { | ||||
|       opacity: 1; | ||||
|     } | ||||
|  | ||||
|     .preview-stack { | ||||
|       position: relative; | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       box-sizing: border-box; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|  | ||||
|     .preview-stack.non-a4 { | ||||
|       padding: 12px; | ||||
|     } | ||||
|  | ||||
|     .preview-canvas { | ||||
|       position: relative; | ||||
|       background: white; | ||||
|       display: block; | ||||
|       max-width: 100%; | ||||
|       max-height: 100%; | ||||
|       width: auto; | ||||
|       height: auto; | ||||
|       object-fit: contain; | ||||
|       image-rendering: auto; | ||||
|       -webkit-font-smoothing: antialiased; | ||||
|       box-shadow: 0 1px 3px ${cssManager.bdTheme('rgba(0, 0, 0, 0.1)', 'rgba(0, 0, 0, 0.3)')}; | ||||
|     } | ||||
|  | ||||
|     .non-a4 .preview-canvas { | ||||
|       border: 1px solid ${cssManager.bdTheme('hsl(214 31% 92%)', 'hsl(217 25% 24%)')}; | ||||
|       border-radius: 4px; | ||||
|     } | ||||
|  | ||||
|     .preview-info { | ||||
|       position: absolute; | ||||
|       bottom: 8px; | ||||
|       left: 8px; | ||||
|       right: 8px; | ||||
|       padding: 6px 10px; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 100% / 0.92)', 'hsl(215 20% 12% / 0.92)')}; | ||||
|       border-radius: 6px; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 6px; | ||||
|       font-size: 12px; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|       backdrop-filter: blur(12px); | ||||
|       box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); | ||||
|       z-index: 10; | ||||
|     } | ||||
|  | ||||
|     .preview-info dees-icon { | ||||
|       font-size: 13px; | ||||
|       color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-pages { | ||||
|       font-weight: 500; | ||||
|       font-size: 11px; | ||||
|     } | ||||
|  | ||||
|     .preview-overlay { | ||||
|       position: absolute; | ||||
|       top: 0; | ||||
|       left: 0; | ||||
|       right: 0; | ||||
|       bottom: 0; | ||||
|       background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.7)', 'rgba(0, 0, 0, 0.8)')}; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       gap: 8px; | ||||
|       opacity: 0; | ||||
|       transition: opacity 0.2s ease; | ||||
|       z-index: 20; | ||||
|     } | ||||
|  | ||||
|     .preview-overlay dees-icon { | ||||
|       font-size: 24px; | ||||
|       color: white; | ||||
|     } | ||||
|  | ||||
|     .preview-overlay span { | ||||
|       font-size: 14px; | ||||
|       font-weight: 500; | ||||
|       color: white; | ||||
|     } | ||||
|  | ||||
|     .preview-loading, | ||||
|     .preview-error { | ||||
|       position: absolute; | ||||
|       top: 0; | ||||
|       left: 0; | ||||
|       right: 0; | ||||
|       bottom: 0; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       gap: 12px; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-loading { | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 99%)', 'hsl(215 20% 14%)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-error { | ||||
|       background: ${cssManager.bdTheme('hsl(0 72% 98%)', 'hsl(0 62% 20%)')}; | ||||
|       color: ${cssManager.bdTheme('hsl(0 72% 40%)', 'hsl(0 70% 68%)')}; | ||||
|     } | ||||
|  | ||||
|     .preview-spinner { | ||||
|       width: 24px; | ||||
|       height: 24px; | ||||
|       border-radius: 50%; | ||||
|       border: 2px solid ${cssManager.bdTheme('hsl(214 31% 86%)', 'hsl(217 25% 28%)')}; | ||||
|       border-top-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')}; | ||||
|       animation: spin 0.8s linear infinite; | ||||
|     } | ||||
|  | ||||
|     @keyframes spin { | ||||
|       to { | ||||
|         transform: rotate(360deg); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .preview-text { | ||||
|       font-size: 13px; | ||||
|       font-weight: 500; | ||||
|     } | ||||
|  | ||||
|     .preview-error dees-icon { | ||||
|       font-size: 32px; | ||||
|     } | ||||
|  | ||||
|     .preview-page-indicator { | ||||
|       position: absolute; | ||||
|       top: 8px; | ||||
|       left: 8px; | ||||
|       right: 8px; | ||||
|       padding: 5px 8px; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 0% / 0.7)', 'hsl(0 0% 100% / 0.9)')}; | ||||
|       color: ${cssManager.bdTheme('white', 'hsl(215 20% 12%)')}; | ||||
|       border-radius: 4px; | ||||
|       font-size: 11px; | ||||
|       font-weight: 600; | ||||
|       text-align: center; | ||||
|       backdrop-filter: blur(12px); | ||||
|       z-index: 15; | ||||
|       pointer-events: none; | ||||
|       animation: fadeIn 0.2s ease; | ||||
|     } | ||||
|  | ||||
|     @keyframes fadeIn { | ||||
|       from { | ||||
|         opacity: 0; | ||||
|         transform: translateY(-4px); | ||||
|       } | ||||
|       to { | ||||
|         opacity: 1; | ||||
|         transform: translateY(0); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     /* Responsive sizes */ | ||||
|     :host([size="small"]) .preview-container { | ||||
|       width: 150px; | ||||
|       height: 195px; | ||||
|     } | ||||
|  | ||||
|     :host([size="large"]) .preview-container { | ||||
|       width: 250px; | ||||
|       height: 325px; | ||||
|     } | ||||
|  | ||||
|     /* Grid optimizations */ | ||||
|     :host([grid-mode]) .preview-container { | ||||
|       will-change: auto; | ||||
|     } | ||||
|  | ||||
|     :host([grid-mode]) .preview-canvas { | ||||
|       image-rendering: -webkit-optimize-contrast; | ||||
|       image-rendering: crisp-edges; | ||||
|     } | ||||
|   `, | ||||
| ]; | ||||
							
								
								
									
										135
									
								
								ts_web/elements/dees-pdf-shared/CanvasPool.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										135
									
								
								ts_web/elements/dees-pdf-shared/CanvasPool.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,135 @@ | ||||
| export interface PooledCanvas { | ||||
|   canvas: HTMLCanvasElement; | ||||
|   ctx: CanvasRenderingContext2D; | ||||
|   inUse: boolean; | ||||
|   lastUsed: number; | ||||
| } | ||||
|  | ||||
| export class CanvasPool { | ||||
|   private static pool: PooledCanvas[] = []; | ||||
|   private static maxPoolSize = 20; | ||||
|   private static readonly MIN_CANVAS_SIZE = 256; | ||||
|   private static readonly MAX_CANVAS_SIZE = 4096; | ||||
|  | ||||
|   public static acquire(width: number, height: number): PooledCanvas { | ||||
|     // Try to find a suitable canvas from the pool | ||||
|     const suitable = this.pool.find( | ||||
|       (item) => !item.inUse && | ||||
|       item.canvas.width >= width && | ||||
|       item.canvas.height >= height && | ||||
|       item.canvas.width <= width * 1.5 && | ||||
|       item.canvas.height <= height * 1.5 | ||||
|     ); | ||||
|  | ||||
|     if (suitable) { | ||||
|       suitable.inUse = true; | ||||
|       suitable.lastUsed = Date.now(); | ||||
|  | ||||
|       // Clear and resize if needed | ||||
|       suitable.canvas.width = width; | ||||
|       suitable.canvas.height = height; | ||||
|       suitable.ctx.clearRect(0, 0, width, height); | ||||
|  | ||||
|       return suitable; | ||||
|     } | ||||
|  | ||||
|     // Create new canvas if pool not full | ||||
|     if (this.pool.length < this.maxPoolSize) { | ||||
|       const canvas = document.createElement('canvas'); | ||||
|       const ctx = canvas.getContext('2d', { | ||||
|         alpha: true, | ||||
|         desynchronized: true, | ||||
|       }) as CanvasRenderingContext2D; | ||||
|  | ||||
|       canvas.width = Math.min(Math.max(width, this.MIN_CANVAS_SIZE), this.MAX_CANVAS_SIZE); | ||||
|       canvas.height = Math.min(Math.max(height, this.MIN_CANVAS_SIZE), this.MAX_CANVAS_SIZE); | ||||
|  | ||||
|       const pooledCanvas: PooledCanvas = { | ||||
|         canvas, | ||||
|         ctx, | ||||
|         inUse: true, | ||||
|         lastUsed: Date.now(), | ||||
|       }; | ||||
|  | ||||
|       this.pool.push(pooledCanvas); | ||||
|       return pooledCanvas; | ||||
|     } | ||||
|  | ||||
|     // Evict and reuse least recently used canvas | ||||
|     const lru = this.pool | ||||
|       .filter((item) => !item.inUse) | ||||
|       .sort((a, b) => a.lastUsed - b.lastUsed)[0]; | ||||
|  | ||||
|     if (lru) { | ||||
|       lru.canvas.width = width; | ||||
|       lru.canvas.height = height; | ||||
|       lru.ctx.clearRect(0, 0, width, height); | ||||
|       lru.inUse = true; | ||||
|       lru.lastUsed = Date.now(); | ||||
|       return lru; | ||||
|     } | ||||
|  | ||||
|     // Fallback: create temporary canvas (shouldn't normally happen) | ||||
|     const canvas = document.createElement('canvas'); | ||||
|     const ctx = canvas.getContext('2d') as CanvasRenderingContext2D; | ||||
|     canvas.width = width; | ||||
|     canvas.height = height; | ||||
|  | ||||
|     return { | ||||
|       canvas, | ||||
|       ctx, | ||||
|       inUse: true, | ||||
|       lastUsed: Date.now(), | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   public static release(pooledCanvas: PooledCanvas) { | ||||
|     if (this.pool.includes(pooledCanvas)) { | ||||
|       pooledCanvas.inUse = false; | ||||
|       // Clear canvas to free memory | ||||
|       pooledCanvas.ctx.clearRect(0, 0, pooledCanvas.canvas.width, pooledCanvas.canvas.height); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public static releaseAll() { | ||||
|     for (const item of this.pool) { | ||||
|       item.inUse = false; | ||||
|       item.ctx.clearRect(0, 0, item.canvas.width, item.canvas.height); | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   public static destroy() { | ||||
|     for (const item of this.pool) { | ||||
|       item.canvas.width = 0; | ||||
|       item.canvas.height = 0; | ||||
|     } | ||||
|     this.pool = []; | ||||
|   } | ||||
|  | ||||
|   public static getStats() { | ||||
|     return { | ||||
|       poolSize: this.pool.length, | ||||
|       maxPoolSize: this.maxPoolSize, | ||||
|       inUse: this.pool.filter((item) => item.inUse).length, | ||||
|       available: this.pool.filter((item) => !item.inUse).length, | ||||
|     }; | ||||
|   } | ||||
|  | ||||
|   public static adjustPoolSize(newSize: number) { | ||||
|     if (newSize < this.pool.length) { | ||||
|       // Remove excess canvases | ||||
|       const toRemove = this.pool.length - newSize; | ||||
|       const removed = this.pool | ||||
|         .filter((item) => !item.inUse) | ||||
|         .slice(0, toRemove); | ||||
|  | ||||
|       for (const item of removed) { | ||||
|         const index = this.pool.indexOf(item); | ||||
|         if (index > -1) { | ||||
|           this.pool.splice(index, 1); | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     this.maxPoolSize = newSize; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										36
									
								
								ts_web/elements/dees-pdf-shared/PdfManager.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								ts_web/elements/dees-pdf-shared/PdfManager.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,36 @@ | ||||
| import { domtools } from '@design.estate/dees-element'; | ||||
|  | ||||
| export class PdfManager { | ||||
|   private static pdfjsLib: any; | ||||
|   private static initialized = false; | ||||
|  | ||||
|   public static async initialize() { | ||||
|     if (this.initialized) return; | ||||
|  | ||||
|     // @ts-ignore | ||||
|     this.pdfjsLib = await import('https://cdn.jsdelivr.net/npm/pdfjs-dist@4.0.379/+esm'); | ||||
|     this.pdfjsLib.GlobalWorkerOptions.workerSrc = 'https://cdn.jsdelivr.net/npm/pdfjs-dist@4.0.379/build/pdf.worker.mjs'; | ||||
|  | ||||
|     this.initialized = true; | ||||
|   } | ||||
|  | ||||
|   public static async loadDocument(url: string): Promise<any> { | ||||
|     await this.initialize(); | ||||
|  | ||||
|     // IMPORTANT: Disabled caching to ensure component isolation | ||||
|     // Each viewer instance gets its own document to prevent state sharing | ||||
|     // This fixes issues where multiple viewers interfere with each other | ||||
|     const loadingTask = this.pdfjsLib.getDocument(url); | ||||
|     const document = await loadingTask.promise; | ||||
|  | ||||
|     return document; | ||||
|   } | ||||
|  | ||||
|   public static releaseDocument(_url: string) { | ||||
|     // No-op since we're not caching documents anymore | ||||
|     // Each viewer manages its own document lifecycle | ||||
|   } | ||||
|  | ||||
|   // Cache methods removed to ensure component isolation | ||||
|   // Each viewer now manages its own document lifecycle | ||||
| } | ||||
							
								
								
									
										98
									
								
								ts_web/elements/dees-pdf-shared/utils.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										98
									
								
								ts_web/elements/dees-pdf-shared/utils.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,98 @@ | ||||
| export function debounce<T extends (...args: any[]) => any>( | ||||
|   func: T, | ||||
|   wait: number | ||||
| ): (...args: Parameters<T>) => void { | ||||
|   let timeout: number | undefined; | ||||
|  | ||||
|   return function executedFunction(...args: Parameters<T>) { | ||||
|     const later = () => { | ||||
|       clearTimeout(timeout); | ||||
|       func(...args); | ||||
|     }; | ||||
|  | ||||
|     clearTimeout(timeout); | ||||
|     timeout = window.setTimeout(later, wait); | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export function throttle<T extends (...args: any[]) => any>( | ||||
|   func: T, | ||||
|   limit: number | ||||
| ): (...args: Parameters<T>) => void { | ||||
|   let inThrottle: boolean; | ||||
|  | ||||
|   return function executedFunction(...args: Parameters<T>) { | ||||
|     if (!inThrottle) { | ||||
|       func.apply(this, args); | ||||
|       inThrottle = true; | ||||
|       setTimeout(() => inThrottle = false, limit); | ||||
|     } | ||||
|   }; | ||||
| } | ||||
|  | ||||
| export function formatFileSize(bytes: number): string { | ||||
|   if (bytes === 0) return '0 Bytes'; | ||||
|  | ||||
|   const k = 1024; | ||||
|   const sizes = ['Bytes', 'KB', 'MB', 'GB']; | ||||
|   const i = Math.floor(Math.log(bytes) / Math.log(k)); | ||||
|  | ||||
|   return Math.round(bytes / Math.pow(k, i) * 100) / 100 + ' ' + sizes[i]; | ||||
| } | ||||
|  | ||||
| export function isInViewport(element: Element, margin = 0): boolean { | ||||
|   const rect = element.getBoundingClientRect(); | ||||
|   return ( | ||||
|     rect.top >= -margin && | ||||
|     rect.left >= -margin && | ||||
|     rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) + margin && | ||||
|     rect.right <= (window.innerWidth || document.documentElement.clientWidth) + margin | ||||
|   ); | ||||
| } | ||||
|  | ||||
| export class PerformanceMonitor { | ||||
|   private static marks = new Map<string, number>(); | ||||
|   private static measures: Array<{ name: string; duration: number }> = []; | ||||
|  | ||||
|   public static mark(name: string) { | ||||
|     this.marks.set(name, performance.now()); | ||||
|   } | ||||
|  | ||||
|   public static measure(name: string, startMark: string) { | ||||
|     const start = this.marks.get(startMark); | ||||
|     if (start) { | ||||
|       const duration = performance.now() - start; | ||||
|       this.measures.push({ name, duration }); | ||||
|       this.marks.delete(startMark); | ||||
|       return duration; | ||||
|     } | ||||
|     return 0; | ||||
|   } | ||||
|  | ||||
|   public static getReport() { | ||||
|     const report = { | ||||
|       measures: [...this.measures], | ||||
|       averages: {} as Record<string, number>, | ||||
|     }; | ||||
|  | ||||
|     // Calculate averages for repeated measures | ||||
|     const grouped = new Map<string, number[]>(); | ||||
|     for (const measure of this.measures) { | ||||
|       if (!grouped.has(measure.name)) { | ||||
|         grouped.set(measure.name, []); | ||||
|       } | ||||
|       grouped.get(measure.name)!.push(measure.duration); | ||||
|     } | ||||
|  | ||||
|     for (const [name, durations] of grouped) { | ||||
|       report.averages[name] = durations.reduce((a, b) => a + b, 0) / durations.length; | ||||
|     } | ||||
|  | ||||
|     return report; | ||||
|   } | ||||
|  | ||||
|   public static clear() { | ||||
|     this.marks.clear(); | ||||
|     this.measures = []; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										1023
									
								
								ts_web/elements/dees-pdf-viewer/component.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1023
									
								
								ts_web/elements/dees-pdf-viewer/component.ts
									
									
									
									
									
										Normal file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										69
									
								
								ts_web/elements/dees-pdf-viewer/demo.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								ts_web/elements/dees-pdf-viewer/demo.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,69 @@ | ||||
| import { html } from '@design.estate/dees-element'; | ||||
|  | ||||
| export const demo = () => html` | ||||
|   <style> | ||||
|     .demo-container { | ||||
|       padding: 40px; | ||||
|       background: #f5f5f5; | ||||
|     } | ||||
|  | ||||
|     .demo-section { | ||||
|       margin-bottom: 40px; | ||||
|     } | ||||
|  | ||||
|     h3 { | ||||
|       margin-bottom: 20px; | ||||
|       font-size: 18px; | ||||
|       font-weight: 600; | ||||
|     } | ||||
|  | ||||
|     dees-pdf-viewer { | ||||
|       border: 1px solid #ddd; | ||||
|       border-radius: 8px; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|  | ||||
|     .viewer-tall { | ||||
|       height: 800px; | ||||
|     } | ||||
|  | ||||
|     .viewer-compact { | ||||
|       height: 500px; | ||||
|     } | ||||
|   </style> | ||||
|  | ||||
|   <div class="demo-container"> | ||||
|     <div class="demo-section"> | ||||
|       <h3>Full Featured PDF Viewer with Toolbar</h3> | ||||
|       <dees-pdf-viewer | ||||
|         class="viewer-tall" | ||||
|         pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/web/compressed.tracemonkey-pldi-09.pdf" | ||||
|         showToolbar="true" | ||||
|         showSidebar="false" | ||||
|         initialZoom="page-fit" | ||||
|       ></dees-pdf-viewer> | ||||
|     </div> | ||||
|  | ||||
|     <div class="demo-section"> | ||||
|       <h3>PDF Viewer with Sidebar Navigation</h3> | ||||
|       <dees-pdf-viewer | ||||
|         class="viewer-tall" | ||||
|         pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/web/compressed.tracemonkey-pldi-09.pdf" | ||||
|         showToolbar="true" | ||||
|         showSidebar="true" | ||||
|         initialZoom="page-width" | ||||
|       ></dees-pdf-viewer> | ||||
|     </div> | ||||
|  | ||||
|     <div class="demo-section"> | ||||
|       <h3>Compact Viewer without Controls</h3> | ||||
|       <dees-pdf-viewer | ||||
|         class="viewer-compact" | ||||
|         pdfUrl="https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf" | ||||
|         showToolbar="false" | ||||
|         showSidebar="false" | ||||
|         initialZoom="auto" | ||||
|       ></dees-pdf-viewer> | ||||
|     </div> | ||||
|   </div> | ||||
| `; | ||||
							
								
								
									
										1
									
								
								ts_web/elements/dees-pdf-viewer/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								ts_web/elements/dees-pdf-viewer/index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| export * from './component.js'; | ||||
							
								
								
									
										291
									
								
								ts_web/elements/dees-pdf-viewer/styles.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										291
									
								
								ts_web/elements/dees-pdf-viewer/styles.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,291 @@ | ||||
| import { css, cssManager } from '@design.estate/dees-element'; | ||||
|  | ||||
| export const viewerStyles = [ | ||||
|   cssManager.defaultStyles, | ||||
|   css` | ||||
|     :host { | ||||
|       display: block; | ||||
|       width: 100%; | ||||
|       height: 600px; | ||||
|       position: relative; | ||||
|       font-family: 'Geist Sans', sans-serif; | ||||
|       contain: layout style; | ||||
|     } | ||||
|  | ||||
|     .pdf-viewer { | ||||
|       width: 100%; | ||||
|       height: 100%; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 97%)', 'hsl(215 20% 10%)')}; | ||||
|       position: relative; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|  | ||||
|     .toolbar { | ||||
|       height: 48px; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(215 20% 15%)')}; | ||||
|       border-bottom: 1px solid ${cssManager.bdTheme('hsl(214 31% 91%)', 'hsl(217 25% 22%)')}; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       padding: 0 16px; | ||||
|       gap: 16px; | ||||
|       flex-shrink: 0; | ||||
|     } | ||||
|  | ||||
|     .toolbar-group { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 4px; | ||||
|     } | ||||
|  | ||||
|     .toolbar-group--end { | ||||
|       margin-left: auto; | ||||
|     } | ||||
|  | ||||
|     .toolbar-button { | ||||
|       width: 32px; | ||||
|       height: 32px; | ||||
|       border-radius: 6px; | ||||
|       background: transparent; | ||||
|       border: none; | ||||
|       cursor: pointer; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       transition: background 0.15s ease; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|     } | ||||
|  | ||||
|     .toolbar-button:hover:not(:disabled) { | ||||
|       background: ${cssManager.bdTheme('hsl(214 31% 92%)', 'hsl(217 25% 22%)')}; | ||||
|     } | ||||
|  | ||||
|     .toolbar-button:disabled { | ||||
|       opacity: 0.4; | ||||
|       cursor: not-allowed; | ||||
|     } | ||||
|  | ||||
|     .toolbar-button dees-icon { | ||||
|       font-size: 16px; | ||||
|     } | ||||
|  | ||||
|     .page-info { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       gap: 8px; | ||||
|       padding: 0 8px; | ||||
|       font-size: 14px; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|     } | ||||
|  | ||||
|     .page-input { | ||||
|       width: 48px; | ||||
|       height: 28px; | ||||
|       border-radius: 4px; | ||||
|       border: 1px solid ${cssManager.bdTheme('hsl(214 31% 86%)', 'hsl(217 25% 28%)')}; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(215 20% 12%)')}; | ||||
|       color: ${cssManager.bdTheme('hsl(222 47% 11%)', 'hsl(210 20% 96%)')}; | ||||
|       text-align: center; | ||||
|       font-size: 14px; | ||||
|       font-family: inherit; | ||||
|       outline: none; | ||||
|     } | ||||
|  | ||||
|     .page-input:focus { | ||||
|       border-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')}; | ||||
|     } | ||||
|  | ||||
|     .page-separator { | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 60%)', 'hsl(215 16% 50%)')}; | ||||
|     } | ||||
|  | ||||
|     .zoom-level { | ||||
|       font-size: 13px; | ||||
|       font-weight: 500; | ||||
|       min-width: 48px; | ||||
|       text-align: center; | ||||
|     } | ||||
|  | ||||
|     .viewer-container { | ||||
|       flex: 1; | ||||
|       display: flex; | ||||
|       overflow: hidden; | ||||
|       position: relative; | ||||
|       min-height: 0; | ||||
|     } | ||||
|  | ||||
|     .sidebar { | ||||
|       width: 200px; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 100%)', 'hsl(215 20% 15%)')}; | ||||
|       border-right: 1px solid ${cssManager.bdTheme('hsl(214 31% 91%)', 'hsl(217 25% 22%)')}; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       height: 100%; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|  | ||||
|     .sidebar-header { | ||||
|       height: 40px; | ||||
|       padding: 0 12px; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       justify-content: space-between; | ||||
|       border-bottom: 1px solid ${cssManager.bdTheme('hsl(214 31% 91%)', 'hsl(217 25% 22%)')}; | ||||
|       font-size: 13px; | ||||
|       font-weight: 600; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|     } | ||||
|  | ||||
|     .sidebar-close { | ||||
|       width: 24px; | ||||
|       height: 24px; | ||||
|       border-radius: 4px; | ||||
|       background: transparent; | ||||
|       border: none; | ||||
|       cursor: pointer; | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|       transition: background 0.15s ease; | ||||
|     } | ||||
|  | ||||
|     .sidebar-close:hover { | ||||
|       background: ${cssManager.bdTheme('hsl(214 31% 92%)', 'hsl(217 25% 22%)')}; | ||||
|     } | ||||
|  | ||||
|     .sidebar-close dees-icon { | ||||
|       font-size: 14px; | ||||
|     } | ||||
|  | ||||
|     .sidebar-content { | ||||
|       flex: 1; | ||||
|       overflow-y: auto; | ||||
|       overflow-x: hidden; | ||||
|       padding: 12px; | ||||
|       display: block; | ||||
|       overscroll-behavior: contain; | ||||
|       min-height: 0; | ||||
|     } | ||||
|  | ||||
|     .thumbnail { | ||||
|       position: relative; | ||||
|       border-radius: 8px; | ||||
|       overflow: hidden; | ||||
|       cursor: pointer; | ||||
|       border: 2px solid transparent; | ||||
|       transition: border-color 0.15s ease; | ||||
|       background: ${cssManager.bdTheme('hsl(0 0% 95%)', 'hsl(215 20% 18%)')}; | ||||
|       display: block; | ||||
|       width: 100%; | ||||
|       margin-bottom: 12px; | ||||
|       /* Default A4 aspect ratio (297mm / 210mm ≈ 1.414) */ | ||||
|       min-height: calc(176px * 1.414); | ||||
|     } | ||||
|  | ||||
|     .thumbnail:last-child { | ||||
|       margin-bottom: 0; | ||||
|     } | ||||
|  | ||||
|     .thumbnail:hover { | ||||
|       border-color: ${cssManager.bdTheme('hsl(214 31% 86%)', 'hsl(217 25% 35%)')}; | ||||
|     } | ||||
|  | ||||
|     .thumbnail.active { | ||||
|       border-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')}; | ||||
|     } | ||||
|  | ||||
|     .thumbnail-canvas { | ||||
|       display: block; | ||||
|       width: 100%; | ||||
|       height: auto; | ||||
|       image-rendering: -webkit-optimize-contrast; | ||||
|       image-rendering: crisp-edges; | ||||
|     } | ||||
|  | ||||
|     .thumbnail-number { | ||||
|       position: absolute; | ||||
|       bottom: 4px; | ||||
|       right: 4px; | ||||
|       background: ${cssManager.bdTheme('rgba(0, 0, 0, 0.7)', 'rgba(0, 0, 0, 0.8)')}; | ||||
|       color: white; | ||||
|       font-size: 11px; | ||||
|       font-weight: 500; | ||||
|       padding: 2px 6px; | ||||
|       border-radius: 4px; | ||||
|     } | ||||
|  | ||||
|     .viewer-main { | ||||
|       flex: 1; | ||||
|       overflow-y: auto; | ||||
|       overflow-x: hidden; | ||||
|       padding: 20px; | ||||
|       scroll-behavior: smooth; | ||||
|       overscroll-behavior: contain; | ||||
|       min-height: 0; | ||||
|       position: relative; | ||||
|     } | ||||
|  | ||||
|     .loading-container { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       height: 100%; | ||||
|       gap: 16px; | ||||
|       color: ${cssManager.bdTheme('hsl(215 16% 45%)', 'hsl(215 16% 75%)')}; | ||||
|     } | ||||
|  | ||||
|     .loading-spinner { | ||||
|       width: 32px; | ||||
|       height: 32px; | ||||
|       border-radius: 50%; | ||||
|       border: 3px solid ${cssManager.bdTheme('hsl(214 31% 86%)', 'hsl(217 25% 28%)')}; | ||||
|       border-top-color: ${cssManager.bdTheme('hsl(217 91% 60%)', 'hsl(213 93% 68%)')}; | ||||
|       animation: spin 0.8s linear infinite; | ||||
|     } | ||||
|  | ||||
|     @keyframes spin { | ||||
|       to { | ||||
|         transform: rotate(360deg); | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .loading-text { | ||||
|       font-size: 14px; | ||||
|       font-weight: 500; | ||||
|     } | ||||
|  | ||||
|     .pages-container { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       align-items: center; | ||||
|       gap: 20px; | ||||
|     } | ||||
|  | ||||
|     .page-wrapper { | ||||
|       display: flex; | ||||
|       justify-content: center; | ||||
|       width: 100%; | ||||
|     } | ||||
|  | ||||
|     .canvas-container { | ||||
|       background: white; | ||||
|       box-shadow: 0 2px 12px ${cssManager.bdTheme('rgba(0, 0, 0, 0.1)', 'rgba(0, 0, 0, 0.3)')}; | ||||
|       border-radius: 4px; | ||||
|       overflow: hidden; | ||||
|       display: inline-block; | ||||
|     } | ||||
|  | ||||
|     .page-canvas { | ||||
|       display: block; | ||||
|       image-rendering: -webkit-optimize-contrast; | ||||
|       image-rendering: crisp-edges; | ||||
|     } | ||||
|  | ||||
|     .pdf-viewer.with-sidebar .viewer-main { | ||||
|       margin-left: 0; | ||||
|     } | ||||
|   `, | ||||
| ]; | ||||
| @@ -1,6 +1,8 @@ | ||||
| import { DeesElement, property, html, customElement, domtools, type TemplateResult, type CSSResult, } from '@design.estate/dees-element'; | ||||
| 
 | ||||
| import { Deferred } from '@push.rocks/smartpromise'; | ||||
| import { DeesContextmenu } from '../dees-contextmenu.js'; | ||||
| import '../dees-icon.js'; | ||||
| 
 | ||||
| // import type pdfjsTypes from 'pdfjs-dist';
 | ||||
| 
 | ||||
| @@ -10,6 +12,11 @@ declare global { | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| /** | ||||
|  * @deprecated Use DeesPdfViewer or DeesPdfPreview instead | ||||
|  * - DeesPdfViewer: Full-featured PDF viewing with controls, navigation, zoom | ||||
|  * - DeesPdfPreview: Lightweight, performance-optimized preview for grids | ||||
|  */ | ||||
| @customElement('dees-pdf') | ||||
| export class DeesPdf extends DeesElement { | ||||
|   // DEMO
 | ||||
| @@ -21,6 +28,8 @@ export class DeesPdf extends DeesElement { | ||||
|   public pdfUrl: string = | ||||
|     'https://raw.githubusercontent.com/mozilla/pdf.js/ba2edeae/examples/learning/helloworld.pdf'; | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|   constructor() { | ||||
|     super(); | ||||
| 
 | ||||
| @@ -44,9 +53,15 @@ export class DeesPdf extends DeesElement { | ||||
|         #pdfcanvas { | ||||
|           box-shadow: 0px 0px 5px #ccc; | ||||
|           width: 100%; | ||||
|           cursor: pointer; | ||||
|         } | ||||
|       </style> | ||||
|       <canvas id="pdfcanvas" .height=${0} .width=${0}></canvas> | ||||
|       <canvas | ||||
|         id="pdfcanvas" | ||||
|         .height=${0} | ||||
|         .width=${0} | ||||
| 
 | ||||
|       ></canvas> | ||||
|     `;
 | ||||
|   } | ||||
| 
 | ||||
| @@ -64,6 +79,8 @@ export class DeesPdf extends DeesElement { | ||||
|     } | ||||
|     await DeesPdf.pdfJsReady; | ||||
|     this.displayContent(); | ||||
| 
 | ||||
| 
 | ||||
|   } | ||||
| 
 | ||||
|   public async displayContent() { | ||||
| @@ -107,4 +124,37 @@ export class DeesPdf extends DeesElement { | ||||
|       } | ||||
|     ); | ||||
|   } | ||||
| } | ||||
| 
 | ||||
|   /** | ||||
|    * Provide context menu items for the global context menu handler | ||||
|    */ | ||||
|   public getContextMenuItems() { | ||||
|     return [ | ||||
|       { | ||||
|         name: 'Open PDF in New Tab', | ||||
|         iconName: 'lucide:ExternalLink', | ||||
|         action: async () => { | ||||
|           window.open(this.pdfUrl, '_blank'); | ||||
|         } | ||||
|       }, | ||||
|       { divider: true }, | ||||
|       { | ||||
|         name: 'Copy PDF URL', | ||||
|         iconName: 'lucide:Copy', | ||||
|         action: async () => { | ||||
|           await navigator.clipboard.writeText(this.pdfUrl); | ||||
|         } | ||||
|       }, | ||||
|       { | ||||
|         name: 'Download PDF', | ||||
|         iconName: 'lucide:Download', | ||||
|         action: async () => { | ||||
|           const link = document.createElement('a'); | ||||
|           link.href = this.pdfUrl; | ||||
|           link.download = this.pdfUrl.split('/').pop() || 'document.pdf'; | ||||
|           link.click(); | ||||
|         } | ||||
|       } | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
							
								
								
									
										1
									
								
								ts_web/elements/dees-pdf/index.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								ts_web/elements/dees-pdf/index.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | ||||
| export * from './component.js'; | ||||
| @@ -48,7 +48,9 @@ export * from './dees-mobilenavigation.js'; | ||||
| export * from './dees-modal.js'; | ||||
| export * from './dees-input-multitoggle.js'; | ||||
| export * from './dees-panel.js'; | ||||
| export * from './dees-pdf.js'; | ||||
| export * from './dees-pdf/index.js'; // @deprecated - Use dees-pdf-viewer or dees-pdf-preview instead | ||||
| export * from './dees-pdf-viewer/index.js'; | ||||
| export * from './dees-pdf-preview/index.js'; | ||||
| export * from './dees-searchbar.js'; | ||||
| export * from './dees-shopping-productcard.js'; | ||||
| export * from './dees-simple-appdash.js'; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user