Compare commits
	
		
			3 commits
		
	
	
		
			21afb718f7
			...
			46e7f1aaea
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 46e7f1aaea | ||
|  | eb461d3708 | ||
|  | 8bf133c1cf | 
					 14 changed files with 360 additions and 25 deletions
				
			
		
							
								
								
									
										1
									
								
								.gitattributes
									
										
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								.gitattributes
									
										
									
									
										vendored
									
									
										Normal file
									
								
							|  | @ -0,0 +1 @@ | |||
| *.lockb binary diff=lockb | ||||
							
								
								
									
										
											BIN
										
									
								
								bun.lockb
									
										
									
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								bun.lockb
									
										
									
									
									
								
							
										
											Binary file not shown.
										
									
								
							|  | @ -24,6 +24,7 @@ | |||
|     "vite-plugin-pwa": "^0.20.5", | ||||
|     "vite-plugin-solid": "^2.10.2", | ||||
|     "vite-plugin-solid-styled": "^0.11.1", | ||||
|     "workbox-build": "^7.1.1", | ||||
|     "wrangler": "^3.78.2" | ||||
|   }, | ||||
|   "dependencies": { | ||||
|  | @ -49,7 +50,9 @@ | |||
|     "solid-js": "^1.8.22", | ||||
|     "solid-styled": "^0.11.1", | ||||
|     "stacktrace-js": "^2.0.2", | ||||
|     "web-animations-js": "^2.3.2" | ||||
|     "web-animations-js": "^2.3.2", | ||||
|     "workbox-core": "^7.1.0", | ||||
|     "workbox-precaching": "^7.1.0" | ||||
|   }, | ||||
|   "packageManager": "bun@1.1.21" | ||||
| } | ||||
|  |  | |||
							
								
								
									
										54
									
								
								src/App.tsx
									
										
									
									
									
								
							
							
						
						
									
										54
									
								
								src/App.tsx
									
										
									
									
									
								
							|  | @ -5,6 +5,7 @@ import { | |||
|   createEffect, | ||||
|   createMemo, | ||||
|   createRenderEffect, | ||||
|   createSignal, | ||||
|   ErrorBoundary, | ||||
|   lazy, | ||||
|   onCleanup, | ||||
|  | @ -17,6 +18,14 @@ import { | |||
| import { $accounts, updateAcctInf } from "./accounts/stores.js"; | ||||
| import { useStore } from "@nanostores/solid"; | ||||
| import { DateFnScope, useLanguage } from "./platform/i18n.jsx"; | ||||
| import { useRegisterSW } from "virtual:pwa-register/solid"; | ||||
| import { | ||||
|   isJSONRPCResult, | ||||
|   ResultDispatcher, | ||||
|   type JSONRPC, | ||||
| } from "./serviceworker/services.js"; | ||||
| import { makeEventListener } from "@solid-primitives/event-listener"; | ||||
| import { ServiceWorkerProvider } from "./platform/host.js"; | ||||
| 
 | ||||
| const AccountSignIn = lazy(() => import("./accounts/SignIn.js")); | ||||
| const AccountMastodonOAuth2Callback = lazy( | ||||
|  | @ -58,6 +67,43 @@ const App: Component = () => { | |||
|   const theme = useRootTheme(); | ||||
|   const accts = useStore($accounts); | ||||
|   const lang = useLanguage(); | ||||
|   const [serviceWorker, setServiceWorker] = createSignal<ServiceWorker>(); | ||||
|   const dispatcher = new ResultDispatcher(); | ||||
| 
 | ||||
|   let checkAge = 0; | ||||
|   const untilServiceWorkerAlive = async ( | ||||
|     worker: ServiceWorker, | ||||
|     expectedAge: number, | ||||
|   ) => { | ||||
|     const [call, ret] = dispatcher.createTypedCall("ping", undefined); | ||||
|     worker.postMessage(await call); | ||||
|     const result = await ret; | ||||
|     console.assert(!result.error, result); | ||||
|     if (expectedAge === checkAge) { | ||||
|       setServiceWorker(worker); | ||||
|     } | ||||
|   }; | ||||
| 
 | ||||
|   makeEventListener(window, "message", (event: MessageEvent<JSONRPC>) => { | ||||
|     if (isJSONRPCResult(event.data)) { | ||||
|       dispatcher.dispatch(event.data.id, event.data); | ||||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   const { | ||||
|     needRefresh: [needRefresh], | ||||
|     offlineReady: [offlineReady], | ||||
|   } = useRegisterSW({ | ||||
|     onRegisteredSW(scriptUrl, reg) { | ||||
|       console.info("service worker is registered from %s", scriptUrl); | ||||
|       const active = reg?.active; | ||||
|       if (!active) { | ||||
|         console.warn("No service is in activating or activated"); | ||||
|         return; | ||||
|       } | ||||
|       untilServiceWorkerAlive(active, checkAge++); | ||||
|     }, | ||||
|   }); | ||||
| 
 | ||||
|   const clients = createMemo(() => { | ||||
|     return accts().map((x) => ({ | ||||
|  | @ -103,7 +149,15 @@ const App: Component = () => { | |||
|       <ThemeProvider theme={theme()}> | ||||
|         <DateFnScope> | ||||
|           <ClientProvider value={clients}> | ||||
|             <ServiceWorkerProvider | ||||
|               value={{ | ||||
|                 needRefresh, | ||||
|                 offlineReady, | ||||
|                 serviceWorker, | ||||
|               }} | ||||
|             > | ||||
|               <Routing /> | ||||
|             </ServiceWorkerProvider> | ||||
|           </ClientProvider> | ||||
|         </DateFnScope> | ||||
|       </ThemeProvider> | ||||
|  |  | |||
|  | @ -1,12 +1,37 @@ | |||
| import { createContext, useContext, type Accessor } from "solid-js"; | ||||
| import { useRegisterSW } from "virtual:pwa-register/solid"; | ||||
| 
 | ||||
| export function isiOS() { | ||||
|   return [ | ||||
|     'iPad Simulator', | ||||
|     'iPhone Simulator', | ||||
|     'iPod Simulator', | ||||
|     'iPad', | ||||
|     'iPhone', | ||||
|     'iPod' | ||||
|   ].includes(navigator.platform) | ||||
|   return ( | ||||
|     [ | ||||
|       "iPad Simulator", | ||||
|       "iPhone Simulator", | ||||
|       "iPod Simulator", | ||||
|       "iPad", | ||||
|       "iPhone", | ||||
|       "iPod", | ||||
|     ].includes(navigator.platform) || | ||||
|     // iPad on iOS 13 detection
 | ||||
|   || (navigator.userAgent.includes("Mac") && "ontouchend" in document) | ||||
|     (navigator.userAgent.includes("Mac") && "ontouchend" in document) | ||||
|   ); | ||||
| } | ||||
| 
 | ||||
| export type ServiceWorkerService = { | ||||
|   needRefresh: Accessor<boolean>; | ||||
|   offlineReady: Accessor<boolean>; | ||||
|   serviceWorker: Accessor<ServiceWorker | undefined>; | ||||
| }; | ||||
| 
 | ||||
| const ServiceWorkerContext = /* @__PURE__ */ createContext< | ||||
|   ServiceWorkerService | ||||
| >(({ | ||||
|   needRefresh: () => false, | ||||
|   offlineReady: () => false, | ||||
|   serviceWorker: () => undefined | ||||
| })); | ||||
| 
 | ||||
| export const ServiceWorkerProvider = ServiceWorkerContext.Provider; | ||||
| 
 | ||||
| export function useServiceWorker(): ServiceWorkerService { | ||||
|   return useContext(ServiceWorkerContext); | ||||
| } | ||||
							
								
								
									
										35
									
								
								src/serviceworker/main.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								src/serviceworker/main.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,35 @@ | |||
| import { cleanupOutdatedCaches, precacheAndRoute } from "workbox-precaching"; | ||||
| import { clientsClaim } from "workbox-core"; | ||||
| import type { AnyCall } from "./services"; | ||||
| 
 | ||||
| function checkServiceWorker( | ||||
|   self: WorkerGlobalScope, | ||||
| ): self is ServiceWorkerGlobalScope { | ||||
|   return !!(self as unknown as ServiceWorkerGlobalScope).registration; | ||||
| } | ||||
| 
 | ||||
| if (checkServiceWorker(self)) { | ||||
|   cleanupOutdatedCaches(); | ||||
|   precacheAndRoute(self.__WB_MANIFEST, { | ||||
|     cleanURLs: false | ||||
|   }); | ||||
| 
 | ||||
|   // auto update
 | ||||
|   self.skipWaiting(); | ||||
|   clientsClaim(); | ||||
| } else { | ||||
|   throw new TypeError("This entry point must be run in a service worker"); | ||||
| } | ||||
| 
 | ||||
| self.addEventListener("message", (event: MessageEvent<AnyCall>) => { | ||||
|   if (event.data.method === "ping") { | ||||
|     event.source.postMessage({id: event.data.id, jsonrpc: "2.0", result: undefined}) | ||||
|   } else { | ||||
|     event.source.postMessage({ | ||||
|       error: { | ||||
|         code: -32601, | ||||
|         message: "Method not found" | ||||
|       } | ||||
|     }) | ||||
|   } | ||||
| }) | ||||
							
								
								
									
										177
									
								
								src/serviceworker/services.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										177
									
								
								src/serviceworker/services.ts
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,177 @@ | |||
| export type JSONRPC = { | ||||
|   jsonrpc: "2.0"; | ||||
|   id?: string | number; | ||||
| }; | ||||
| 
 | ||||
| export type Call<T = undefined> = JSONRPC & { | ||||
|   method: string; | ||||
|   params: T; | ||||
| }; | ||||
| 
 | ||||
| export type RemoteError<E = undefined> = { data: E } & ( | ||||
|   | { | ||||
|       code: number; | ||||
|       message: string; | ||||
|     } | ||||
|   | { | ||||
|       code: -32700; | ||||
|       message: "Parse Error"; | ||||
|     } | ||||
|   | { | ||||
|       code: -32600; | ||||
|       message: "Invalid Request"; | ||||
|     } | ||||
|   | { | ||||
|       code: -32601; | ||||
|       message: "Method not found"; | ||||
|     } | ||||
|   | { | ||||
|       code: -32602; | ||||
|       message: "Invalid params"; | ||||
|     } | ||||
|   | { | ||||
|       code: -32603; | ||||
|       message: "Internal error"; | ||||
|     } | ||||
| ); | ||||
| 
 | ||||
| export type Result<T, E> = JSONRPC & { id: string | number } & ( | ||||
|     | { | ||||
|         result: T; | ||||
|         error: undefined; | ||||
|       } | ||||
|     | { error: RemoteError<E>; result: undefined } | ||||
|   ); | ||||
| 
 | ||||
| export function isJSONRPCResult( | ||||
|   object: Record<string, unknown>, | ||||
| ): object is Result<unknown, unknown> { | ||||
|   return object["jsonrpc"] === "2.0" && object["id"] && !object["method"]; | ||||
| } | ||||
| 
 | ||||
| export function isJSONRPCCall( | ||||
|   object: Record<string, unknown>, | ||||
| ): object is Call<unknown> { | ||||
|   return object["jsonrpc"] === "2.0" && !!object["method"]; | ||||
| } | ||||
| 
 | ||||
| export class ResultDispatcher { | ||||
|   private map: Map< | ||||
|     number | string, | ||||
|     ((value: Result<unknown, unknown>) => void) | true // `true` = a call is generated, but the promise not created
 | ||||
|   >; | ||||
|   private nextId: number = Number.MIN_SAFE_INTEGER; | ||||
| 
 | ||||
|   constructor() { | ||||
|     this.map = new Map(); | ||||
|   } | ||||
| 
 | ||||
|   private rollId() { | ||||
|     let id = 0; | ||||
|     while (this.map.get((id = this.nextId++))) { | ||||
|       if (this.nextId >= Number.MAX_SAFE_INTEGER) { | ||||
|         this.nextId = Number.MIN_SAFE_INTEGER; | ||||
|       } | ||||
|     } | ||||
|     return id; | ||||
|   } | ||||
| 
 | ||||
|   createCall<T>( | ||||
|     method: string, | ||||
|     params: T, | ||||
|   ): [Promise<Call<T>>, Promise<Result<unknown, unknown>>] { | ||||
|     const id = this.rollId(); | ||||
|     const p = new Promise<Result<unknown, unknown>>((resolve) => | ||||
|       this.map.set(id, resolve), | ||||
|     ); | ||||
|     this.map.set(id, true); | ||||
|     const call: Call<T> = { | ||||
|       jsonrpc: "2.0", | ||||
|       id, | ||||
|       method, | ||||
|       params, | ||||
|     }; | ||||
| 
 | ||||
|     return [ | ||||
|       new Promise((resolve) => { | ||||
|         const waitUntilTheIdSet = () => { | ||||
|           // We must do this check to make sure the id is set before the call made.
 | ||||
|           // or the dispatching may lost the callback
 | ||||
|           if (this.map.get(id)) { | ||||
|             resolve(call); | ||||
|           } else { | ||||
|             setTimeout(waitUntilTheIdSet, 0); | ||||
|           } | ||||
|         }; | ||||
| 
 | ||||
|         waitUntilTheIdSet(); | ||||
|       }), | ||||
|       p, | ||||
|     ]; | ||||
|   } | ||||
| 
 | ||||
|   dispatch(id: string | number, message: Result<unknown, unknown>) { | ||||
|     { | ||||
|       const callback = this.map.get(id); | ||||
|       if (!callback) return; | ||||
|       if (typeof callback !== "boolean") { | ||||
|         callback(message); | ||||
|         this.map.delete(id); | ||||
|         return Promise.resolve(); | ||||
|       } | ||||
|     } | ||||
| 
 | ||||
|     return new Promise<void>((resolve) => { | ||||
|       let retried = 0; | ||||
| 
 | ||||
|       const checkAndDispatch = () => { | ||||
|         const callback = this.map.get(id); | ||||
|         if (typeof callback !== "boolean") { | ||||
|           callback(message); | ||||
|           this.map.delete(id); | ||||
|           resolve(); | ||||
|           return; | ||||
|         } | ||||
|         setTimeout(checkAndDispatch, 0); | ||||
|         if (++retried > 3) { | ||||
|           console.warn( | ||||
|             `retried ${retried} time(s) but the callback is still disappeared, id is "${id}"`, | ||||
|           ); | ||||
|         } | ||||
|       }; | ||||
| 
 | ||||
|       // start the loop
 | ||||
|       checkAndDispatch(); | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   createTypedCall< | ||||
|     K extends keyof RequestParams, | ||||
|     P extends RequestParams[K], | ||||
|     R extends ResponseResults[K], | ||||
|     E extends ResponseErrorDatas[K], | ||||
|   >(method: K, params: P): [Promise<Call<P>>, Promise<Result<R, E>>] { | ||||
|     return this.createCall(method, params) as [ | ||||
|       Promise<Call<P>>, | ||||
|       Promise<Result<R, E>>, | ||||
|     ]; | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| interface RequestParams { | ||||
|   ping: void; | ||||
| } | ||||
| 
 | ||||
| interface ResponseResults { | ||||
|   ping: void; | ||||
| } | ||||
| 
 | ||||
| interface ResponseErrorDatas { | ||||
|   [key: string]: void; | ||||
| } | ||||
| 
 | ||||
| export type AnyCall<K extends keyof RequestParams = keyof RequestParams> = | ||||
|   JSONRPC & { | ||||
|     method: K; | ||||
|     params: RequestParams[K]; | ||||
|   }; | ||||
							
								
								
									
										5
									
								
								src/serviceworker/tsconfig.json
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								src/serviceworker/tsconfig.json
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,5 @@ | |||
| { | ||||
|   "compilerOptions": { | ||||
|     "lib": ["ESNext", "WebWorker"], | ||||
|   }, | ||||
| } | ||||
|  | @ -45,6 +45,7 @@ import { | |||
| } from "../platform/i18n.jsx"; | ||||
| import { type Template } from "@solid-primitives/i18n"; | ||||
| import BottomSheet from "../material/BottomSheet.jsx"; | ||||
| import { useServiceWorker } from "../platform/host.js"; | ||||
| 
 | ||||
| type Strings = { | ||||
|   ["lang.auto"]: Template<{ detected: string }>; | ||||
|  | @ -60,9 +61,7 @@ const Settings: ParentComponent = (props) => { | |||
|   ); | ||||
|   const navigate = useNavigate(); | ||||
|   const settings$ = useStore($settings); | ||||
|   const { | ||||
|     needRefresh: [needRefresh], | ||||
|   } = useRegisterSW(); | ||||
|   const { needRefresh, offlineReady } = useServiceWorker(); | ||||
|   const dateFnLocale = useDateFnLocale(); | ||||
| 
 | ||||
|   const [profiles] = useSignedInProfiles(); | ||||
|  | @ -236,6 +235,16 @@ const Settings: ParentComponent = (props) => { | |||
|             </Show> | ||||
|           </ListItem> | ||||
|           <Divider /> | ||||
|           <ListItem> | ||||
|             <ListItemText secondary={ | ||||
|               offlineReady() | ||||
|                 ? t("availability.offline") | ||||
|                 : t("availability.online") | ||||
|             }> | ||||
|               {t("availability")} | ||||
|             </ListItemText> | ||||
|           </ListItem> | ||||
|           <Divider /> | ||||
|         </li> | ||||
|       </List> | ||||
|     </Scaffold> | ||||
|  |  | |||
|  | @ -32,5 +32,9 @@ | |||
|   "motions.gifs": "GIFs", | ||||
|   "motions.gifs.autoplay": "Auto-play GIFs", | ||||
|   "motions.vids": "Videos", | ||||
|   "motions.vids.autoplay": "Auto-play Videos" | ||||
|   "motions.vids.autoplay": "Auto-play Videos", | ||||
| 
 | ||||
|   "availability": "Availability", | ||||
|   "availability.offline": "Offline ready", | ||||
|   "availability.online": "Online only" | ||||
| } | ||||
|  | @ -32,5 +32,9 @@ | |||
|   "motions.gifs": "动图", | ||||
|   "motions.gifs.autoplay": "自动播放动图", | ||||
|   "motions.vids": "视频", | ||||
|   "motions.vids.autoplay": "自动播放视频" | ||||
|   "motions.vids.autoplay": "自动播放视频", | ||||
| 
 | ||||
|   "availability": "离线可用程度", | ||||
|   "availability.offline": "可以离线使用", | ||||
|   "availability.online": "需要联网使用" | ||||
| } | ||||
|  | @ -24,6 +24,7 @@ import { css } from "solid-styled"; | |||
| import { vibrate } from "../platform/hardware"; | ||||
| import { createTimeSource, TimeSourceProvider } from "../platform/timesrc"; | ||||
| import TootComposer from "./TootComposer"; | ||||
| import { useDocumentTitle } from "../utils"; | ||||
| 
 | ||||
| let cachedEntry: [string, mastodon.v1.Status] | undefined; | ||||
| 
 | ||||
|  | @ -106,8 +107,14 @@ const TootBottomSheet: Component = (props) => { | |||
|     } | ||||
|   }); | ||||
| 
 | ||||
|   useDocumentTitle(() => { | ||||
|     const t = toot()?.reblog ?? toot() | ||||
|     const name = t?.account.displayName ?? "Someone" | ||||
|     return `${name}'s toot` | ||||
|   }) | ||||
| 
 | ||||
|   const tootDisplayName = () => { | ||||
|     const t = toot(); | ||||
|     const t = toot()?.reblog ?? toot(); | ||||
|     if (t) { | ||||
|       return resolveCustomEmoji(t.account.displayName, t.account.emojis); | ||||
|     } | ||||
|  |  | |||
|  | @ -1,18 +1,23 @@ | |||
| import { createRenderEffect, createSignal, onCleanup } from "solid-js"; | ||||
| import { | ||||
|   createRenderEffect, | ||||
|   onCleanup, | ||||
|   type Accessor, | ||||
| } from "solid-js"; | ||||
| 
 | ||||
| export function useDocumentTitle(newTitle?: string) { | ||||
| export function useDocumentTitle(newTitle?: string | Accessor<string>) { | ||||
|   const capturedTitle = document.title; | ||||
|   const [title, setTitle] = createSignal(newTitle ?? capturedTitle); | ||||
| 
 | ||||
|   createRenderEffect(() => { | ||||
|     document.title = title(); | ||||
|     if (newTitle) | ||||
|       document.title = typeof newTitle === "string" ? newTitle : newTitle(); | ||||
|   }); | ||||
| 
 | ||||
|   onCleanup(() => { | ||||
|     document.title = capturedTitle; | ||||
|   }); | ||||
| 
 | ||||
|   return setTitle; | ||||
|   return (x: ((x: string) => string) | string) => | ||||
|     (document.title = typeof x === "string" ? x : x(document.title)); | ||||
| } | ||||
| 
 | ||||
| export function mergeClass(c1: string | undefined, c2: string | undefined) { | ||||
|  |  | |||
|  | @ -16,10 +16,16 @@ export default defineConfig(({ mode }) => ({ | |||
|       }, | ||||
|     }), | ||||
|     VitePWA({ | ||||
|       strategies: "injectManifest", | ||||
|       registerType: "autoUpdate", | ||||
|       devOptions: { | ||||
|         enabled: mode === "staging", | ||||
|         enabled: mode === "staging" || mode === "dev", | ||||
|       }, | ||||
|       srcDir: "src/serviceworker", | ||||
|       filename: "main.ts", | ||||
|       manifest: { | ||||
|         theme_color: "#673ab7" | ||||
|       } | ||||
|     }), | ||||
|     version(), | ||||
|   ], | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue