Compare commits
	
		
			6 commits
		
	
	
		
			3dc2196f1a
			...
			49b7f2891b
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 49b7f2891b | ||
|  | ee39cf8651 | ||
|  | 40a785a538 | ||
|  | dbb93fd76c | ||
|  | eefe2812df | ||
|  | 4fa7449d71 | 
					 5 changed files with 79 additions and 38 deletions
				
			
		|  | @ -1,7 +1,7 @@ | ||||||
| { | { | ||||||
|   "$schema": "https://json.schemastore.org/package", |   "$schema": "https://json.schemastore.org/package", | ||||||
|   "name": "tutu", |   "name": "tutu", | ||||||
|   "version": "1.0.7", |   "version": "1.0.8", | ||||||
|   "description": "", |   "description": "", | ||||||
|   "private": true, |   "private": true, | ||||||
|   "type": "module", |   "type": "module", | ||||||
|  |  | ||||||
|  | @ -51,9 +51,19 @@ const MediaAttachmentGrid: Component<{ | ||||||
|     setViewerIndex(index); |     setViewerIndex(index); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|  |   const columnCount = () => { | ||||||
|  |     if (props.attachments.length === 1) { | ||||||
|  |       return 1; | ||||||
|  |     } else if (props.attachments.length % 2 === 0) { | ||||||
|  |       return 2; | ||||||
|  |     } else { | ||||||
|  |       return 3; | ||||||
|  |     } | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   css` |   css` | ||||||
|     .attachments { |     .attachments { | ||||||
|       column-count: ${(props.attachments.length === 1 ? 1 : 3).toString()}; |       column-count: ${columnCount.toString()}; | ||||||
|     } |     } | ||||||
|   `;
 |   `;
 | ||||||
|   return ( |   return ( | ||||||
|  | @ -104,17 +114,17 @@ const MediaAttachmentGrid: Component<{ | ||||||
|                 /> |                 /> | ||||||
|               ); |               ); | ||||||
|             case "gifv": // Later we can handle the preview
 |             case "gifv": // Later we can handle the preview
 | ||||||
|             return ( |               return ( | ||||||
|               <video |                 <video | ||||||
|                 src={item.url || undefined} |                   src={item.url || undefined} | ||||||
|                 style={style()} |                   style={style()} | ||||||
|                 onLoadedMetadata={[setLoaded, true]} |                   onLoadedMetadata={[setLoaded, true]} | ||||||
|                 autoplay={false} |                   autoplay={false} | ||||||
|                 controls |                   controls | ||||||
|                 playsinline /* or safari on iOS will play in full-screen */ |                   playsinline /* or safari on iOS will play in full-screen */ | ||||||
|                 loop |                   loop | ||||||
|               /> |                 /> | ||||||
|             ); |               ); | ||||||
| 
 | 
 | ||||||
|             case "audio": |             case "audio": | ||||||
|             case "unknown": |             case "unknown": | ||||||
|  |  | ||||||
|  | @ -28,11 +28,13 @@ const PullDownToRefresh: Component<{ | ||||||
|   let rootElement: HTMLDivElement; |   let rootElement: HTMLDivElement; | ||||||
|   const [pullDown, setPullDown] = createSignal(0); |   const [pullDown, setPullDown] = createSignal(0); | ||||||
| 
 | 
 | ||||||
|   const pullDownDistance = () => { |   const stopPos = () => 160; | ||||||
|  | 
 | ||||||
|  |   const indicatorOfsY = () => { | ||||||
|     if (props.loading) { |     if (props.loading) { | ||||||
|       return 140; |       return stopPos() * 0.875; | ||||||
|     } |     } | ||||||
|     return Math.max(Math.min(160, pullDown()), 0); |     return pullDown(); | ||||||
|   }; |   }; | ||||||
| 
 | 
 | ||||||
|   const obvx = createVisibilityObserver({ |   const obvx = createVisibilityObserver({ | ||||||
|  | @ -41,46 +43,66 @@ const PullDownToRefresh: Component<{ | ||||||
| 
 | 
 | ||||||
|   const rootVisible = obvx(() => rootElement); |   const rootVisible = obvx(() => rootElement); | ||||||
| 
 | 
 | ||||||
|  |   createEffect(() => { | ||||||
|  |     if (!rootVisible()) setPullDown(0); | ||||||
|  |   }); | ||||||
|  | 
 | ||||||
|   let released = true; |   let released = true; | ||||||
|   let v = 0; |   let v = 0; | ||||||
|   let lts = -1; |   let lts = -1; | ||||||
|   let ds = 0; |   let ds = 0; | ||||||
|   let holding = false; |   let holding = false; | ||||||
|   const M = 1; |   const K = 10; | ||||||
|   const K = -10; |  | ||||||
|   const D = -10; |  | ||||||
|   const updatePullDown = (ts: number) => { |   const updatePullDown = (ts: number) => { | ||||||
|     released = false; |     released = false; | ||||||
|     try { |     try { | ||||||
|       const x = untrack(pullDown); |       const x = untrack(pullDown); | ||||||
|       const dt = lts !== -1 ? ts - lts : 1 / 60; |       const dt = lts !== -1 ? ts - lts : 1 / 60; | ||||||
|       const fs = (ds / Math.pow(dt, 2)) * M; |       const vspring = holding ? 0 : K * x * dt; | ||||||
|       const fh = (x + ds) * K + D * v; |       v = ds / dt - vspring; | ||||||
|       const f = fs + fh; | 
 | ||||||
|       const a = f / M; |       setPullDown(Math.max(Math.min(x + v * dt, stopPos()), 0)); | ||||||
|       v += a * dt; | 
 | ||||||
|       if (holding && v < 0) { |  | ||||||
|         v = 0 |  | ||||||
|       } |  | ||||||
|       setPullDown(x + v * dt); |  | ||||||
|       if (Math.abs(x) > 1 || Math.abs(v) > 1) { |       if (Math.abs(x) > 1 || Math.abs(v) > 1) { | ||||||
|         requestAnimationFrame(updatePullDown); |         requestAnimationFrame(updatePullDown); | ||||||
|       } else { |       } else { | ||||||
|         v = 0; |         v = 0; | ||||||
|         lts = -1; |         lts = -1; | ||||||
|       } |       } | ||||||
|       if (!holding && untrack(pullDownDistance) >= 160 && !props.loading && props.onRefresh) { | 
 | ||||||
|         setTimeout(props.onRefresh, 0) |       if ( | ||||||
|  |         !holding && | ||||||
|  |         untrack(pullDown) >= stopPos() && | ||||||
|  |         !props.loading && | ||||||
|  |         props.onRefresh | ||||||
|  |       ) { | ||||||
|  |         setTimeout(props.onRefresh, 0); | ||||||
|       } |       } | ||||||
|     } finally { |     } finally { | ||||||
|       ds = 0; |       ds = 0; | ||||||
|       released = true; |       released = true; | ||||||
|     } |     } | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|  |   let wheelTimeout: ReturnType<typeof setTimeout> | undefined; | ||||||
|  | 
 | ||||||
|  |   const onWheelNotUpdated = () => { | ||||||
|  |     wheelTimeout = undefined; | ||||||
|  |     holding = false; | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|   const handleLinkedWheel = (event: WheelEvent) => { |   const handleLinkedWheel = (event: WheelEvent) => { | ||||||
|     const scrollTop = (event.target as HTMLElement).scrollTop; |     const scrollTop = (event.target as HTMLElement).scrollTop; | ||||||
|     if (scrollTop >= 0 && scrollTop < 1) { |     if (scrollTop >= 0 && scrollTop < 1) { | ||||||
|       ds = -(event.deltaY / window.devicePixelRatio / 4); |       const d = untrack(pullDown); | ||||||
|  |       if (event.deltaY <= 0 || d > 0) event.preventDefault(); | ||||||
|  |       ds = -(event.deltaY / window.devicePixelRatio / 2); | ||||||
|  |       holding = d < stopPos(); | ||||||
|  |       if (wheelTimeout) { | ||||||
|  |         clearTimeout(wheelTimeout); | ||||||
|  |       } | ||||||
|  |       wheelTimeout = setTimeout(onWheelNotUpdated, 200); | ||||||
|  | 
 | ||||||
|       if (released) { |       if (released) { | ||||||
|         released = false; |         released = false; | ||||||
|         requestAnimationFrame(updatePullDown); |         requestAnimationFrame(updatePullDown); | ||||||
|  | @ -107,11 +129,13 @@ const PullDownToRefresh: Component<{ | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|     const item = event.targetTouches.item(0)!; |     const item = event.targetTouches.item(0)!; | ||||||
|  |     if (untrack(pullDown) > 0) event.preventDefault(); | ||||||
|     if (lastTouchId && item.identifier !== lastTouchId) { |     if (lastTouchId && item.identifier !== lastTouchId) { | ||||||
|       lastTouchId = undefined; |       lastTouchId = undefined; | ||||||
|       lastTouchScreenY = 0; |       lastTouchScreenY = 0; | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     holding = true; |     holding = true; | ||||||
|     if (lastTouchScreenY !== 0) { |     if (lastTouchScreenY !== 0) { | ||||||
|       ds = item.screenY - lastTouchScreenY; |       ds = item.screenY - lastTouchScreenY; | ||||||
|  | @ -127,7 +151,11 @@ const PullDownToRefresh: Component<{ | ||||||
|     lastTouchId = undefined; |     lastTouchId = undefined; | ||||||
|     lastTouchScreenY = 0; |     lastTouchScreenY = 0; | ||||||
|     holding = false; |     holding = false; | ||||||
|     if (untrack(pullDownDistance) >= 160 && !props.loading && props.onRefresh) { |     if ( | ||||||
|  |       untrack(indicatorOfsY) >= stopPos() && | ||||||
|  |       !props.loading && | ||||||
|  |       props.onRefresh | ||||||
|  |     ) { | ||||||
|       setTimeout(props.onRefresh, 0); |       setTimeout(props.onRefresh, 0); | ||||||
|     } else { |     } else { | ||||||
|       if (released) { |       if (released) { | ||||||
|  | @ -169,14 +197,14 @@ const PullDownToRefresh: Component<{ | ||||||
|       aspect-ratio: 1/1; |       aspect-ratio: 1/1; | ||||||
|       width: 2rem; |       width: 2rem; | ||||||
|       color: var(--tutu-color-primary); |       color: var(--tutu-color-primary); | ||||||
|       transform: translateY(${`${pullDownDistance() - 2}px`}); |       transform: translateY(${`${indicatorOfsY() - 2}px`}); | ||||||
|       will-change: transform; |       will-change: transform; | ||||||
|       z-index: var(--tutu-zidx-nav); |       z-index: var(--tutu-zidx-nav); | ||||||
|       background-color: var(--tutu-color-surface); |       background-color: var(--tutu-color-surface); | ||||||
| 
 | 
 | ||||||
|       > :global(.refresh-icon) { |       > :global(.refresh-icon) { | ||||||
|         transform: rotate( |         transform: rotate( | ||||||
|           ${`${((pullDownDistance() / 160) * 180).toString()}deg`} |           ${`${((indicatorOfsY() / 160) * 180).toString()}deg`} | ||||||
|         ); |         ); | ||||||
|         will-change: transform; |         will-change: transform; | ||||||
|       } |       } | ||||||
|  |  | ||||||
|  | @ -1,5 +1,6 @@ | ||||||
| import { | import { | ||||||
|   createEffect, |   createEffect, | ||||||
|  |   createMemo, | ||||||
|   createSignal, |   createSignal, | ||||||
|   createUniqueId, |   createUniqueId, | ||||||
|   onMount, |   onMount, | ||||||
|  | @ -212,6 +213,10 @@ const TootComposer: Component<{ | ||||||
|   const [langPickerOpen, setLangPickerOpen] = createSignal(false); |   const [langPickerOpen, setLangPickerOpen] = createSignal(false); | ||||||
|   const appLanguage = useLanguage(); |   const appLanguage = useLanguage(); | ||||||
| 
 | 
 | ||||||
|  |   const randomPlaceholder = createMemo(() => | ||||||
|  |     randomChoose(Math.random(), ["What's happening?", "What do your think?"]), | ||||||
|  |   ); | ||||||
|  | 
 | ||||||
|   createEffect(() => { |   createEffect(() => { | ||||||
|     const lang = appLanguage().split("-")[0]; |     const lang = appLanguage().split("-")[0]; | ||||||
|     setLanguage(lang); |     setLanguage(lang); | ||||||
|  | @ -311,10 +316,7 @@ const TootComposer: Component<{ | ||||||
|           placeholder={ |           placeholder={ | ||||||
|             props.replyToDisplayName |             props.replyToDisplayName | ||||||
|               ? `Reply to ${props.replyToDisplayName}...` |               ? `Reply to ${props.replyToDisplayName}...` | ||||||
|               : randomChoose(Math.random(), [ |               : randomPlaceholder() | ||||||
|                   "What's happening?", |  | ||||||
|                   "What do your think?", |  | ||||||
|                 ]) |  | ||||||
|           } |           } | ||||||
|           style={{ width: "100%", border: "none" }} |           style={{ width: "100%", border: "none" }} | ||||||
|           disabled={sending()} |           disabled={sending()} | ||||||
|  |  | ||||||
|  | @ -3,6 +3,7 @@ | ||||||
|   --card-gut: 16px; |   --card-gut: 16px; | ||||||
|   --toot-avatar-size: 40px; |   --toot-avatar-size: 40px; | ||||||
|   margin-block: 0; |   margin-block: 0; | ||||||
|  |   position: relative; | ||||||
| 
 | 
 | ||||||
|   &.toot { |   &.toot { | ||||||
|     /* fix composition ordering: I think the css module processor should aware the overriding and behaves, but no */ |     /* fix composition ordering: I think the css module processor should aware the overriding and behaves, but no */ | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue