File size: 43,418 Bytes
6da9beb
 
7d9ad19
 
 
451df29
7d9ad19
bcfd882
7d9ad19
 
b2a6404
7d9ad19
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
58132fb
7d9ad19
 
 
 
 
 
 
 
 
2e9848b
58132fb
6da9beb
bcfd882
 
b2a6404
bcfd882
 
 
 
6da9beb
bcfd882
6da9beb
9d89208
 
93ad82e
b2a6404
6da9beb
3e3029e
6da9beb
9d89208
b2a6404
5bf59c5
6da9beb
 
 
5bf59c5
93ad82e
 
 
6da9beb
9d89208
 
6da9beb
93ad82e
893530c
 
3e3029e
6da9beb
9d89208
b2a6404
5bf59c5
6da9beb
 
 
5bf59c5
93ad82e
9433d44
893530c
 
51365b6
93ad82e
 
6da9beb
a7a0c57
1c1a4ec
3411c0f
63ac902
b2a6404
 
6da9beb
68ec2c8
 
 
 
ca5fb80
3e3029e
ca5fb80
 
 
3e3029e
ca5fb80
3e3029e
ca5fb80
 
68ec2c8
 
 
3e3029e
 
68ec2c8
ca5fb80
 
 
 
 
 
3e3029e
 
5bf59c5
6821e3e
5bf59c5
 
 
6821e3e
5bf59c5
 
bcfd882
96fd583
5bf59c5
bcfd882
5bf59c5
96fd583
5bf59c5
 
6821e3e
5bf59c5
58132fb
5bf59c5
893530c
5bf59c5
 
 
d0216d7
3a86e21
9d89208
 
 
 
 
 
 
5bf59c5
 
 
 
 
 
9d89208
5bf59c5
 
ca5fb80
5bf59c5
 
 
63aee38
5bf59c5
 
 
3a86e21
9d89208
5bf59c5
9d89208
5bf59c5
 
3a86e21
9433d44
 
 
 
 
 
5bf59c5
 
 
 
 
 
9d89208
b2a6404
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
9d89208
5bf59c5
 
ca5fb80
5bf59c5
d0216d7
5bf59c5
 
 
ca5fb80
58132fb
5bf59c5
 
 
 
3a86e21
5bf59c5
 
 
 
 
 
 
 
 
 
 
d0216d7
5bf59c5
 
ca5fb80
5bf59c5
 
 
b2a6404
 
 
 
ca5fb80
5bf59c5
 
 
 
 
 
3a86e21
 
 
 
 
 
 
 
 
 
 
 
9433d44
 
 
 
 
5bf59c5
 
 
 
 
 
9433d44
5bf59c5
 
ca5fb80
5bf59c5
58132fb
5bf59c5
 
51365b6
ca5fb80
58132fb
5bf59c5
58132fb
5bf59c5
 
3a86e21
5bf59c5
 
 
 
 
 
 
 
 
 
 
 
3a86e21
 
 
 
 
 
 
5bf59c5
 
ca5fb80
5bf59c5
 
 
 
3a86e21
5bf59c5
 
 
 
 
3a86e21
9433d44
 
 
 
 
5bf59c5
 
 
 
 
 
 
 
 
 
 
ca5fb80
5bf59c5
 
 
 
 
 
51365b6
 
3827e0c
51365b6
3827e0c
5bf59c5
 
 
 
 
 
 
 
 
b2a6404
 
5bf59c5
 
b2a6404
 
 
bcfd882
9433d44
85d2530
5bf59c5
 
3827e0c
 
3a86e21
3d3c71c
3a86e21
 
5bf59c5
3a86e21
 
 
ca5fb80
5bf59c5
 
 
 
 
3827e0c
5bf59c5
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
3a86e21
 
 
3827e0c
5bf59c5
85d2530
63ac902
 
85d2530
5bf59c5
85d2530
02f7d63
85d2530
6821e3e
6da9beb
 
ca5fb80
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
68ec2c8
3827e0c
96139a3
7d9ad19
96139a3
7d9ad19
3827e0c
ca5fb80
68ec2c8
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
ca5fb80
 
68ec2c8
 
 
 
 
 
 
ca5fb80
 
 
 
 
 
 
 
 
 
 
 
 
 
93ad82e
 
7fd8bde
 
 
93ad82e
63ac902
 
 
 
 
 
 
 
 
 
 
fc3fcdf
63ac902
 
 
 
 
fc3fcdf
63ac902
ca5fb80
 
63ac902
 
ca5fb80
b2a6404
63ac902
 
 
 
 
 
 
 
 
b2a6404
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
6da9beb
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
d0216d7
2e9848b
 
 
 
 
 
 
 
 
 
58132fb
2e9848b
 
6da9beb
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c78f7b4
2e9848b
 
 
c78f7b4
2e9848b
 
c78f7b4
2e9848b
 
c78f7b4
2e9848b
 
 
 
 
 
c78f7b4
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c78f7b4
2e9848b
c78f7b4
2e9848b
 
 
 
 
c78f7b4
2e9848b
 
6da9beb
2e9848b
 
 
 
 
 
 
 
93ad82e
2e9848b
9d89208
2e9848b
9d89208
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
9d89208
2e9848b
 
 
 
 
 
 
 
9d89208
2e9848b
 
 
 
93ad82e
2e9848b
93ad82e
 
2e9848b
93ad82e
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
93ad82e
2e9848b
 
 
 
 
 
 
 
 
68ec2c8
 
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
68ec2c8
2e9848b
 
 
 
 
 
 
 
 
 
 
68ec2c8
2e9848b
 
58132fb
 
2e9848b
58132fb
 
 
 
 
 
2e9848b
 
68ec2c8
2e9848b
 
 
 
 
 
 
 
 
c8c2f6c
 
 
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c8c2f6c
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
c8c2f6c
2e9848b
c8c2f6c
 
2e9848b
 
 
 
 
 
 
 
 
 
 
93ad82e
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
93ad82e
 
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
93ad82e
2e9848b
 
6da9beb
93ad82e
2e9848b
 
 
 
 
 
 
93ad82e
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
58132fb
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
893530c
2e9848b
 
 
93ad82e
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
51365b6
 
 
2e9848b
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
6da9beb
 
2e9848b
6da9beb
2e9848b
6da9beb
 
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
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
"use client"

import React, { useEffect, useRef, useTransition } from "react"
import { IoMdPhonePortrait } from "react-icons/io"
import { GiRollingDices } from "react-icons/gi"
import { FaCloudDownloadAlt, FaDiscord } from "react-icons/fa"
import { GiSpellBook } from "react-icons/gi"
import { useLocalStorage } from "usehooks-ts"
import { ClapProject, ClapMediaOrientation, ClapSegmentCategory, updateClap } from "@aitube/clap"
import Image from "next/image"
import { useSearchParams } from "next/navigation"
import { useFilePicker } from "use-file-picker"
import { DeviceFrameset } from "react-device-frameset"
import "react-device-frameset/styles/marvel-devices.min.css"

import { Card, CardContent, CardHeader } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { Toaster } from "@/components/ui/sonner"
import { TextareaField } from "@/components/form/textarea-field"
import { cn } from "@/lib/utils/cn"

import { createClap } from "./server/aitube/createClap"
import { editClapEntities } from "./server/aitube/editClapEntities"
import { editClapDialogues } from "./server/aitube/editClapDialogues"
import { editClapStoryboards } from "./server/aitube/editClapStoryboards"
import { editClapSounds } from "./server/aitube/editClapSounds"
import { editClapMusic } from "./server/aitube/editClapMusic"
import { editClapVideos } from "./server/aitube/editClapVideos"
import { exportClapToVideo } from "./server/aitube/exportClapToVideo"

import { useStore } from "./store"
import HFLogo from "./hf-logo.svg"
import { Input } from "@/components/ui/input"
import { Field } from "@/components/form/field"
import { Label } from "@/components/form/label"
import { getParam } from "@/lib/utils/getParam"
import { GenerationStage } from "@/types"
import { FileContent } from "use-file-picker/dist/interfaces"
import { generateRandomStory } from "@/lib/utils/generateRandomStory"
import { logImage } from "@/lib/utils/logImage"
import { defaultPrompt } from "./config"
import { Tooltip, TooltipContent, TooltipProvider, TooltipTrigger } from "@/components/ui/tooltip"

export function Main() {
  const [storyPromptDraft, setStoryPromptDraft] = useLocalStorage<string>(
    "AI_STORIES_FACTORY_STORY_PROMPT_DRAFT",
    defaultPrompt
  )
  const promptDraftRef = useRef("")
  promptDraftRef.current = storyPromptDraft

  const [_isPending, startTransition] = useTransition()

  const storyPrompt = useStore(s => s.storyPrompt)
  const mainCharacterImage = useStore(s => s.mainCharacterImage)
  const mainCharacterVoice = useStore(s => s.mainCharacterVoice)
  const orientation = useStore(s => s.orientation)
  const setOrientation = useStore(s => s.setOrientation)
  const status = useStore(s => s.status)
  const parseGenerationStatus = useStore(s => s.parseGenerationStatus)
  const storyGenerationStatus = useStore(s => s.storyGenerationStatus)
  const assetGenerationStatus = useStore(s => s.assetGenerationStatus)
  const soundGenerationStatus = useStore(s => s.soundGenerationStatus)
  const musicGenerationStatus = useStore(s => s.musicGenerationStatus)
  const voiceGenerationStatus = useStore(s => s.voiceGenerationStatus)
  const imageGenerationStatus = useStore(s => s.imageGenerationStatus)
  const videoGenerationStatus = useStore(s => s.videoGenerationStatus)
  const finalGenerationStatus = useStore(s => s.finalGenerationStatus)
  const currentClap = useStore(s => s.currentClap)
  const currentVideo = useStore(s => s.currentVideo)
  const currentVideoOrientation = useStore(s => s.currentVideoOrientation)
  const setStoryPrompt = useStore(s => s.setStoryPrompt)
  const setMainCharacterImage = useStore(s => s.setMainCharacterImage)
  const setMainCharacterVoice = useStore(s => s.setMainCharacterVoice)
  const setStatus = useStore(s => s.setStatus)
  const toggleOrientation = useStore(s => s.toggleOrientation)
  const error = useStore(s => s.error)
  const setError = useStore(s => s.setError)
  const setParseGenerationStatus = useStore(s => s.setParseGenerationStatus)
  const setStoryGenerationStatus = useStore(s => s.setStoryGenerationStatus)
  const setAssetGenerationStatus = useStore(s => s.setAssetGenerationStatus)
  const setSoundGenerationStatus = useStore(s => s.setSoundGenerationStatus)
  const setMusicGenerationStatus = useStore(s => s.setMusicGenerationStatus)
  const setVoiceGenerationStatus = useStore(s => s.setVoiceGenerationStatus)
  const setImageGenerationStatus = useStore(s => s.setImageGenerationStatus)
  const setVideoGenerationStatus = useStore(s => s.setVideoGenerationStatus)
  const setFinalGenerationStatus = useStore(s => s.setFinalGenerationStatus)
  const setCurrentClap = useStore(s => s.setCurrentClap)
  const setCurrentVideo = useStore(s => s.setCurrentVideo)
  const progress = useStore(s => s.progress)
  const setProgress = useStore(s => s.setProgress)
  const saveVideo = useStore(s => s.saveVideo)
  const saveClap = useStore(s => s.saveClap)
  const loadClap = useStore(s => s.loadClap)

  // let's disable this for now
  const canSeeBetaFeatures = true // getParam<boolean>("beta", false)

  const isBusy = useStore(s => s.isBusy)
  const busyRef = useRef(isBusy)
  busyRef.current = isBusy

  const importStory = async (fileData: FileContent<ArrayBuffer>): Promise<{
    clap: ClapProject
    regenerateVideo: boolean
  }> => {
    if (!fileData?.name) { throw new Error(`invalid file (missing file name)`) }

    const {
      setParseGenerationStatus,
    } = useStore.getState()

    setParseGenerationStatus("generating")

    try {
      const blob = new Blob([fileData.content])
      const res = await loadClap(blob, fileData.name)

      if (!res?.clap) { throw new Error(`failed to load the clap file`) }

      setParseGenerationStatus("finished")
      return res
    } catch (err) {
      console.error("failed to load the Clap file:", err)
      setParseGenerationStatus("error")
      throw err
    }
  }


  const generateStory = async (): Promise<ClapProject> => {

    let clap: ClapProject | undefined = undefined
    try {
      setProgress(0)

      setStatus("generating")
      setStoryGenerationStatus("generating")
      setStoryPrompt(promptDraftRef.current)

      clap = await createClap({
        prompt: promptDraftRef.current,
        orientation: useStore.getState().orientation,

        turbo: true,
      })

      if (!clap) { throw new Error(`failed to create the clap`) }

      if (clap.segments.length <= 1) { throw new Error(`failed to generate more than one segments`) }

      console.log(`handleSubmit(): received a clap = `, clap)
      setCurrentClap(clap)
      setStoryGenerationStatus("finished")

      console.log("---------------- GENERATED STORY ----------------")
      console.table(clap.segments, [
        // 'startTimeInMs',
        'endTimeInMs',
        // 'track',
        'category',
        'prompt'
      ])
      return clap
    } catch (err) {
      setStoryGenerationStatus("error")
      throw err
    }
  }

  const generateEntities = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(20)
      setAssetGenerationStatus("generating")
      clap = await editClapEntities({
        clap,

        // generating entities requires a "smart" LLM
        turbo: false,
        // turbo: true,
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the entities`) }

      console.log(`handleSubmit(): received a clap with entities = `, clap)
      setAssetGenerationStatus("finished")
      console.log("---------------- GENERATED ENTITIES ----------------")
      console.table(clap.entities, [
        'category',
        'label',
        'imagePrompt',
        'appearance'
      ])
      return clap
    } catch (err) {
      setAssetGenerationStatus("error")
      throw err
    }
  }

  const generateSounds = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(30)
      setSoundGenerationStatus("generating")

      clap = await editClapSounds({
        clap,
        turbo: true
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the sound`) }

      console.log(`handleSubmit(): received a clap with sound = `, clap)
      setSoundGenerationStatus("finished")
      console.log("---------------- GENERATED SOUND ----------------")
      console.table(clap.segments.filter(s => s.category === ClapSegmentCategory.SOUND), [
        'endTimeInMs',
        'prompt',
        'entityId',
      ])
      return clap
    } catch (err) {
      setSoundGenerationStatus("error")
      throw err
    }
  }

  const generateMusic = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(30)
      setMusicGenerationStatus("generating")

      clap = await editClapMusic({
        clap,
        turbo: true
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the music`) }

      console.log(`handleSubmit(): received a clap with music = `, clap)
      setMusicGenerationStatus("finished")
      console.log("---------------- GENERATED MUSIC ----------------")
      console.table(clap.segments.filter(s => s.category === ClapSegmentCategory.MUSIC), [
        'endTimeInMs',
        'prompt',
        'entityId',
      ])
      return clap
    } catch (err) {
      setMusicGenerationStatus("error")
      throw err
    }
  }

  const generateStoryboards = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(40)
      setImageGenerationStatus("generating")
      clap = await editClapStoryboards({
        clap,
        // if we use entities, then we MUST use turbo
        // that's because turbo uses PulID,
        // but SDXL doesn't
        turbo: true,
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the storyboards`) }

      // const fusion = 
      console.log(`handleSubmit(): received a clap with images = `, clap)
      setImageGenerationStatus("finished")
      console.log("---------------- GENERATED STORYBOARDS ----------------")
      clap.segments
        .filter(s => s.category === ClapSegmentCategory.STORYBOARD)
        .forEach((s, i) => {
          if (s.status === "completed" && s.assetUrl) {
            // console.log(`  [${i}] storyboard: ${s.prompt}`)
            logImage(s.assetUrl, 0.35)
          } else {
            console.log(`  [${i}] failed to generate storyboard`)
          }
          // console.log(`------------------`)
        })
      console.table(clap.segments.filter(s => s.category === ClapSegmentCategory.STORYBOARD), [
        'endTimeInMs',
        'prompt',
        'assetUrl'
      ])
      return clap
    } catch (err) {
      setImageGenerationStatus("error")
      throw err
    }
  }

  const generateVideos = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(50)
      setVideoGenerationStatus("generating")

      clap = await editClapVideos({
        clap,
        turbo: false
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the videos`) }

      console.log(`handleSubmit(): received a clap with videos = `, clap)
      setVideoGenerationStatus("finished")
      console.log("---------------- GENERATED VIDEOS ----------------")
      console.table(clap.segments.filter(s => s.category === ClapSegmentCategory.VIDEO), [
        'endTimeInMs',
        'prompt',
        'entityId',
      ])
      return clap
    } catch (err) {
      setVideoGenerationStatus("error")
      throw err
    }
  }

  const generateStoryboardsThenVideos = async (clap: ClapProject): Promise<ClapProject> => {
    clap = await generateStoryboards(clap)
    clap = await generateVideos(clap)
    return clap
  }


  const generateDialogues = async (clap: ClapProject): Promise<ClapProject> => {
    try {
      // setProgress(70)
      setVoiceGenerationStatus("generating")
      clap = await editClapDialogues({
        clap,
        turbo: true
      }).then(r => r.promise)

      if (!clap) { throw new Error(`failed to edit the dialogues`) }

      console.log(`handleSubmit(): received a clap with dialogues = `, clap)
      setVoiceGenerationStatus("finished")
      console.log("---------------- GENERATED DIALOGUES ----------------")
      console.table(clap.segments.filter(s => s.category === ClapSegmentCategory.DIALOGUE), [
        'endTimeInMs',
        'prompt',
        'entityId',
      ])
      return clap
    } catch (err) {
      setVoiceGenerationStatus("error")
      throw err
    }
  }

  const generateFinalVideo = async (clap: ClapProject): Promise<string> => {

    let assetUrl = ""
    try {
      // setProgress(85)
      setFinalGenerationStatus("generating")
      assetUrl = await exportClapToVideo({
        clap,
        turbo: true
      })

      setCurrentVideo(assetUrl)

      if (assetUrl.length < 128) { throw new Error(`generateFinalVideo(): the generated video is too small, so we failed`) }

      console.log(`generateFinalVideo(): received a video: ${assetUrl.slice(0, 120)}...`)
      setFinalGenerationStatus("finished")
      return assetUrl
    } catch (err) {
      setFinalGenerationStatus("error")
      throw err
    }
  }
  
  const handleSubmit = async () => {
    setStatus("generating")
    busyRef.current = true

    startTransition(async () => {
      setStatus("generating")
      busyRef.current = true
  
      console.log(`handleSubmit(): generating a clap using prompt = "${promptDraftRef.current}" `)

      try {
        let clap = await generateStory()
        
        setCurrentClap(clap)

        const tasks = [
          generateMusic(clap),
          generateStoryboardsThenVideos(clap)
        ]

        const claps = await Promise.all(tasks)

        console.log(`finished processing ${tasks.length} tasks in parallel`)

        for (const newerClap of claps) {
          clap = await updateClap(clap, newerClap, {
            overwriteMeta: false,
            inlineReplace: true,
          })
          setCurrentClap(clap)
        }

        /*
        clap = await claps.reduce(async (existingClap, newerClap) =>
          updateClap(existingClap, newerClap, {
            overwriteMeta: false,
            inlineReplace: true,
          })
        , Promise.resolve(clap)
        */

       
        // We can't have consistent characters with video (yet)
        // clap = await generateEntities(clap)

        /*
        if (mainCharacterImage) {
          console.log("handleSubmit(): User specified a main character image")
          // various strategies here, for instance we can assume that the first character is the main character,
          // or maybe a more reliable way is to count the number of occurrences.
          // there is a risk of misgendering, so ideally we should add some kind of UI to do this,
          // such as a list of characters.
        }
        */

        // let's skip storyboards for now
        // clap = await generateStoryboards(clap)

        // clap = await generateVideos(clap)
        // clap = await generateDialogues(clap)
     
        
        
        console.log("final clap: ", clap)
        setCurrentClap(clap)
        await generateFinalVideo(clap)

        setStatus("finished")
        setError("")
      } catch (err) {
        console.error(`failed to generate: `, err)
        setStatus("error")
        setError(`Error, please contact an admin on Discord (${err})`)
      }
    })
  }


  const { openFilePicker, filesContent } = useFilePicker({
    accept: '.clap',
    readAs: "ArrayBuffer"
  })
  const fileData = filesContent[0]

  useEffect(() => {
    const fn = async () => {
      if (!fileData?.name) { return }

      const { setStatus, setProgress } = useStore.getState()

      setProgress(0)
      setStatus("generating")

      try {
        let { clap, regenerateVideo } = await importStory(fileData)
        
        // clap = await generateSounds(clap)

        // setCurrentClap(clap)

        console.log("loadClap(): clap = ", clap)

        // it is important to skip regeneration if we already have a video
        if (regenerateVideo) {
          console.log(`regenerating music and videos..`)
          const claps = await Promise.all([
            generateMusic(clap),
            generateVideos(clap)
          ])

          // console.log("finished processing the 2 tasks in parallel")

          for (const newerClap of claps) {
            clap = await updateClap(clap, newerClap, {
              overwriteMeta: false,
              inlineReplace: true,
            })
          }


          setCurrentClap(clap)

          await generateFinalVideo(clap)

        } else {
          console.log(`skipping music and video regeneration`)
        }

        setStatus("finished")
        setProgress(100)
        setError("")
      } catch (err) {
        console.error(`failed to import: `, err)
        setStatus("error")
        setError(`${err}`)
      }
     
    }
    fn()
  }, [fileData?.name])
  
  // note: we are interested in the *current* video orientation,
  // not the requested video orientation requested for the next video
  const isLandscape = currentVideoOrientation === ClapMediaOrientation.LANDSCAPE
  const isPortrait = currentVideoOrientation === ClapMediaOrientation.PORTRAIT
  const isSquare = currentVideoOrientation === ClapMediaOrientation.SQUARE

  const runningRef = useRef(false)
  const timerRef = useRef<NodeJS.Timeout>()

  const timerFn = async () => {
    const { isBusy, progress, stage } = useStore.getState()

    clearTimeout(timerRef.current)
    if (!isBusy || stage === "idle") {
      return
    }

    /*
    console.log("progress function:", {
      stage,
      delay: progressDelayInMsPerStage[stage],
      progress,
    })
    */
    useStore.setState({
      // progress: Math.min(maxProgressPerStage[stage], progress + 1) 
      progress: Math.min(100, progress + 1) 
    })

    // timerRef.current = setTimeout(timerFn, progressDelayInMsPerStage[stage])
    timerRef.current = setTimeout(timerFn, 1200)
  }

  useEffect(() => {
    timerFn()
    clearTimeout(timerRef.current)
    if (!isBusy) { return }
    timerRef.current = setTimeout(timerFn, 0)
  }, [isBusy])

  // this is how we support query string parameters
  // ?prompt=hello <- set a default prompt
  // ?prompt=hello&autorun=true <- automatically run the app
  // ?orientation=landscape <- can be "landscape" or "portrait" (default)
  const searchParams = useSearchParams()
  const queryStringPrompt = (searchParams?.get('prompt') as string) || ""
  const queryStringAutorun = (searchParams?.get('autorun') as string) || ""
  const queryStringOrientation = (searchParams?.get('orientation') as string) || ""
  useEffect(() => {
    if (queryStringOrientation?.length > 1) {
      console.log(`orientation = "${queryStringOrientation}"`)
      const orientation =
        queryStringOrientation.trim().toLowerCase() === "landscape"
        ? ClapMediaOrientation.LANDSCAPE
        : ClapMediaOrientation.PORTRAIT
      setOrientation(orientation)
    }
    if (queryStringPrompt?.length > 1) {
      console.log(`prompt = "${queryStringPrompt}"`)
      if (queryStringPrompt !== promptDraftRef.current) {
        setStoryPromptDraft(queryStringPrompt)
      }
      const maybeAutorun = queryStringAutorun.trim().toLowerCase()
      console.log(`autorun = "${maybeAutorun}"`)

      // note: during development we will be called twice,
      // which is why we have a guard on  busyRef.current
      if (maybeAutorun === "true" || maybeAutorun === "1" && !busyRef.current) {
        handleSubmit()
      }
    }
  }, [queryStringPrompt, queryStringAutorun, queryStringOrientation])

  return (
    <TooltipProvider>
      <div className={cn(
        `fixed`,
        // `bg-zinc-800`,
        // old style, more "amber"
        // `bg-gradient-to-br from-amber-600 to-yellow-500`, 

        // nice style!
        // `bg-gradient-to-br from-amber-700 to-yellow-300`, 

        // warm orange, a bit flash but not bad, not bad at all
        // `bg-gradient-to-br from-orange-700 to-yellow-400`, 
    
        // nice "AiTube" vibe
        // `bg-gradient-to-br from-red-700 to-yellow-400`, 
    
        // pretty cool lime!
        `bg-gradient-to-br from-lime-700 to-yellow-400`, 

        // new style, pretty "fresh" - maybe too bright? 
        // use a dark logo for this one
        // `bg-gradient-to-br from-yellow-200 to-yellow-500`, 
        
        // too pastel
        // `bg-gradient-to-br from-yellow-200 to-red-300`, 
        
        // `bg-gradient-to-br from-sky-400 to-sky-300/30`, 
        `w-screen h-full overflow-y-scroll md:overflow-hidden`,
      )}

        // this version is a bit too aggressive on mobile
        // style={{ boxShadow: "inset 0 0px 250px 0 rgb(0 0 0 / 60%)" }}

        style={{ boxShadow: "inset 0 0 10vh 0 rgb(0 0 0 / 50%)" }}
        >
        <div className="flex flex-col w-screen h-screen">
          <div className="
          flex flex-col md:flex-row w-full
          items-center md:justify-center
          flex-1
          "
          >
            <div className={cn(
              `flex flex-col md:h-full md:items-center md:justify-center`,
              `w-full md:w-1/2`,
              `transition-all duration-200 ease-in-out`,
              `ml-0`,
              `pt-4 sm:pt-0`,
            )}>
              <Card className={cn(
              //  `shadow-2xl z-30 rounded-3xl`,
              `shadow-none`,
              `w-full md:ml-[12%] md:w-[95%]`,
                `transition-all duration-200 ease-in-out`,
                `bg-transparent dark:bg-transparent`,
              // `backdrop-blur-2xl dark:backdrop-blur-2xl`,
                // `bg-amber-500 dark:bg-amber-500`,


                `border-transparent dark:border-transparent`,
                // `bg-stone-50/90 dark:bg-stone-50/90`,
                // `border-yellow-100 dark:border-yellow-100`,
                // `bg-yellow-500 dark:bg-yellow-500`,
                // `border-yellow-400 dark:border-yellow-400`,

              )}>
                <CardHeader>
                  <div className="flex flex-col justify-start">
                    <div className="
                      flex flex-row
                      items-center justify-center
                      transition-all duration-200 ease-in-out
                      px-3
                      
                      rounded-full">
                      <div
                      className="
                        flex
                        transition-all duration-200 ease-in-out
                        items-center justify-center text-center
                        w-10 h-10 md:w-12 md:h-12 lg:w-16 lg:h-16
                        text-3xl md:text-4xl lg:text-5xl
                        rounded-lg
                        mr-2
                        font-sans
                        bg-amber-400 dark:bg-amber-400
                      
                        text-stone-950/80 dark:text-stone-950/80 font-bold
                        "
                        >AI</div>
                        <div
                          className="
                          transition-all duration-200 ease-in-out
                          text-amber-400 dark:text-amber-400
                          text-3xl md:text-4xl lg:text-5xl
                          "

                          style={{ textShadow: "#00000035 0px 0px 2px" }}
                      
                          /*
                          className="
                          text-5xl
                          bg-gradient-to-br from-yellow-300 to-yellow-500
                          inline-block text-transparent bg-clip-text
                          py-6
                          "
                          */
                        >Stories Factory</div>
                      </div>

                      <p
                        className="
                        transition-all duration-200 ease-in-out
                        text-stone-900/90 dark:text-stone-900/90
                        text-lg md:text-xl lg:text-2xl
                        text-center 
                        pt-2 md:pt-4
                        "
                        style={{ textShadow: "rgb(255 255 255 / 19%) 0px 0px 2px" }}
                      >Make video stories using AI ✨</p>
                    </div>
                  </CardHeader>
                  <CardContent
                    className="flex flex-col space-y-3"
                    >
                    
                    {/* LEFT MENU BUTTONS + MAIN PROMPT INPUT */}
                    <div className="flex flex-row space-x-3 w-full">
                    
                  
                      {/*

                      TODO: To finish by Julian a bit later

                      <div className="
                        flex flex-col
                        
                        w-32 bg-yellow-600
                        transition-all duration-200 ease-in-out
                        space-y-2 md:space-y-4
                      ">
                        <Input
                            type="file"
                            className=""
                            onChange={async (e: React.ChangeEvent<HTMLInputElement>) => {
                              if (e.target.files && e.target.files.length > 0) {
                                const file = e.target.files[0];
                                const newImageBase64 = await fileToBase64(file)
                                setMainCharacterImage(newImageBase64)
                              }
                            }}
                            accept="image/*"
                          />
                      </div>
                      */}

                      {/* MAIN PROMPT INPUT */}
                      <div className="
                        flex flex-col
                        flex-1
                        transition-all duration-200 ease-in-out
                        space-y-2 md:space-y-4
                      ">
                        <TextareaField
                          id="story-prompt-draft"
                          // label="My story:"
                          // disabled={modelState != 'ready'}
                          onChange={(e) => {
                            setStoryPromptDraft(e.target.value)
                            promptDraftRef.current = e.target.value
                          }}
                          placeholder={defaultPrompt}
                          inputClassName="
                          transition-all duration-200 ease-in-out
                          h-32 md:h-56 lg:h-64
                      
                          "
                          disabled={isBusy}
                          value={storyPromptDraft}
                        />
      
                      
                      {/* END OF MAIN PROMPT INPUT */}
                      </div>

                      {/* END OF LEFT MENU BUTTONS + MAIN PROMPT INPUT */}
                    </div>

                    {/* ACTION BAR */}

                    <div className="
                      w-full
                      flex flex-col lg:flex-row
                      justify-between items-center
                      space-y-3 lg:space-x-3 lg:space-y-0">
                
                      {/*
                        <Button
                          onClick={() => load()}
                          disabled={isBusy}
                          // variant="ghost"
                          className={cn(
                            `text-sm md:text-base lg:text-lg`,
                            `bg-stone-800/90 text-amber-400/100 dark:bg-stone-800/90 dark:text-amber-400/100`,
                            `font-bold`,
                            `hover:bg-stone-800/100 hover:text-amber-300/100 dark:hover:bg-stone-800/100 dark:hover:text-amber-300/100`,
                            storyPromptDraft ? "opacity-100" : "opacity-80"
                          )}
                        >
                        <span className="mr-1">Load project</span>
                        </Button>
                      */}

                      <div className="
                
                      flex flex-row
                      justify-between items-center
                      space-x-3">
                        
                
                      {canSeeBetaFeatures && 
                        <Tooltip>
                          <TooltipTrigger asChild><Button
              
                            onClick={openFilePicker}
                            disabled={isBusy}
                            // variant="ghost"
                            className={cn(
                              `text-xs md:text-sm lg:text-base`,
                              `bg-stone-800/90 text-amber-400/100 dark:bg-stone-800/90 dark:text-amber-400/100`,
                              `font-bold`,
                              `hover:bg-stone-800/100 hover:text-amber-300/100 dark:hover:bg-stone-800/100 dark:hover:text-amber-300/100`,
                              storyPromptDraft ? "opacity-100" : "opacity-80"
                            )}
                          >
                            <span className="hidden xl:inline mr-1">Load .clap</span>
                            <span className="inline xl:hidden mr-1">Load .clap</span>
                          </Button></TooltipTrigger>
                        <TooltipContent side="top">
                          <p className="text-xs font-normal text-stone-100/90 text-center">
                            Clap is a new AI format,<br/>check out the academy<br/>to learn more about it.
                          </p>
                        </TooltipContent>
                      </Tooltip>}
                
    
    
                      {canSeeBetaFeatures &&
                        <Tooltip>
                          <TooltipTrigger asChild><Button
                        onClick={() => saveClap()}
                        disabled={!currentClap || isBusy}
                        // variant="ghost"
                        className={cn(
                          `text-xs md:text-sm lg:text-base`,
                          `bg-stone-800/90 text-amber-400/100 dark:bg-stone-800/90 dark:text-amber-400/100`,
                          `font-bold`,
                          `hover:bg-stone-800/100 hover:text-amber-300/100 dark:hover:bg-stone-800/100 dark:hover:text-amber-300/100`,
                          storyPromptDraft ? "opacity-100" : "opacity-80"
                        )}
                      >
                        <span className="hidden xl:inline mr-1">Save .clap</span>
                        <span className="inline xl:hidden mr-1">Save .clap</span>
                      </Button></TooltipTrigger>
                        <TooltipContent side="top">
                          <p className="text-xs font-normal text-stone-100/90 text-center">
                          Clap is a new AI format,<br/>check out the academy<br/>to learn more about it.
                          </p>
                        </TooltipContent>
                      </Tooltip> 
                      }
                      </div>
    
                      <div className=" 
                      flex flex-row
                      justify-between items-center
                      space-x-3
                      select-none
                      ">

  
                        {/* RANDOMNESS SWITCH */}
                        <div className=" 
                        flex flex-row
                        justify-between items-center
                        cursor-pointer
                        transition-all duration-150 ease-in-out
                        hover:scale-110 active:scale-150
                      text-stone-800
                        hover:text-stone-950
                        active:text-black
                        group
                        "
                        onClick={() => {
                          const randomStory = generateRandomStory()
                          setStoryPromptDraft(randomStory)
                          promptDraftRef.current = randomStory
                        }}>
                          <div>
                          </div>
                          <div className="
                          w-6 h-8
                          flex flex-row items-center justify-center
                          transition-all duration-150 ease-out
                          group-hover:animate-swing
                          "
                          >
                            <GiRollingDices size={24} />
                          </div>
                        </div>
                        {/* END OF RANDOMNESS SWITCH */}


                        {/* ORIENTATION SWITCH */}
                        <div className=" 
                        flex flex-row
                        justify-between items-center
                        cursor-pointer
                        transition-all duration-150 ease-out
                        hover:scale-110 active:scale-150
                        text-stone-800
                        hover:text-stone-950
                        active:text-black
                        group
                        "
                        onClick={() => toggleOrientation()}>
                          <div>
                          </div>
                          <div className="
                          w-8 h-8
                          flex flex-row items-center justify-center
                          transition-all duration-150 ease-in-out
                          group-hover:animate-swing
                          "
                          >
                            <div className={cn(
                              `transition-all duration-200 ease-in-out`,
                              orientation === ClapMediaOrientation.LANDSCAPE ? `rotate-90` : `rotate-0`
                            )}>
                              <IoMdPhonePortrait size={24} />
                            </div>
                          </div>
                        </div>
                        {/* END OF ORIENTATION SWITCH */}
                        <Button
                          onClick={handleSubmit}
                          disabled={!storyPromptDraft || isBusy}
                          // variant="ghost"
                          className={cn(
                            `text-base md:text-lg lg:text-xl xl:text-2xl`,
                            `bg-stone-800/90 text-amber-400/100 dark:bg-stone-800/90 dark:text-amber-400/100`,
                            `font-bold`,
                            `hover:bg-stone-800/100 hover:text-amber-300/100 dark:hover:bg-stone-800/100 dark:hover:text-amber-300/100`,
                            storyPromptDraft ? "opacity-100" : "opacity-80"
                          )}
                        >
                        <span className="mr-1.5">Create</span><span className="hidden md:inline">πŸ‘‰</span><span className="inline md:hidden">πŸ‘‡</span>
                        </Button>
                      </div>

                    {/* END OF ACTION BAR */}
                    </div>

                </CardContent>
              </Card>
            </div>
            <div className={cn(
              `flex flex-col items-center justify-center`,
              `flex-1 h-full`,
              // `transition-all duration-200 ease-in-out`

              `-mt-[20px] -mb-[90px] md:-mt-0 md:-mb-0`,
            )}>
              
              <div className={cn(`
                -mt-8 md:mt-0
                transition-all duration-200 ease-in-out
              `,
              isLandscape
                ? `scale-[0.9] md:scale-[0.75] lg:scale-[0.9] xl:scale-[1.0] 2xl:scale-[1.1]`
                : `scale-[0.8] md:scale-[0.9] lg:scale-[1.1]`
                )}>
                <DeviceFrameset
                  device="Nexus 5"
                  // color="black"

                  landscape={isLandscape}

                  // note 1: videos are generated in 1024x576 or 576x1024
                  // so we need to keep the same ratio here

                  // note 2: width and height are fixed, if width always stays 512
                  // that's because the landscape={} parameter will do the switch for us

                  width={288}
                  height={512}
                >
                  <div className="
                  flex flex-col items-center justify-center
                  w-full h-full
                  bg-black text-white
                  ">
                    {isBusy ? <div className="
                    flex flex-col 
                    items-center justify-center
                    text-center space-y-1.5">
                      <p className="text-2xl font-bold">{progress}%</p> 
                      <p className="text-base text-white/70">{isBusy
                        ? (
                          // note: some of those tasks are running in parallel,
                          // and some are super-slow (like music or video)
                          // by carefully selecting in which order we set the ternaries,
                          // we can create the illusion that we just have a succession of reasonably-sized tasks
                          storyGenerationStatus === "generating" ? "Writing story.."
                          : parseGenerationStatus === "generating" ? "Loading the project.."
                          : assetGenerationStatus === "generating" ? "Casting characters.."
                          : imageGenerationStatus === "generating" ? "Creating storyboards.."
                          : soundGenerationStatus === "generating" ? "Recording sounds.."
                          : videoGenerationStatus === "generating" ? "Filming shots.."
                          : musicGenerationStatus === "generating" ? "Producing music.."
                          : voiceGenerationStatus === "generating" ? "Recording dialogues.."
                          : finalGenerationStatus === "generating" ? "Editing final cut.."
                          : "Please wait.."
                        )
                        : status === "error"
                        ? <span>{error || ""}</span>
                        : <span>{error ? error : <span>&nbsp;</span>}</span> // to prevent layout changes
                      }</p>
                      </div>
                    : (currentVideo && currentVideo?.length > 128) ? <video
                      src={currentVideo}
                      controls
                      playsInline
                      // I think we can't autoplay with sound,
                      // so let's disable auto-play
                      // autoPlay
                      // muted
                      loop
                      className="object-cover"
                      style={{
                      }}
                    /> : <div  className="
                    flex flex-col 
                    items-center justify-center
                    text-lg text-center"></div>}
                  </div>

                  <div className={cn(`
                    fixed
                    flex flex-row items-center justify-center
                    bg-transparent
                    font-sans
                    -mb-0
                    `,
                    isLandscape ? 'h-4' : 'h-14'
                  )}
                    style={{ width: isPortrait ? 288 : 512 }}>
                    <span className="text-stone-100/50 text-4xs"
                      style={{ textShadow: "rgb(0 0 0 / 80%) 0px 0px 2px" }}>
                      Powered by
                    </span>
                    <span className="ml-1 mr-0.5">
                      <Image src={HFLogo} alt="Hugging Face" width={14} height={13} />
                    </span>
                    <span className="text-stone-100/80 text-3xs font-semibold"
                      style={{ textShadow: "rgb(0 0 0 / 80%) 0px 0px 2px" }}>Hugging Face</span>
              
                  </div>
                </DeviceFrameset>
                
                {(currentVideo && currentVideo.length > 128) ? <div
                  className={cn(`
                  w-full
                  flex flex-row
                  items-center justify-center
                  transition-all duration-150 ease-in-out
  
                text-stone-800
            
                  group
                  pt-2 md:pt-4
                  `, 
                    isBusy ? 'opacity-50' : 'cursor-pointer opacity-100 hover:scale-110 active:scale-150 hover:text-stone-950 active:text-black'
                  )}
                  style={{ textShadow: "rgb(255 255 255 / 19%) 0px 0px 2px" }}
                  onClick={isBusy ? undefined : saveVideo}
                >
                  <div className="
                  text-base md:text-lg lg:text-xl
                  transition-all duration-150 ease-out
                  group-hover:animate-swing
                  "><FaCloudDownloadAlt /></div>
                  <div className="text-xs md:text-sm lg:text-base">&nbsp;Download</div>
                </div> : null}
              </div>
            </div>
          </div>
          <div
            className="
              fixed
            
              left-4 md:left-8
              bottom-4
              flex flex-col md:flex-row
              md:items-center justify-center
              space-y-4 md:space-x-4 md:space-y-0
      
            ">
            <a
              className="
              flex
              no-underline
              animation-all duration-150 ease-in-out
              group
              text-stone-950/60 hover:text-stone-950/80 scale-95 hover:scale-100"
              href="https://discord.gg/AEruz9B92B"
              target="_blank">
              <div className="
                text-base md:text-lg lg:text-xl
                transition-all duration-150 ease-out
                group-hover:animate-swing
              "><FaDiscord /></div>
              <div className="text-xs md:text-sm lg:text-base ml-1.5">
                <span className="hidden md:block">Chat on Discord</span>
                <span className="block md:hidden">Discord</span>
              </div>
            </a>
            <a
              className="
              flex
              no-underline
              animation-all duration-150 ease-in-out
              group
              text-stone-950/60 hover:text-stone-950/80 scale-95 hover:scale-100"
              href="https://latent-store.notion.site/AI-Stories-Academy-8e3ce6ff2d5946ffadc94193619dd5cd"
              target="_blank">
              <div className="
                text-base md:text-lg lg:text-xl
                transition-all duration-150 ease-out
                group-hover:animate-swing
              "><GiSpellBook /></div>
              <div className="text-xs md:text-sm lg:text-base ml-1.5">
                <span className="hidden md:block">Prompt academy</span>
                <span className="block md:hidden">Academy</span>
              </div>
            </a>
          </div>
        </div>
        <Toaster />
      </div>
    </TooltipProvider>
  );
}