loading.tsx 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. import { render } from "solid-js/web"
  2. import { MetaProvider } from "@solidjs/meta"
  3. import "@opencode-ai/app/index.css"
  4. import { Font } from "@opencode-ai/ui/font"
  5. import { Splash } from "@opencode-ai/ui/logo"
  6. import { Progress } from "@opencode-ai/ui/progress"
  7. import "./styles.css"
  8. import { createEffect, createMemo, createSignal, onCleanup, onMount } from "solid-js"
  9. import { commands, events, InitStep } from "./bindings"
  10. import { Channel } from "@tauri-apps/api/core"
  11. import { initI18n, t } from "./i18n"
  12. const root = document.getElementById("root")!
  13. const lines = [
  14. t("desktop.loading.status.initial"),
  15. t("desktop.loading.status.migrating"),
  16. t("desktop.loading.status.waiting"),
  17. ]
  18. const delays = [3000, 9000]
  19. void initI18n()
  20. render(() => {
  21. const [step, setStep] = createSignal<InitStep | null>(null)
  22. const [line, setLine] = createSignal(0)
  23. const [percent, setPercent] = createSignal(0)
  24. const phase = createMemo(() => step()?.phase)
  25. const value = createMemo(() => {
  26. if (phase() === "done") return 100
  27. return Math.max(25, Math.min(100, percent()))
  28. })
  29. const channel = new Channel<InitStep>()
  30. channel.onmessage = (next) => setStep(next)
  31. commands.awaitInitialization(channel as any).catch(() => undefined)
  32. onMount(() => {
  33. setLine(0)
  34. setPercent(0)
  35. const timers = delays.map((ms, i) => setTimeout(() => setLine(i + 1), ms))
  36. const listener = events.sqliteMigrationProgress.listen((e) => {
  37. if (e.payload.type === "InProgress") setPercent(Math.max(0, Math.min(100, e.payload.value)))
  38. if (e.payload.type === "Done") setPercent(100)
  39. })
  40. onCleanup(() => {
  41. listener.then((cb) => cb())
  42. timers.forEach(clearTimeout)
  43. })
  44. })
  45. createEffect(() => {
  46. if (phase() !== "done") return
  47. const timer = setTimeout(() => events.loadingWindowComplete.emit(null), 1000)
  48. onCleanup(() => clearTimeout(timer))
  49. })
  50. const status = createMemo(() => {
  51. if (phase() === "done") return t("desktop.loading.status.done")
  52. if (phase() === "sqlite_waiting") return lines[line()]
  53. return t("desktop.loading.status.initial")
  54. })
  55. return (
  56. <MetaProvider>
  57. <div class="w-screen h-screen bg-background-base flex items-center justify-center">
  58. <Font />
  59. <div class="flex flex-col items-center gap-11">
  60. <Splash class="w-20 h-25 opacity-15" />
  61. <div class="w-60 flex flex-col items-center gap-4" aria-live="polite">
  62. <span class="w-full overflow-hidden text-center text-ellipsis whitespace-nowrap text-text-strong text-14-normal">
  63. {status()}
  64. </span>
  65. <Progress
  66. value={value()}
  67. class="w-20 [&_[data-slot='progress-track']]:h-1 [&_[data-slot='progress-track']]:border-0 [&_[data-slot='progress-track']]:rounded-none [&_[data-slot='progress-track']]:bg-surface-weak [&_[data-slot='progress-fill']]:rounded-none [&_[data-slot='progress-fill']]:bg-icon-warning-base"
  68. aria-label={t("desktop.loading.progressAria")}
  69. getValueLabel={({ value }) => `${Math.round(value)}%`}
  70. />
  71. </div>
  72. </div>
  73. </div>
  74. </MetaProvider>
  75. )
  76. }, root)