File size: 23,557 Bytes
10b14be
 
4db4b63
 
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
 
 
4db4b63
10b14be
 
 
 
 
 
4db4b63
10b14be
4db4b63
10b14be
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
 
 
 
 
4db4b63
10b14be
4db4b63
10b14be
 
 
 
 
 
 
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
4db4b63
10b14be
 
 
 
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
4db4b63
10b14be
 
4db4b63
10b14be
 
 
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
 
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
4db4b63
10b14be
4db4b63
10b14be
4db4b63
10b14be
 
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
10b14be
 
4db4b63
10b14be
 
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
10b14be
 
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
10b14be
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
 
 
4db4b63
10b14be
 
 
 
 
 
 
 
 
 
 
4db4b63
10b14be
 
 
 
4db4b63
10b14be
 
 
3713bef
4db4b63
 
 
 
 
 
 
3713bef
4db4b63
 
3713bef
 
4db4b63
3713bef
4db4b63
 
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
 
 
669cc4d
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
4db4b63
 
3713bef
 
 
 
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
4db4b63
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
4db4b63
 
 
 
 
 
 
3713bef
4db4b63
3713bef
4db4b63
 
 
 
 
3713bef
4db4b63
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3713bef
10b14be
 
4db4b63
10b14be
89ce340
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
import api from '@/services'
import { debounce } from 'lodash'
// Using html2canvas instead of html-to-image for better compatibility
// import { toJpeg } from 'html-to-image'

class DataSyncService {
  private currentPPTId: string | null = null
  private saveTimeout: number | null = null
  private isOnline = true
  private autoSaveDelay = 300000 // Default 5 minutes, configurable
  private isInitialized = false
  private debouncedSave: any = null

  constructor() {
    this.setupNetworkMonitoring()
  }

  // Delayed initialization, called after Pinia is available
  async initialize() {
    if (this.isInitialized) return
    await this.setupAutoSave()
    this.isInitialized = true
  }

  // Set auto-save delay time (milliseconds)
  setAutoSaveDelay(delay: number) {
    this.autoSaveDelay = Math.max(500, delay) // Minimum 500ms
    if (this.isInitialized) {
      this.setupAutoSave() // Reset auto-save
    }
  }

  // Get current auto-save delay time
  getAutoSaveDelay(): number {
    return this.autoSaveDelay
  }

  // Set current PPT ID
  setCurrentPPTId(pptId: string) {
    this.currentPPTId = pptId
  }

  // Auto-save functionality
  private async setupAutoSave() {
    if (this.debouncedSave) {
      this.debouncedSave.cancel()
    }

    this.debouncedSave = debounce(async () => {
      await this.savePPT()
    }, this.autoSaveDelay) // Use configurable delay time

    // Listen for slides changes
    try {
      const { useSlidesStore } = await import('@/store')
      const slidesStore = useSlidesStore()
      slidesStore.$subscribe(() => {
        if (this.isOnline && this.currentPPTId) {
          this.debouncedSave()
        }
      })
    } 
    catch (error) {
      // console.warn('Unable to set up auto-save, store not ready:', error)
    }
  }

  // Network status monitoring
  private setupNetworkMonitoring() {
    window.addEventListener('online', () => {
      this.isOnline = true
      // console.log('Network connected, resuming auto-save')
    })

    window.addEventListener('offline', () => {
      this.isOnline = false
      // console.log('Network disconnected, pausing auto-save')
    })
  }

  // Save PPT to backend
  async savePPT(force = false): Promise<boolean> {
    // Dynamically import store to avoid dependency issues during initialization
    const { useAuthStore, useSlidesStore } = await import('@/store')
    
    try {
      const authStore = useAuthStore()
      const slidesStore = useSlidesStore()

      if (!authStore.isLoggedIn) {
        // console.warn('User not logged in, unable to save')
        return false
      }

      // If no current PPT ID and forced save, create new PPT
      if (!this.currentPPTId && force) {
        try {
          const response = await api.createPPT(slidesStore.title || 'Untitled Presentation')
          this.currentPPTId = response.pptId
          
          // Update slides store data to match newly created PPT
          if (response.ppt) {
            slidesStore.setSlides(response.ppt.slides)
            slidesStore.setTitle(response.ppt.title)
            slidesStore.setTheme(response.ppt.theme)
          }
          
          // console.log('Created new PPT and saved successfully')
          return true
        } 
        catch (createError) {
          // console.error('Failed to create new PPT:', createError)
          return false
        }
      }

      if (!this.currentPPTId && !force) {
        // console.warn('No current PPT ID')
        return false
      }

      const pptData = {
        pptId: this.currentPPTId,
        title: slidesStore.title,
        slides: slidesStore.slides,
        theme: slidesStore.theme,
        // Add critical size information
        viewportSize: slidesStore.viewportSize,
        viewportRatio: slidesStore.viewportRatio
      }

      await api.savePPT(pptData)
      // console.log('PPT saved successfully')
      return true
    } 
    catch (error) {
      // console.error('Failed to save PPT:', error)
      return false
    }
  }

  // Create new PPT
  async createNewPPT(title: string): Promise<string | null> {
    const { useAuthStore } = await import('@/store')
    const authStore = useAuthStore()

    if (!authStore.isLoggedIn) {
      throw new Error('User not logged in')
    }

    try {
      const response = await api.createPPT(title)
      this.setCurrentPPTId(response.pptId)
      return response.pptId
    } 
    catch (error) {
      // console.error('Failed to create PPT:', error)
      throw error
    }
  }

  // Load PPT
  async loadPPT(pptId: string): Promise<boolean> {
    const { useAuthStore, useSlidesStore } = await import('@/store')
    const authStore = useAuthStore()
    const slidesStore = useSlidesStore()
    
    if (!authStore.isLoggedIn) {
      throw new Error('User not logged in')
    }
    
    try {
      // console.log(`πŸ” Starting to load PPT: ${pptId}`)
      const pptData = await api.getPPT(pptId)
      
      if (!pptData) {
        throw new Error('PPT data is empty')
      }
      
      // console.log('πŸ“‹ PPT data loaded successfully, updating store:', {...})
      
      // Fix: Ensure complete PPT data is loaded and set in correct order
      
      // 1. First set viewport information to ensure correct canvas size
      if (pptData.viewportSize && pptData.viewportSize > 0) {
        slidesStore.setViewportSize(pptData.viewportSize)
        // console.log('βœ… Viewport size set successfully:', pptData.viewportSize)
      }
      else {
        // console.log('⚠️ Using default viewport size: 1000')
        slidesStore.setViewportSize(1000)
      }
      
      if (pptData.viewportRatio && pptData.viewportRatio > 0) {
        slidesStore.setViewportRatio(pptData.viewportRatio)
        // console.log('βœ… Viewport ratio set successfully:', pptData.viewportRatio)
      }
      else {
        // console.log('⚠️ Using default viewport ratio: 0.5625')
        slidesStore.setViewportRatio(0.5625)
      }
      
      // 2. Set theme
      if (pptData.theme) {
        slidesStore.setTheme(pptData.theme)
        // console.log('βœ… Theme set successfully')
      }
      
      // 3. Set title
      slidesStore.setTitle(pptData.title || 'Untitled Presentation')
      // console.log('βœ… Title set successfully:', pptData.title)
      
      // 4. Finally set slides, ensuring it is done after viewport information is set
      if (Array.isArray(pptData.slides) && pptData.slides.length > 0) {
        // Validate slides data integrity
        const validSlides = pptData.slides.filter((slide: any) => 
          slide && typeof slide === 'object' && Array.isArray(slide.elements)
        )
        
        if (validSlides.length !== pptData.slides.length) {
          // console.warn(`⚠️ Found ${pptData.slides.length - validSlides.length} invalid slides, filtered out`)
        }
        
        slidesStore.setSlides(validSlides)
        // console.log('βœ… Slides set successfully:', validSlides.length, 'pages')
      }
      else {
        // console.warn('⚠️ PPT has no valid slides, creating default slide')
        slidesStore.setSlides([{
          id: 'default-slide',
          elements: [],
          background: { type: 'solid', color: '#ffffff' }
        }])
      }
      
      // 5. Set current PPT ID
      const actualPptId = pptData.id || pptData.pptId || pptId
      this.setCurrentPPTId(actualPptId)
      // console.log('βœ… PPT ID set successfully:', actualPptId)
      
      // console.log('πŸŽ‰ PPT loaded successfully!')
      return true
    } 
    catch (error: any) {
      // console.error('❌ Failed to load PPT:', { pptId, error: error.message, stack: error.stack })
      
      // Provide more specific error information
      if (error.message?.includes('404') || error.message?.includes('not found')) {
        throw new Error(`PPT not found (ID: ${pptId})`)
      }
      if (error.message?.includes('403') || error.message?.includes('unauthorized')) {
        throw new Error('No access permission')
      }
      if (error.message?.includes('Network')) {
        throw new Error('Network connection failed, please check network status')
      }
      throw new Error(`Failed to load: ${error.message}`)
    }
  }

  // Get PPT list
  async getPPTList() {
    const { useAuthStore } = await import('@/store')
    const authStore = useAuthStore()

    if (!authStore.isLoggedIn) {
      throw new Error('User not logged in')
    }

    return await api.getPPTList()
  }

  // Delete PPT
  async deletePPT(pptId: string): Promise<boolean> {
    const { useAuthStore } = await import('@/store')
    const authStore = useAuthStore()

    if (!authStore.isLoggedIn) {
      throw new Error('User not logged in')
    }

    try {
      await api.deletePPT(pptId)
      
      // If the deleted PPT is the current PPT, clear the current PPT ID
      if (this.currentPPTId === pptId) {
        this.currentPPTId = null
      }
      
      // console.log('PPT deleted successfully')
      return true
    } 
    catch (error) {
      // console.error('Failed to delete PPT:', error)
      throw error
    }
  }

  // Generate share link
  async generateShareLink(slideIndex = 0) {
    const { useAuthStore } = await import('@/store')
    const authStore = useAuthStore()

    if (!authStore.isLoggedIn || !this.currentPPTId) {
      throw new Error('User not logged in or no current PPT')
    }

    try {
      const response = await api.generateShareLink(
        authStore.currentUser!.id,
        this.currentPPTId,
        slideIndex
      )
      return response
    } 
    catch (error) {
      // console.error('Failed to generate share link:', error)
      throw error
    }
  }

  // Manual save
  async manualSave(): Promise<boolean> {
    return await this.savePPT(true)
  }

  // New screenshot solution: use direct image API
  async generateScreenshotUrl(slideIndex = 0, options: any = {}) {
    const { useAuth = true, format = 'jpeg', quality = 90 } = options;
    
    if (useAuth) {
      const { useAuthStore } = await import('@/store')
      const authStore = useAuthStore()

      if (!authStore.isLoggedIn || !this.currentPPTId) {
        throw new Error('User not logged in or no current PPT')
      }

      const baseUrl = window.location.origin
      
      // πŸ”§ NEW: Use frontend screenshot data API for direct generation
      console.log('πŸš€ Using new frontend screenshot strategy...')
      
      try {
        // Get PPT data for frontend generation
        const dataResponse = await fetch(`${baseUrl}/api/public/screenshot-data/${authStore.currentUser!.id}/${this.currentPPTId}/${slideIndex}`)
        
        if (dataResponse.ok) {
          const screenshotData = await dataResponse.json()
          
          // Return data for frontend direct generation
          return {
            type: 'frontend-data',
            data: screenshotData,
            isDataUrl: false,
            info: {
              format,
              quality,
              strategy: 'frontend-direct-generation',
              pptTitle: screenshotData.pptData?.title,
              slideIndex: screenshotData.slideIndex,
              dimensions: `${screenshotData.exportConfig?.width}x${screenshotData.exportConfig?.height}`
            }
          }
        }
      } catch (error) {
        console.warn('Frontend data API failed, falling back to export page:', error)
      }
      
      // Fallback: Frontend export page (user opens in new tab)
      const frontendExportUrl = `${baseUrl}/api/public/image/${authStore.currentUser!.id}/${this.currentPPTId}/${slideIndex}?format=${format}&quality=${quality}`
      
      return {
        type: 'frontend-export-page',
        url: frontendExportUrl,
        isDataUrl: false,
        info: {
          format,
          quality,
          strategy: 'frontend-export-page',
          usage: 'Open this URL in a new tab to generate and download screenshot'
        }
      }
    } else {
      // Public access mode
      const baseUrl = window.location.origin
      return {
        type: 'frontend-export-page-public',
        url: `${baseUrl}/api/public/image/public/public/${slideIndex}?format=${format}&quality=${quality}`,
        isDataUrl: false
      }
    }
  }

  // Generate image and upload to backend for external link access
  async generateAndUploadImage(slideIndex = 0, options: any = {}) {
    const { format = 'jpeg', quality = 90, elementSelector = '.slide-viewport, .canvas-viewport, .viewport-wrapper' } = options;
    
    try {
      const { useAuthStore, useSlidesStore } = await import('@/store')
      const authStore = useAuthStore()
      const slidesStore = useSlidesStore()

      if (!authStore.isLoggedIn || !this.currentPPTId) {
        throw new Error('User not logged in or no current PPT')
      }

      // Find target element for screenshot
      let targetElement: HTMLElement | null = null
      const selectors = elementSelector.split(',').map((s: string) => s.trim())
      
      for (const selector of selectors) {
        targetElement = document.querySelector(selector) as HTMLElement
        if (targetElement) {
          console.log(`βœ… Found screenshot target element: ${selector}`)
          break
        }
      }

      if (!targetElement) {
        throw new Error(`Screenshot target element not found. Tried selectors: ${selectors.join(', ')}`)
      }

      // Dynamically load html2canvas
      const html2canvas = await this.loadHtml2Canvas()
      
      console.log('πŸ–ΌοΈ Starting to generate image for upload...', { format, quality })
      
      // Wait for fonts and images to load
      await this.ensureResourcesReady()
      
      // Generate screenshot
      const canvas = await html2canvas(targetElement, {
        scale: 2, // High resolution
        useCORS: true,
        allowTaint: true,
        backgroundColor: slidesStore.theme?.backgroundColor || '#ffffff',
        logging: false,
        onclone: function(clonedDoc) {
          // Ensure correct fonts in cloned document
          const clonedElements = clonedDoc.querySelectorAll('*')
          clonedElements.forEach((el: any) => {
            if (el.style) {
              el.style.fontFamily = 'Microsoft YaHei, PingFang SC, Hiragino Sans GB, Source Han Sans SC, Noto Sans SC, WenQuanYi Micro Hei, SimHei, SimSun, Arial, Helvetica, sans-serif'
            }
          })
        }
      })

      // Convert to specified format
      const imageDataUrl = canvas.toDataURL(`image/${format}`, quality / 100)
      
      console.log('βœ… Image generated, uploading to backend...')
      
      // Upload to backend
      const baseUrl = window.location.origin
      const uploadResponse = await fetch(`${baseUrl}/api/public/save-image/${authStore.currentUser!.id}/${this.currentPPTId}/${slideIndex}`, {
        method: 'POST',
        headers: {
          'Content-Type': 'application/json'
        },
        body: JSON.stringify({
          imageData: imageDataUrl,
          format,
          quality
        })
      })
      
      if (!uploadResponse.ok) {
        throw new Error(`Upload failed: ${uploadResponse.status} ${uploadResponse.statusText}`)
      }
      
      const uploadResult = await uploadResponse.json()
      
      console.log('βœ… Image uploaded successfully, external URL:', uploadResult.imageUrl)
      
      return {
        success: true,
        imageUrl: uploadResult.imageUrl,
        imageId: uploadResult.imageId,
        expiresAt: uploadResult.expiresAt,
        format,
        quality,
        width: canvas.width,
        height: canvas.height
      }
      
    } catch (error: any) {
      console.error('❌ Failed to generate and upload image:', error)
      throw error
    }
  }

  // Get PPT screenshot data (for frontend direct rendering)
  async getScreenshotData(slideIndex = 0) {
    const { useAuthStore } = await import('@/store')
    const authStore = useAuthStore()

    if (!authStore.isLoggedIn || !this.currentPPTId) {
      throw new Error('User not logged in or no current PPT')
    }

    try {
      const baseUrl = window.location.origin
      const response = await fetch(`${baseUrl}/api/public/screenshot-data/${authStore.currentUser!.id}/${this.currentPPTId}/${slideIndex}`)
      
      if (!response.ok) {
        throw new Error(`Failed to get screenshot data: ${response.status}`)
      }

      const data = await response.json()
      return data
    } catch (error: any) {
      console.error('❌ Failed to get screenshot data:', error)
      throw new Error(`Failed to get screenshot data: ${error.message}`)
    }
  }

  // Generate screenshot directly in current page (best solution)
  async generateDirectScreenshot(slideIndex = 0, options: any = {}) {
    try {
      const { useSlidesStore } = await import('@/store')
      const slidesStore = useSlidesStore()
      
      // Check if slide data is available
      if (!slidesStore.slides || slideIndex >= slidesStore.slides.length) {
        throw new Error('Invalid slide index or no slides available')
      }

      const {
        format = 'jpeg',
        quality = 90,
        width = slidesStore.viewportSize || 1000,
        height = Math.ceil((slidesStore.viewportSize || 1000) * (slidesStore.viewportRatio || 0.562)),
        elementSelector = '.canvas-main, .slide-container, .editor-main'
      } = options

      // Find screenshot target element
      let targetElement: HTMLElement | null = null
      const selectors = elementSelector.split(',').map((s: string) => s.trim())
      
      for (const selector of selectors) {
        targetElement = document.querySelector(selector) as HTMLElement
        if (targetElement) {
          console.log(`βœ… Found screenshot target element: ${selector}`)
          break
        }
      }

      if (!targetElement) {
        throw new Error(`Screenshot target element not found. Tried selectors: ${selectors.join(', ')}`)
      }

      // Dynamically load html2canvas
      const html2canvas = await this.loadHtml2Canvas()
      
      console.log('πŸ–ΌοΈ Starting to generate direct screenshot...', { width, height, quality })
      
      // Wait for fonts and images to load
      await this.ensureResourcesReady()
      
      // Generate screenshot
      const canvas = await html2canvas(targetElement, {
        width,
        height,
        scale: 2, // High resolution
        useCORS: true,
        allowTaint: true,
        backgroundColor: slidesStore.theme?.backgroundColor || '#ffffff',
        logging: false,
        onclone: function(clonedDoc) {
          // Ensure correct fonts in cloned document
          const clonedElements = clonedDoc.querySelectorAll('*')
          clonedElements.forEach((el: any) => {
            if (el.style) {
              el.style.fontFamily = 'Microsoft YaHei, PingFang SC, Hiragino Sans GB, Source Han Sans SC, Noto Sans SC, WenQuanYi Micro Hei, SimHei, SimSun, Arial, Helvetica, sans-serif'
            }
          })
        }
      })

      // Convert to specified format
      const imageDataUrl = canvas.toDataURL(`image/${format}`, quality / 100)
      
      console.log('βœ… Direct screenshot generated successfully, size:', imageDataUrl.length)
      
      return {
        type: 'direct-frontend',
        url: imageDataUrl,
        isDataUrl: true,
        width: canvas.width,
        height: canvas.height,
        format,
        quality
      }
      
    } catch (error: any) {
      console.error('❌ Failed to generate direct screenshot:', error)
      
      // Fallback to frontend export page solution
      console.log('πŸ”„ Falling back to frontend export page solution...')
      return await this.generateScreenshotUrl(slideIndex, options)
    }
  }

  // Dynamically load html2canvas library
  private async loadHtml2Canvas(): Promise<any> {
    // Check if already loaded
    if ((window as any).html2canvas) {
      return (window as any).html2canvas
    }

    return new Promise((resolve, reject) => {
      const script = document.createElement('script')
      script.src = 'https://cdn.jsdelivr.net/npm/[email protected]/dist/html2canvas.min.js'
      script.onload = () => {
        resolve((window as any).html2canvas)
      }
      script.onerror = (error) => {
        reject(new Error('Failed to load html2canvas library'))
      }
      document.head.appendChild(script)
    })
  }

  // Wait for resources to be ready
  private async ensureResourcesReady(): Promise<void> {
    // Wait for font loading
    if (document.fonts && document.fonts.ready) {
      await document.fonts.ready
    }

    // Wait for image loading
    const images = document.querySelectorAll('img')
    const imagePromises = Array.from(images).map(img => {
      if (img.complete) return Promise.resolve()
      
      return new Promise((resolve) => {
        img.onload = () => resolve(void 0)
        img.onerror = () => resolve(void 0) // Continue even if failed
        setTimeout(() => resolve(void 0), 2000) // 2 second timeout
      })
    })

    await Promise.all(imagePromises)
    
    // Additional wait to ensure rendering is complete
    await new Promise(resolve => setTimeout(resolve, 300))
  }

  // Simplified frontend screenshot generation (for compatibility)
  async generateFrontendScreenshot(elementId = 'slideContainer', options: any = {}) {
    console.warn('⚠️ generateFrontendScreenshot is deprecated, use generateDirectScreenshot instead')
    return await this.generateDirectScreenshot(0, { ...options, elementSelector: `#${elementId}` })
  }

  // Ensure element is fully rendered (for compatibility)
  ensureElementReady(element: HTMLElement) {
    console.warn('⚠️ ensureElementReady is deprecated, now using ensureResourcesReady')
    return this.ensureResourcesReady()
  }

  // Generate frontend fallback screenshot (for compatibility)
  generateFallbackScreenshot(width = 1000, height = 562) {
    const canvas = document.createElement('canvas')
    canvas.width = width
    canvas.height = height
    const ctx = canvas.getContext('2d')

    if (!ctx) return ''

    // Draw background
    const gradient = ctx.createLinearGradient(0, 0, width, height)
    gradient.addColorStop(0, '#f8f9fa')
    gradient.addColorStop(1, '#e9ecef')
    ctx.fillStyle = gradient
    ctx.fillRect(0, 0, width, height)

    // Draw border
    ctx.strokeStyle = '#dee2e6'
    ctx.lineWidth = 3
    ctx.setLineDash([15, 10])
    ctx.strokeRect(20, 20, width - 40, height - 40)

    // Optimized font rendering
    ctx.fillStyle = '#495057'
    ctx.font = 'bold 32px "Microsoft YaHei", "PingFang SC", "Hiragino Sans GB", "Source Han Sans SC", "Noto Sans SC", "WenQuanYi Micro Hei", "SimHei", "SimSun", Arial, sans-serif'
    ctx.textAlign = 'center'
    ctx.fillText('PPT Preview', width / 2, height * 0.35)

    ctx.fillStyle = '#6c757d'
    ctx.font = '18px "Microsoft YaHei", "PingFang SC", "Hiragino Sans GB", "Source Han Sans SC", "Noto Sans SC", "WenQuanYi Micro Hei", "SimHei", "SimSun", Arial, sans-serif'
    ctx.fillText('Frontend Screenshot Service', width / 2, height * 0.5)

    ctx.fillStyle = '#adb5bd'
    ctx.font = '16px "Microsoft YaHei", "PingFang SC", "Hiragino Sans GB", "Source Han Sans SC", "Noto Sans SC", "WenQuanYi Micro Hei", "SimHei", "SimSun", Arial, sans-serif'
    ctx.fillText(`Size: ${width} Γ— ${height}`, width / 2, height * 0.6)

    return canvas.toDataURL('image/jpeg', 0.9)
  }
}

// Create singleton instance
export const dataSyncService = new DataSyncService()
export default dataSyncService