# VRM to Spritesheet - https://www.youtube.com/watch?v=z_oDpQ4vzNA - https://hackmd.io/@XR/avatarlod#Webaverse-Pipeline Sprites can help with optimization, like generating imposters that improve concurrent user count with assets that look like the original custom avatar. ![image](https://hackmd.io/_uploads/r16YVtMja.png) ## Character Studio Pipeline - https://github.com/m3-org/CharacterStudio using character studio to generate the sprites, example animation / manifest file ![joy2](https://hackmd.io/_uploads/S1yM4Kzia.gif) ![image](https://hackmd.io/_uploads/HJqMEtzsp.png) ![walk](https://hackmd.io/_uploads/BybivYGj6.gif) It works with different sized models as well ![walk2](https://hackmd.io/_uploads/rJH9TKfi6.gif) --- ### r3f-sprite-animator - https://codesandbox.io/p/sandbox/r3f-sprite-animator-s12ijv?file=%2Fpublic%2Fboy_hash.json%3A1%2C1-204%2C2 ```json! {"frames": { "Celebration_000": { "frame": {"x":1,"y":1,"w":560,"h":440}, "rotated": false, "trimmed": false, "spriteSourceSize": {"x":0,"y":0,"w":560,"h":440}, "sourceSize": {"w":560,"h":440} }, "Celebration_001": { "frame": {"x":1,"y":443,"w":560,"h":440}, "rotated": false, "trimmed": false, "spriteSourceSize": {"x":0,"y":0,"w":560,"h":440}, "sourceSize": {"w":560,"h":440} }, "Celebration_002": { "frame": {"x":1,"y":885,"w":560,"h":440}, "rotated": false, "trimmed": false, "spriteSourceSize": {"x":0,"y":0,"w":560,"h":440}, "sourceSize": {"w":560,"h":440} }, "Celebration_003": { "frame": {"x":1,"y":1327,"w":560,"h":440}, "rotated": false, "trimmed": false, "spriteSourceSize": {"x":0,"y":0,"w":560,"h":440}, "sourceSize": {"w":560,"h":440} }, ``` https://twitter.com/antpb/status/1752868882291790262 > The biggest thing we need is a list of standard animations. My current implementation is > - idle > - idleBackward > - walk > - walkBackward > >For jumps you could use the walk animation clamped to a frame where the leg is forward. These are all very old problems though so I’m sure there’s some very standard way that is a one file sheet. The names matter most because the implementations could just not care what the animations do and play on the state associated like walk --- ## Notes - https://community.khronos.org/t/adding-billboard-sprites-to-gltf/106664 - https://x.com/UX3DGpuSoftware/status/1461706770309689348?s=20 - https://registry.khronos.org/glTF/specs/2.0/glTF-2.0.html#morph-targets ![image](https://hackmd.io/_uploads/Bkr6LtGjT.png) ```json! { "Apple": { "frame": {"x":292,"y":304,"w":60,"h":61}, "rotated": false, "trimmed": true, "spriteSourceSize": {"x":23,"y":29,"w":60,"h":61}, "sourceSize": {"w":90,"h":90}, "pivot": {"x":0.5,"y":0.5} }, ... } ``` snippet from Eidur in Webgamedev discord > I'm implementing vanilla three and threlte components and utilities at the same time. ```javascript! export type SpritesheetFormat = { frames: [x: number, y: number, w: number, h: number][]; animations: Record<string, [frameId: number, duration: number][]>; sheetSize: [w: number, h: number]; animationLengths: number[]; }; ``` --- ## glTF extension notes rough sketch by lasershark ```json! { "extensionsUsed": ["m3_spritesheet_animations"], "images": [ { "uri": "..." } ], "extensions": { "m3_spritesheet_animations": { "spritesheets": [ { "image": 0, "name": "Sprite Animations", "dimensions": { "width": 1024, "height": 1024, "framesH": 4, "framesV": 4 }, "frameRate": 12, "animations": [ { "name": "walking", "startFrame": 0, "endFrame": 3, "loop": true }, { "name": "jumping", "startFrame": 4, "endFrame": 7, "loop": false } ] } ] } } } ```