UNPKG

create-expo-cljs-app

Version:

Create a react native application with Expo and Shadow-CLJS!

142 lines (127 loc) 3.54 kB
import { defineAnimation } from './util'; import { Animation, AnimationCallback, AnimationObject, AnimatableValue, Timestamp, } from './commonTypes'; import { Platform } from 'react-native'; interface DecayConfig { deceleration?: number; velocityFactor?: number; clamp?: number[]; velocity?: number; } interface DefaultDecayConfig { deceleration: number; velocityFactor: number; clamp?: number[]; velocity: number; } export interface DecayAnimation extends Animation<DecayAnimation> { lastTimestamp: Timestamp; startTimestamp: Timestamp; initialVelocity: number; velocity: number; current: AnimatableValue; } export interface InnerDecayAnimation extends Omit<DecayAnimation, 'current'>, AnimationObject { current: number; } export function withDecay( userConfig: DecayConfig, callback?: AnimationCallback ): Animation<DecayAnimation> { 'worklet'; return defineAnimation<DecayAnimation>(0, () => { 'worklet'; const config: DefaultDecayConfig = { deceleration: 0.998, velocityFactor: Platform.OS !== 'web' ? 1 : 1000, velocity: 0, }; if (userConfig) { Object.keys(userConfig).forEach( (key) => ((config as any)[key] = userConfig[key as keyof typeof userConfig]) ); } const VELOCITY_EPS = Platform.OS !== 'web' ? 1 : 1 / 20; const SLOPE_FACTOR = 0.1; function decay(animation: InnerDecayAnimation, now: number): boolean { const { lastTimestamp, startTimestamp, initialVelocity, current, velocity, } = animation; const deltaTime = Math.min(now - lastTimestamp, 64); const v = velocity * Math.exp( -(1 - config.deceleration) * (now - startTimestamp) * SLOPE_FACTOR ); animation.current = current + (v * config.velocityFactor * deltaTime) / 1000; // /1000 because time is in ms not in s animation.velocity = v; animation.lastTimestamp = now; if (config.clamp) { if (initialVelocity < 0 && animation.current <= config.clamp[0]) { animation.current = config.clamp[0]; return true; } else if ( initialVelocity > 0 && animation.current >= config.clamp[1] ) { animation.current = config.clamp[1]; return true; } } return Math.abs(v) < VELOCITY_EPS; } function validateConfig(): void { if (config.clamp) { if (!Array.isArray(config.clamp)) { throw Error( `config.clamp must be an array but is ${typeof config.clamp}` ); } if (config.clamp.length !== 2) { throw Error( `clamp array must contain 2 items but is given ${config.clamp.length}` ); } } if (config.velocityFactor <= 0) { throw Error( `config.velocityFactor must be greather then 0 but is ${config.velocityFactor}` ); } } function onStart( animation: DecayAnimation, value: number, now: Timestamp ): void { animation.current = value; animation.lastTimestamp = now; animation.startTimestamp = now; animation.initialVelocity = config.velocity; validateConfig(); } return { onFrame: decay, onStart, callback, velocity: config.velocity ?? 0, initialVelocity: 0, current: 0, lastTimestamp: 0, startTimestamp: 0, } as DecayAnimation; }); }