├── 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 |
5 |
6 |
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 |
4 |
5 |
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 |
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 |
--------------------------------------------------------------------------------