├── public ├── robots.txt ├── favicon.ico ├── favicon.webp ├── logo192.png ├── logo512.png ├── manifest.json └── index.html ├── .idea ├── .gitignore ├── vcs.xml ├── prettier.xml ├── modules.xml ├── inspectionProfiles │ └── Project_Default.xml └── RM10k.iml ├── src ├── assets │ ├── base.json │ └── test.json ├── hooks │ ├── useGetLottie.jsx │ └── useToast.jsx ├── theme.js ├── components │ ├── Switch.jsx │ ├── Input.jsx │ ├── AnimationsOptions │ │ └── index.jsx │ ├── Slider.jsx │ ├── NumberInput.jsx │ ├── Layers.jsx │ ├── AlertDialog.jsx │ ├── LayerOptions │ │ └── index.jsx │ └── BoardingModal.jsx ├── index.js ├── App.jsx ├── store.js └── parser.js ├── .gitignore ├── README.md └── package.json /public/robots.txt: -------------------------------------------------------------------------------- 1 | # https://www.robotstxt.org/robotstxt.html 2 | User-agent: * 3 | Disallow: 4 | -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raoufswe/images-to-lottie-editor/HEAD/public/favicon.ico -------------------------------------------------------------------------------- /public/favicon.webp: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raoufswe/images-to-lottie-editor/HEAD/public/favicon.webp -------------------------------------------------------------------------------- /public/logo192.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raoufswe/images-to-lottie-editor/HEAD/public/logo192.png -------------------------------------------------------------------------------- /public/logo512.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/raoufswe/images-to-lottie-editor/HEAD/public/logo512.png -------------------------------------------------------------------------------- /.idea/.gitignore: -------------------------------------------------------------------------------- 1 | # Default ignored files 2 | /shelf/ 3 | /workspace.xml 4 | # Editor-based HTTP Client requests 5 | /httpRequests/ 6 | -------------------------------------------------------------------------------- /.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /src/assets/base.json: -------------------------------------------------------------------------------- 1 | { 2 | "v": "5.5.2", 3 | "fr": 50, 4 | "ip": 0, 5 | "op": 50, 6 | "w": 500, 7 | "h": 500, 8 | "nm": "Base", 9 | "ddd": 0, 10 | "assets": [], 11 | "layers": [], 12 | "markers": [] 13 | } 14 | -------------------------------------------------------------------------------- /.idea/prettier.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 7 | -------------------------------------------------------------------------------- /src/hooks/useGetLottie.jsx: -------------------------------------------------------------------------------- 1 | import { useMutation } from "react-query" 2 | import axios from "axios" 3 | 4 | export default function useGetLottie() { 5 | return useMutation(async (url) => { 6 | const response = await axios.get(url) 7 | return response.data 8 | }) 9 | } 10 | -------------------------------------------------------------------------------- /.idea/modules.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | -------------------------------------------------------------------------------- /.idea/inspectionProfiles/Project_Default.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 6 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. 2 | 3 | # dependencies 4 | /node_modules 5 | /.pnp 6 | .pnp.js 7 | 8 | # testing 9 | /coverage 10 | 11 | # production 12 | /build 13 | 14 | # misc 15 | .DS_Store 16 | .env.local 17 | .env.development.local 18 | .env.test.local 19 | .env.production.local 20 | 21 | npm-debug.log* 22 | yarn-debug.log* 23 | yarn-error.log* 24 | -------------------------------------------------------------------------------- /src/theme.js: -------------------------------------------------------------------------------- 1 | import { extendTheme } from "@chakra-ui/react" 2 | 3 | export default extendTheme({ 4 | styles: { 5 | global: { 6 | body: { overflow: "hidden" } 7 | } 8 | }, 9 | fonts: { 10 | body: "'Montserrat', sans-serif", 11 | heading: "'Montserrat', sans-serif", 12 | mono: "'Montserrat', sans-serif" 13 | }, 14 | config: { 15 | initialColorMode: "dark", 16 | useSystemColorMode: false 17 | } 18 | }) 19 | -------------------------------------------------------------------------------- /src/hooks/useToast.jsx: -------------------------------------------------------------------------------- 1 | import { useToast } from "@chakra-ui/react" 2 | 3 | export function useErrorToast() { 4 | const toast = useToast() 5 | return ({ description = "Something went wrong 😥", position = "bottom" }) => toast({ description, status: "error", position }) 6 | } 7 | 8 | export function useSuccessToast() { 9 | const toast = useToast() 10 | return ({ description = "All is set 🎉", position = "bottom" }) => toast({ description, status: "success", position }) 11 | } 12 | -------------------------------------------------------------------------------- /.idea/RM10k.iml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | -------------------------------------------------------------------------------- /src/components/Switch.jsx: -------------------------------------------------------------------------------- 1 | import { Switch as ChakaraSwitch, FormControl, FormLabel } from "@chakra-ui/react" 2 | 3 | export default function Switch({ id, label, isChecked, onChange, defaultChecked }) { 4 | return ( 5 | 6 | 7 | {label} 8 | 9 | 10 | 11 | ) 12 | } 13 | -------------------------------------------------------------------------------- /src/components/Input.jsx: -------------------------------------------------------------------------------- 1 | import React from "react" 2 | import { Text, Input as ChakraInput, FormErrorMessage, FormControl } from "@chakra-ui/react" 3 | 4 | export default function Input({ label, placeholder, value, onChange, error, size, variant }) { 5 | return ( 6 | 7 | 8 | {label} 9 | 10 | 11 | {error} 12 | 13 | ) 14 | } 15 | -------------------------------------------------------------------------------- /public/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "short_name": "React App", 3 | "name": "Create React App Sample", 4 | "icons": [ 5 | { 6 | "src": "favicon.ico", 7 | "sizes": "64x64 32x32 24x24 16x16", 8 | "type": "image/x-icon" 9 | }, 10 | { 11 | "src": "logo192.png", 12 | "type": "image/png", 13 | "sizes": "192x192" 14 | }, 15 | { 16 | "src": "logo512.png", 17 | "type": "image/png", 18 | "sizes": "512x512" 19 | } 20 | ], 21 | "start_url": ".", 22 | "display": "standalone", 23 | "theme_color": "#000000", 24 | "background_color": "#ffffff" 25 | } 26 | -------------------------------------------------------------------------------- /src/components/AnimationsOptions/index.jsx: -------------------------------------------------------------------------------- 1 | import { Flex, Text, Button } from "@chakra-ui/react" 2 | import { animations } from "../../parser" 3 | import useStore from "../../store" 4 | 5 | export default function AnimationsOptions() { 6 | const { setAnimation } = useStore() 7 | return ( 8 | <> 9 | Animations 10 | 11 | {animations.map((item, index) => ( 12 | 15 | ))} 16 | 17 | 18 | ) 19 | } 20 | -------------------------------------------------------------------------------- /src/components/Slider.jsx: -------------------------------------------------------------------------------- 1 | import { Flex, Text, Slider as ChakaraSlider, SliderTrack, SliderFilledTrack, SliderThumb } from "@chakra-ui/react" 2 | 3 | export default function Slider({ label, onChange, defaultValue }) { 4 | return ( 5 | 6 | 7 | {label} {defaultValue}% 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | ) 17 | } 18 | -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import React from "react" 2 | import ReactDOM from "react-dom" 3 | import { QueryClient, QueryClientProvider } from "react-query" 4 | import { ChakraProvider, ColorModeScript } from "@chakra-ui/react" 5 | import App from "./App" 6 | import theme from "./theme" 7 | 8 | const queryClient = new QueryClient() 9 | 10 | ReactDOM.render( 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | , 19 | document.getElementById("root") 20 | ) 21 | -------------------------------------------------------------------------------- /src/components/NumberInput.jsx: -------------------------------------------------------------------------------- 1 | import { 2 | Flex, 3 | Text, 4 | NumberInput as ChakaraNumberInput, 5 | NumberInputField, 6 | NumberInputStepper, 7 | NumberIncrementStepper, 8 | NumberDecrementStepper 9 | } from "@chakra-ui/react" 10 | 11 | export default function NumberInput({ label, defaultValue, onChange, value, ...props}) { 12 | return ( 13 | 14 | 15 | {label} 16 | 17 | 18 | 19 | 20 | 21 | 22 | 23 | 24 | 25 | ) 26 | } 27 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | 2 |

3 | isthereacovidcureyet 4 |

5 |

6 | Images To Lottie editor 7 |

8 | 9 | A simple web-based images to lottie editor created to get a deeper understanding of how lotties work. 10 | Learn more about Lottie animation [here](https://airbnb.design/lottie/) 11 | 12 | ## Features 13 | 14 | - Create new lottie using images 15 | - Edit existing lottie 16 | - Change opacity of layers 17 | - Change height and width of the asset 18 | - Change positioning of the layer 19 | 20 | and more to come <3 21 | 22 | ## 🚀 Quick start 23 | 24 | 1. **Clone the repository.** 25 | 26 | ```shell 27 | # https://github.com/raoufswe/images-to-lottie-editor.git 28 | 29 | ``` 30 | 31 | 1. **Start developing.** 32 | 33 | ```shell 34 | cd images-to-lottie-editor/ 35 | npm install 36 | npm run start 37 | ``` 38 | 39 | ## Technology Stack and Tools :calling: 40 | 41 | - React ⚛️ 42 | - Chakra UI 🌄 43 | - Zustand 🗄️ 44 | - React Query ⚛️ 45 | --- 46 | -------------------------------------------------------------------------------- /src/components/Layers.jsx: -------------------------------------------------------------------------------- 1 | import { Box, Flex, Text, Heading } from "@chakra-ui/react" 2 | import useStore from "../store" 3 | import { Player } from "@lottiefiles/react-lottie-player" 4 | 5 | export default function Layers() { 6 | const { lottieFile, selectedLayer, setSelectedLayer } = useStore() 7 | let clonedLottieFile = JSON.parse(JSON.stringify(lottieFile)) 8 | 9 | return ( 10 | 11 | 12 | Layers 13 | 14 | {clonedLottieFile.layers.map((layer) => ( 15 | setSelectedLayer(layer)} 18 | alignItems="center" 19 | cursor="pointer" 20 | width="100%" 21 | py="2" 22 | px="4" 23 | mb="2" 24 | bg={selectedLayer?.uuid === layer.uuid ? "teal" : ""} 25 | _hover={{ background: "teal" }} 26 | borderRadius="4px" 27 | > 28 | 29 | i.uuid === layer.uuid) }} /> 30 | 31 | 32 | {layer.nm} 33 | 34 | 35 | ))} 36 | 37 | ) 38 | } 39 | -------------------------------------------------------------------------------- /src/components/AlertDialog.jsx: -------------------------------------------------------------------------------- 1 | import { useState, useRef } from "react" 2 | import { 3 | AlertDialog as ChakaraDeleteDialog, 4 | AlertDialogBody, 5 | AlertDialogFooter, 6 | AlertDialogHeader, 7 | AlertDialogContent, 8 | AlertDialogOverlay, 9 | Button 10 | } from "@chakra-ui/react" 11 | 12 | export default function AlertDialog({ title, onDelete }) { 13 | const [isOpen, setIsOpen] = useState(false) 14 | const onClose = () => setIsOpen(false) 15 | const cancelRef = useRef() 16 | 17 | return ( 18 | <> 19 | 22 | 23 | 24 | 25 | 26 | {title} 27 | 28 | Are you sure? You can't undo this action afterwards. 29 | 30 | 33 | 43 | 44 | 45 | 46 | 47 | 48 | ) 49 | } 50 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "lottie", 3 | "version": "0.1.0", 4 | "private": true, 5 | "dependencies": { 6 | "@chakra-ui/icons": "^1.0.5", 7 | "@chakra-ui/react": "^1.3.3", 8 | "@emotion/react": "^11.1.5", 9 | "@emotion/styled": "^11.1.5", 10 | "@lottiefiles/react-lottie-player": "^3.0.1", 11 | "@testing-library/jest-dom": "^5.11.4", 12 | "@testing-library/react": "^11.1.0", 13 | "@testing-library/user-event": "^12.1.10", 14 | "axios": "^0.21.1", 15 | "filepond": "^4.25.1", 16 | "filepond-plugin-file-encode": "^2.1.9", 17 | "filepond-plugin-image-exif-orientation": "^1.0.9", 18 | "filepond-plugin-image-preview": "^4.6.5", 19 | "framer-motion": "^3.9.1", 20 | "fuse.js": "^6.4.6", 21 | "react": "^17.0.1", 22 | "react-dom": "^17.0.1", 23 | "react-file-base64": "^1.0.3", 24 | "react-filepond": "^7.1.1", 25 | "react-images-upload": "^1.2.8", 26 | "react-query": "^3.12.0", 27 | "react-scripts": "4.0.3", 28 | "react-tracked": "^1.6.5", 29 | "uuid-random": "^1.3.2", 30 | "web-vitals": "^1.0.1", 31 | "zustand": "^3.3.3" 32 | }, 33 | "scripts": { 34 | "start": "react-scripts start", 35 | "build": "react-scripts build", 36 | "test": "react-scripts test", 37 | "eject": "react-scripts eject" 38 | }, 39 | "eslintConfig": { 40 | "extends": [ 41 | "react-app", 42 | "react-app/jest" 43 | ] 44 | }, 45 | "browserslist": { 46 | "production": [ 47 | ">0.2%", 48 | "not dead", 49 | "not op_mini all" 50 | ], 51 | "development": [ 52 | "last 1 chrome version", 53 | "last 1 firefox version", 54 | "last 1 safari version" 55 | ] 56 | } 57 | } 58 | -------------------------------------------------------------------------------- /src/components/LayerOptions/index.jsx: -------------------------------------------------------------------------------- 1 | import { Flex, Box, Divider } from "@chakra-ui/react" 2 | import useStore from "../../store" 3 | import NumberInput from "../NumberInput" 4 | import Switch from "../Switch" 5 | import Slider from "../Slider" 6 | import AlertDialog from "../AlertDialog" 7 | import AnimationsOptions from "../AnimationsOptions" 8 | 9 | export default function LayerOptions() { 10 | const { 11 | opacity, 12 | w, 13 | h, 14 | x, 15 | y, 16 | selectedLayer, 17 | hideLayer, 18 | showLayer, 19 | selectedAsset, 20 | setOpacity, 21 | setAssetWidth, 22 | setAssetHeight, 23 | setAssetXPosition, 24 | setAssetYPosition, 25 | deleteLayer, 26 | visibleLayers 27 | } = useStore() 28 | 29 | if (!selectedLayer) return null 30 | return ( 31 | 32 | layer.uuid === selectedLayer.uuid)} 37 | onChange={({ target }) => (target.checked ? showLayer() : hideLayer())} 38 | /> 39 | 40 | {selectedAsset && ( 41 | <> 42 | 43 | 44 | 45 | 46 | 47 | )} 48 | 49 | 50 | 51 | 52 | 53 | 54 | ) 55 | } 56 | -------------------------------------------------------------------------------- /public/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 16 | 17 | 26 | Images to Lottie Editor 27 | 28 | 29 | 30 |
31 | 41 | 42 | 43 | -------------------------------------------------------------------------------- /src/App.jsx: -------------------------------------------------------------------------------- 1 | import { Flex, Grid, Heading, Divider, Button } from "@chakra-ui/react" 2 | import { Player, Controls } from "@lottiefiles/react-lottie-player" 3 | import FileBase64 from "react-file-base64" 4 | import useStore from "./store" 5 | import NumberInput from "./components/NumberInput" 6 | import BoardingModal from "./components/BoardingModal" 7 | import Layers from "./components/Layers" 8 | import LayerOptions from "./components/LayerOptions" 9 | 10 | function App() { 11 | const { setImage, lottieFile } = useStore() 12 | if (!lottieFile) return 13 | 14 | return ( 15 | 16 | 17 | 18 | 19 | Upload your images 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | 34 | 35 | 36 | ) 37 | } 38 | 39 | function MainPlayer() { 40 | const { lottieFile, visibleLayers } = useStore() 41 | let clonedVisibleLayers = JSON.parse(JSON.stringify(lottieFile.layers.filter((i) => visibleLayers.map((i) => i.nm).includes(i.nm)))) 42 | return ( 43 | 44 | 45 | 46 | ) 47 | } 48 | 49 | function LottieFileConfigs() { 50 | const { lottieFile, setFrameRate } = useStore() 51 | return ( 52 | <> 53 | 54 | 55 | 58 | 59 | 60 | ) 61 | } 62 | 63 | export default App 64 | -------------------------------------------------------------------------------- /src/store.js: -------------------------------------------------------------------------------- 1 | import baseLottie from "./assets/base.json" 2 | import create from "zustand" 3 | import { createTrackedSelector } from "react-tracked" 4 | import { composeNewLayer, deleteLayer, updateOpacity, resizeAsset, getAsset, moveAsset, updateAnimation, updateFrameRate, updateDuration } from "./parser" 5 | import genUuid from "uuid-random" 6 | 7 | const useStore = create((set, get) => ({ 8 | lottieFile: null, 9 | selectedLayer: null, 10 | selectedAsset: null, 11 | visibleLayers: null, 12 | opacity: null, 13 | w: null, 14 | h: null, 15 | x: null, 16 | y: null, 17 | fr: null, 18 | setRemoteLottieFile: (lottieFile) => { 19 | let clonedLottieFile = JSON.parse(JSON.stringify(lottieFile)) 20 | clonedLottieFile.layers = clonedLottieFile.layers.map((layer) => ({ ...layer, uuid: genUuid() })) 21 | set({ lottieFile: clonedLottieFile, visibleLayers: clonedLottieFile.layers }) 22 | }, 23 | setImage: (image, externalBase64, method='simple', duration) => { 24 | const lottieFile = composeNewLayer(get().lottieFile ?? baseLottie, { image, externalBase64 }, method, duration) 25 | set({ lottieFile, visibleLayers: lottieFile.layers }) 26 | }, 27 | setSelectedLayer: (selectedLayer) => { 28 | const selectedAsset = getAsset(get().lottieFile, selectedLayer.refId) 29 | set({ 30 | selectedLayer, 31 | selectedAsset, 32 | ...(selectedAsset ? { w: selectedAsset.w, h: selectedAsset.h } : {}), 33 | x: selectedLayer.ks.a.k[0], 34 | y: selectedLayer.ks.a.k[1], 35 | opacity: selectedLayer.ks.o.k 36 | }) 37 | }, 38 | hideLayer: () => { 39 | set({ 40 | visibleLayers: get().visibleLayers.filter((layer) => layer.uuid !== get().selectedLayer?.uuid) 41 | }) 42 | }, 43 | showLayer: () => { 44 | set({ visibleLayers: get().visibleLayers.concat(get().selectedLayer) }) 45 | }, 46 | deleteLayer: () => { 47 | const lottieFile = deleteLayer(get().lottieFile, get().selectedLayer) 48 | set({ lottieFile, visibleLayers: lottieFile.layers, selectedLayer: null }) 49 | }, 50 | setOpacity: (opacity) => { 51 | const lottieFile = updateOpacity(get().lottieFile, get().selectedLayer, opacity) 52 | set({ opacity, lottieFile }) 53 | }, 54 | setAssetWidth: (w) => { 55 | const lottieFile = resizeAsset(get().lottieFile, get().selectedLayer, { w }) 56 | set({ w, lottieFile }) 57 | }, 58 | setAssetHeight: (h) => { 59 | const lottieFile = resizeAsset(get().lottieFile, get().selectedLayer, { h }) 60 | set({ h, lottieFile }) 61 | }, 62 | setAssetXPosition: (x) => { 63 | const lottieFile = moveAsset(get().lottieFile, get().selectedLayer, { x }) 64 | set({ x, lottieFile }) 65 | }, 66 | setAssetYPosition: (y) => { 67 | const lottieFile = moveAsset(get().lottieFile, get().selectedLayer, { y }) 68 | set({ y, lottieFile }) 69 | }, 70 | setAnimation: (index) => { 71 | const lottieFile = updateAnimation(get().lottieFile, get().selectedLayer, index) 72 | set({ lottieFile }) 73 | }, 74 | setFrameRate: (frameRate) => { 75 | const lottieFile = updateFrameRate(get().lottieFile, frameRate) 76 | set({ lottieFile }) 77 | }, 78 | setDuration: (duration) => { 79 | const lottieFile = updateDuration(get().lottieFile, duration) 80 | set({ lottieFile }) 81 | } 82 | })) 83 | 84 | const useTrackedStore = createTrackedSelector(useStore) 85 | 86 | export default useTrackedStore 87 | -------------------------------------------------------------------------------- /src/components/BoardingModal.jsx: -------------------------------------------------------------------------------- 1 | import { useState, useEffect } from "react" 2 | import { Modal as ChakaraModal, ModalOverlay, ModalContent, ModalHeader, ModalBody, ModalFooter, Button, Divider, Box, Text, Select } from "@chakra-ui/react" 3 | import { FilePond, registerPlugin } from "react-filepond" 4 | import FilePondPluginImageExifOrientation from "filepond-plugin-image-exif-orientation" 5 | import FilePondPluginImagePreview from "filepond-plugin-image-preview" 6 | import FilePondPluginFileEncode from "filepond-plugin-file-encode" 7 | import Input from "./Input" 8 | import NumberInput from "./NumberInput" 9 | import useStore from "../store" 10 | import { useErrorToast } from "../hooks/useToast" 11 | import useGetLottie from "../hooks/useGetLottie" 12 | import "filepond/dist/filepond.min.css" 13 | import "filepond-plugin-image-preview/dist/filepond-plugin-image-preview.css" 14 | 15 | registerPlugin(FilePondPluginImageExifOrientation, FilePondPluginImagePreview, FilePondPluginFileEncode) 16 | 17 | export default function Modal() { 18 | const [lottieUrl, setLottieUrl] = useState("") 19 | const getLottie = useGetLottie() 20 | const errorToast = useErrorToast() 21 | const [files, setFiles] = useState([]) 22 | const [importMethod, setImportMethod] = useState('simple') 23 | const [framesPerImage, setFramesPerImage] = useState(1) 24 | const { setImage, setRemoteLottieFile, setDuration } = useStore() 25 | 26 | const onSubmit = () => { 27 | try { 28 | if (files.length && lottieUrl) errorToast({ description: "You need to pick one option only" }) 29 | else if (files.length) { 30 | files.forEach(({ file, getFileEncodeDataURL, fileExtension }, number) => { 31 | if (["svg", "png", "jpeg", "jpg"].includes(fileExtension)) setImage(file, getFileEncodeDataURL(), importMethod, framesPerImage) 32 | else errorToast({ description: "File format is not supported" }) 33 | }) 34 | if (importMethod === 'sequence') setDuration(files.length * framesPerImage) 35 | } 36 | else if (lottieUrl) { 37 | if (lottieUrl.includes(".json")) getLottie.mutate(lottieUrl) 38 | else errorToast({ description: "URL must be type of Lottie file" }) 39 | } else if (getLottie.isError) errorToast() 40 | else errorToast({ description: "You must add an image or LottieFile to get started" }) 41 | } catch (e) { 42 | console.log(e) 43 | } 44 | } 45 | 46 | useEffect(() => { 47 | if (getLottie.data) setRemoteLottieFile(getLottie.data) 48 | }, [getLottie.data, setRemoteLottieFile]) 49 | 50 | return ( 51 | <> 52 | 53 | 54 | 55 | Get started! 56 | 57 | setLottieUrl(target.value)} 62 | /> 63 | 64 | 65 | Or upload your image here directly 66 | 67 | 68 | 69 | Multiple images import method: 70 | 74 | 75 | { importMethod !== "sequence" ? null : 76 | <> 77 | setFramesPerImage(parseInt(value))}/> 78 | 79 | } 80 | 81 | 82 | 83 | 84 | 85 | 86 | 87 | 90 | 91 | 92 | 93 | 94 | ) 95 | } 96 | -------------------------------------------------------------------------------- /src/parser.js: -------------------------------------------------------------------------------- 1 | import genUuid from "uuid-random" 2 | 3 | export const composeNewLayer = (lottieFile, { image, externalBase64 = "" }, method, duration) => { 4 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 5 | const refId = `refId-${image.name}-${Date.now()}` 6 | const uuid = genUuid() 7 | 8 | const tot = method === 'simple' ? 60.0000024438501 : duration 9 | 10 | const lastLayer = cloned.layers.length ? cloned.layers[cloned.layers.length - 1] : undefined 11 | const ip = lastLayer ? lastLayer.op : 0 12 | const op = lastLayer ? lastLayer.op + duration : duration 13 | 14 | return { 15 | ...cloned, 16 | assets: [ 17 | ...cloned.assets, 18 | { 19 | id: refId, 20 | w: 500, 21 | h: 500, 22 | u: "", 23 | p: image?.base64 ?? externalBase64, 24 | e: 1, 25 | uuid 26 | } 27 | ], 28 | layers: [ 29 | ...cloned.layers, 30 | { 31 | ddd: 0, 32 | ind: 1, 33 | ty: 2, 34 | nm: image.name, 35 | refId, 36 | uuid, 37 | sr: 1, 38 | ks: { 39 | o: { a: 0, k: 100, ix: 11 }, 40 | r: { a: 0, k: 0, ix: 10 }, 41 | p: { k: [0, 0] }, 42 | a: { a: 0, k: [0, 0], ix: 2 }, 43 | s: { 44 | a: 1, 45 | k: [ 46 | { 47 | i: { x: [0.667, 0.667, 0.667], y: [1, 1, 1] }, 48 | o: { x: [0.333, 0.333, 0.333], y: [0, 0, 0] }, 49 | t: 0, 50 | s: [100, 100, 0] 51 | }, 52 | { t: 59.0000024031193, s: [100, 100, 100] } 53 | ], 54 | ix: 6 55 | } 56 | }, 57 | ao: 0, 58 | ip: method === 'simple' ? 0 : ip, 59 | op: method === 'simple' ? tot : op, 60 | st: 0, 61 | bm: 0 62 | } 63 | ] 64 | } 65 | } 66 | 67 | const bounceAnimation = (totalTime, a) => ({ 68 | o: { a: 0, k: 100, ix: 11 }, 69 | r: { a: 0, k: 0, ix: 10 }, 70 | p: { k: [0, 0] }, 71 | a: a, 72 | s: { 73 | a: 1, 74 | k: [ 75 | { 76 | i: { x: [0.667, 0.667, 0.667], y: [1, 1, 1] }, 77 | o: { x: [0.333, 0.333, 0.333], y: [0, 0, 0] }, 78 | t: 0, 79 | s: [100, 100, 100] 80 | }, 81 | { 82 | i: { x: [0.667, 0.667, 0.667], y: [1, 1, 1] }, 83 | o: { x: [0.333, 0.333, 0.333], y: [0, 0, 0] }, 84 | t: (totalTime / 3) * 2, 85 | s: [100, 50, 100] 86 | }, 87 | { 88 | i: { x: [0.667, 0.667, 0.667], y: [1, 1, 1] }, 89 | o: { x: [0.333, 0.333, 0.333], y: [0, 0, 0] }, 90 | t: totalTime, 91 | s: [100, 100, 100] 92 | }, 93 | { t: totalTime, s: [100, 100, 100] } 94 | ], 95 | ix: 6 96 | } 97 | }) 98 | 99 | const appearAnimation = (totalTime, a) => ({ 100 | ty: "tr", 101 | o: { k: 100 }, 102 | r: { k: 0 }, 103 | p: { k: [0, 0] }, 104 | a: a, 105 | s: { 106 | a: 1, 107 | k: [ 108 | { 109 | i: { x: [0.67, 0.67, 0.67], y: [1, 1, 1] }, 110 | o: { x: [0.33, 0.33, 0.33], y: [0, 0, 0] }, 111 | t: 0, 112 | s: [0, 0, 100] 113 | }, 114 | { 115 | i: { x: [0.67, 0.67, 0.67], y: [1, 1, 1] }, 116 | o: { x: [0.33, 0.33, 0.33], y: [0, 0, 0] }, 117 | t: (totalTime / 3) * 2, 118 | s: [100, 100, 100] 119 | }, 120 | { 121 | i: { x: [0.83, 0.83, 0.83], y: [1, 1, 1] }, 122 | o: { x: [0.33, 0.33, 0.33], y: [0, 0, 0] }, 123 | t: totalTime, 124 | s: [100, 100, 100] 125 | }, 126 | { t: totalTime * 2, s: [99, 99, 100] } 127 | ], 128 | ix: 6 129 | }, 130 | sk: { k: 0 }, 131 | sa: { k: 0 } 132 | }) 133 | 134 | const rotateAnimation = (totalTime, a) => ({ 135 | ty: "tr", 136 | o: { k: 100 }, 137 | r: { k: 0 }, 138 | p: { 139 | k: [ 140 | { 141 | i: { x: 0.67, y: 1 }, 142 | o: { x: 0.33, y: 0 }, 143 | t: 0, 144 | s: [0, -512, 0], 145 | to: [0, -76.58, 0], 146 | ti: [0, 2.38, 0] 147 | }, 148 | { 149 | i: { x: 0.67, y: 1 }, 150 | o: { x: 0.33, y: 0 }, 151 | t: totalTime, 152 | s: [0, 384, 0], 153 | to: [0, -6.63, 0], 154 | ti: [0, -0.48, 0] 155 | }, 156 | { t: totalTime, s: [0, 0, 0] } 157 | ], 158 | ix: 2, 159 | a: 1 160 | }, 161 | a: a, 162 | s: { k: [100, 100] }, 163 | sk: { k: 0 }, 164 | sa: { k: 0 } 165 | }) 166 | 167 | export const animations = [ 168 | { name: "Bounce", value: bounceAnimation }, 169 | { name: "Appear", value: appearAnimation }, 170 | { name: "Drop down", value: rotateAnimation }, 171 | { name: "Default", value: (totalTime, a) => ({ a }) } 172 | ] 173 | 174 | export const deleteLayer = (lottieFile, selected_layer) => { 175 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 176 | if (cloned?.assets) cloned.assets = cloned.assets.filter((asset) => asset.id !== selected_layer.uuid) 177 | cloned.layers = cloned.layers.filter((layer) => layer.uuid !== selected_layer.uuid) 178 | return cloned 179 | } 180 | 181 | export const updateOpacity = (lottieFile, selected_layer, opacity) => { 182 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 183 | cloned.layers = [...cloned.layers].map((layer) => { 184 | if (layer.uuid === selected_layer?.uuid) layer.ks.o.k = opacity 185 | return layer 186 | }) 187 | return cloned 188 | } 189 | 190 | export const resizeAsset = (lottieFile, selected_layer, { h, w }) => { 191 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 192 | if (cloned?.assets) { 193 | cloned.assets = [...cloned.assets].map((asset) => { 194 | if (asset.id === selected_layer.refId) { 195 | asset.h = h ? h : asset.h 196 | asset.w = w ? w : asset.w 197 | } 198 | return asset 199 | }) 200 | } 201 | return cloned 202 | } 203 | 204 | export const moveAsset = (lottieFile, selected_layer, { x, y }) => { 205 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 206 | cloned.layers = [...cloned.layers].map((layer) => { 207 | if (layer.uuid === selected_layer.uuid && x !== "-" && y !== "-") { 208 | layer.ks.a = { 209 | ...layer.ks.a, 210 | k: [x ? parseFloat(x) : parseFloat(layer.ks.a.k[0]), y ? parseFloat(y) : parseFloat(layer.ks.a.k[1]), 0] 211 | } 212 | } 213 | return layer 214 | }) 215 | 216 | return cloned 217 | } 218 | 219 | export const getAsset = (lottieFile, id) => { 220 | return lottieFile.assets.find((asset) => asset.id === id) 221 | } 222 | 223 | export const updateFrameRate = (lottieFile, newFrameRate) => { 224 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 225 | cloned.fr = parseFloat(newFrameRate) 226 | return cloned 227 | } 228 | 229 | export const updateDuration = (lottieFile, duration) => { 230 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 231 | cloned.op = parseFloat(duration) 232 | return cloned 233 | } 234 | 235 | export const updateAnimation = (lottieFile, selected_layer, index) => { 236 | let cloned = JSON.parse(JSON.stringify(lottieFile)) 237 | 238 | cloned.layers = [...cloned.layers].map((layer) => { 239 | if (layer.uuid === selected_layer?.uuid) { 240 | layer.ks = animations[index].value(lottieFile.op, layer.ks.a) 241 | return layer 242 | } 243 | return layer 244 | }) 245 | 246 | return cloned 247 | } 248 | -------------------------------------------------------------------------------- /src/assets/test.json: -------------------------------------------------------------------------------- 1 | { 2 | "v": "5.5.7", 3 | "meta": { "g": "LottieFiles AE 0.1.21", "a": "", "k": "", "d": "", "tc": "" }, 4 | "fr": 29.9700012207031, 5 | "ip": 0, 6 | "op": 60.0000024438501, 7 | "w": 1049, 8 | "h": 584, 9 | "nm": "nuxt_js", 10 | "ddd": 0, 11 | "assets": [ 12 | { 13 | "id": "image_0", 14 | "w": 401, 15 | "h": 301, 16 | "u": "", 17 | "p": "", 18 | "e": 1 19 | }, 20 | { 21 | "id": "image_1", 22 | "w": 424, 23 | "h": 399, 24 | "u": "", 25 | "p": "", 26 | "e": 1 27 | }, 28 | { 29 | "id": "image_2", 30 | "w": 404, 31 | "h": 377, 32 | "u": "", 33 | "p": "", 34 | "e": 1 35 | }, 36 | { 37 | "id": "image_3", 38 | "w": 208, 39 | "h": 392, 40 | "u": "", 41 | "p": "", 42 | "e": 1 43 | }, 44 | { 45 | "id": "image_4", 46 | "w": 176, 47 | "h": 376, 48 | "u": "", 49 | "p": "", 50 | "e": 1 51 | } 52 | ], 53 | "layers": [ 54 | { 55 | "ddd": 0, 56 | "ind": 1, 57 | "ty": 2, 58 | "nm": "nuxt", 59 | "refId": "image_0", 60 | "sr": 1, 61 | "ks": { 62 | "o": { "a": 0, "k": 100, "ix": 11 }, 63 | "r": { "a": 0, "k": 0, "ix": 10 }, 64 | "p": { "a": 0, "k": [741.002, 312.465, 0], "ix": 2 }, 65 | "a": { "a": 0, "k": [402.391, 302.085, 0], "ix": 1 }, 66 | "s": { 67 | "a": 1, 68 | "k": [ 69 | { 70 | "i": { "x": [0.667, 0.667, 0.667], "y": [1, 1, 1] }, 71 | "o": { "x": [0.333, 0.333, 0.333], "y": [0, 0, 0] }, 72 | "t": 0, 73 | "s": [100, 100, 100] 74 | }, 75 | { 76 | "i": { "x": [0.667, 0.667, 0.667], "y": [1, 1, 1] }, 77 | "o": { "x": [0.333, 0.333, 0.333], "y": [0, 0, 0] }, 78 | "t": 30, 79 | "s": [75, 75, 100] 80 | }, 81 | { "t": 59.0000024031193, "s": [100, 100, 100] } 82 | ], 83 | "ix": 6, 84 | "x": "var $bm_rt;\n$bm_rt = loopOutDuration('cycle', 0);" 85 | } 86 | }, 87 | "ao": 0, 88 | "ip": 0, 89 | "op": 60.0000024438501, 90 | "st": 0, 91 | "bm": 0 92 | }, 93 | { 94 | "ddd": 0, 95 | "ind": 2, 96 | "ty": 2, 97 | "nm": "woman", 98 | "refId": "image_1", 99 | "sr": 1, 100 | "ks": { 101 | "o": { "a": 0, "k": 100, "ix": 11 }, 102 | "r": { "a": 0, "k": 0, "ix": 10 }, 103 | "p": { "a": 0, "k": [265.254, 384.892, 0], "ix": 2 }, 104 | "a": { "a": 0, "k": [211.75, 199.097, 0], "ix": 1 }, 105 | "s": { "a": 0, "k": [100, 100, 100], "ix": 6 } 106 | }, 107 | "ao": 0, 108 | "ip": 0, 109 | "op": 60.0000024438501, 110 | "st": 0, 111 | "bm": 0 112 | }, 113 | { 114 | "ddd": 0, 115 | "ind": 3, 116 | "ty": 2, 117 | "nm": "others", 118 | "refId": "image_2", 119 | "sr": 1, 120 | "ks": { 121 | "o": { "a": 0, "k": 100, "ix": 11 }, 122 | "r": { "a": 0, "k": 0, "ix": 10 }, 123 | "p": { "a": 0, "k": [200.378, 395.96, 0], "ix": 2 }, 124 | "a": { "a": 0, "k": [201.693, 188.029, 0], "ix": 1 }, 125 | "s": { "a": 0, "k": [100, 100, 100], "ix": 6 } 126 | }, 127 | "ao": 0, 128 | "ip": 0, 129 | "op": 60.0000024438501, 130 | "st": 0, 131 | "bm": 0 132 | }, 133 | { 134 | "ddd": 0, 135 | "ind": 4, 136 | "ty": 2, 137 | "nm": "man", 138 | "refId": "image_3", 139 | "sr": 1, 140 | "ks": { 141 | "o": { "a": 0, "k": 100, "ix": 11 }, 142 | "r": { "a": 0, "k": 0, "ix": 10 }, 143 | "p": { "a": 0, "k": [828.375, 386.755, 0], "ix": 2 }, 144 | "a": { "a": 0, "k": [103.787, 195.844, 0], "ix": 1 }, 145 | "s": { "a": 0, "k": [100, 100, 100], "ix": 6 } 146 | }, 147 | "ao": 0, 148 | "ip": 0, 149 | "op": 60.0000024438501, 150 | "st": 0, 151 | "bm": 0 152 | }, 153 | { 154 | "ddd": 0, 155 | "ind": 5, 156 | "ty": 2, 157 | "nm": "tree", 158 | "refId": "image_4", 159 | "sr": 1, 160 | "ks": { 161 | "o": { "a": 0, "k": 100, "ix": 11 }, 162 | "r": { 163 | "a": 1, 164 | "k": [ 165 | { 166 | "i": { "x": [0.667], "y": [1] }, 167 | "o": { "x": [0.333], "y": [0] }, 168 | "t": 0, 169 | "s": [0] 170 | }, 171 | { 172 | "i": { "x": [0.667], "y": [1] }, 173 | "o": { "x": [0.333], "y": [0] }, 174 | "t": 15, 175 | "s": [3] 176 | }, 177 | { 178 | "i": { "x": [0.667], "y": [1] }, 179 | "o": { "x": [0.333], "y": [0] }, 180 | "t": 30, 181 | "s": [0] 182 | }, 183 | { 184 | "i": { "x": [0.667], "y": [1] }, 185 | "o": { "x": [0.333], "y": [0] }, 186 | "t": 45, 187 | "s": [-3] 188 | }, 189 | { "t": 59.0000024031193, "s": [0] } 190 | ], 191 | "ix": 10, 192 | "x": "var $bm_rt;\n$bm_rt = loopOutDuration('cycle', 0);" 193 | }, 194 | "p": { "a": 0, "k": [991.827, 579.603, 0], "ix": 2 }, 195 | "a": { "a": 0, "k": [89.189, 370.292, 0], "ix": 1 }, 196 | "s": { "a": 0, "k": [100, 100, 100], "ix": 6 } 197 | }, 198 | "ao": 0, 199 | "ip": 0, 200 | "op": 60.0000024438501, 201 | "st": 0, 202 | "bm": 0 203 | } 204 | ], 205 | "markers": [] 206 | } 207 | --------------------------------------------------------------------------------