diff --git a/package-lock.json b/package-lock.json
index 93f28458..72143230 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -531,6 +531,21 @@
"@floating-ui/utils": "^0.2.8"
}
},
+ "node_modules/@floating-ui/react": {
+ "version": "0.26.24",
+ "resolved": "https://registry.npmjs.org/@floating-ui/react/-/react-0.26.24.tgz",
+ "integrity": "sha512-2ly0pCkZIGEQUq5H8bBK0XJmc1xIK/RM3tvVzY3GBER7IOD1UgmC2Y2tjj4AuS+TC+vTE1KJv2053290jua0Sw==",
+ "license": "MIT",
+ "dependencies": {
+ "@floating-ui/react-dom": "^2.1.2",
+ "@floating-ui/utils": "^0.2.8",
+ "tabbable": "^6.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8.0",
+ "react-dom": ">=16.8.0"
+ }
+ },
"node_modules/@floating-ui/react-dom": {
"version": "2.1.2",
"resolved": "https://registry.npmjs.org/@floating-ui/react-dom/-/react-dom-2.1.2.tgz",
@@ -548,6 +563,34 @@
"resolved": "https://registry.npmjs.org/@floating-ui/utils/-/utils-0.2.8.tgz",
"integrity": "sha512-kym7SodPp8/wloecOpcmSnWJsK7M0E5Wg8UcFA+uO4B9s5d0ywXOEro/8HM9x0rW+TljRzul/14UYz3TleT3ig=="
},
+ "node_modules/@headlessui/react": {
+ "version": "2.1.8",
+ "resolved": "https://registry.npmjs.org/@headlessui/react/-/react-2.1.8.tgz",
+ "integrity": "sha512-uajqVkAcVG/wHwG9Fh5PFMcFpf2VxM4vNRNKxRjuK009kePVur8LkuuygHfIE+2uZ7z7GnlTtYsyUe6glPpTLg==",
+ "license": "MIT",
+ "dependencies": {
+ "@floating-ui/react": "^0.26.16",
+ "@react-aria/focus": "^3.17.1",
+ "@react-aria/interactions": "^3.21.3",
+ "@tanstack/react-virtual": "^3.8.1"
+ },
+ "engines": {
+ "node": ">=10"
+ },
+ "peerDependencies": {
+ "react": "^18",
+ "react-dom": "^18"
+ }
+ },
+ "node_modules/@huggingface/jinja": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/@huggingface/jinja/-/jinja-0.2.2.tgz",
+ "integrity": "sha512-/KPde26khDUIPkTGU82jdtTW9UAuvUTumCAbFs/7giR0SxsvZC4hru51PBvpijH6BVkHcROcvZM/lpy5h1jRRA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ }
+ },
"node_modules/@humanwhocodes/config-array": {
"version": "0.13.0",
"resolved": "https://registry.npmjs.org/@humanwhocodes/config-array/-/config-array-0.13.0.tgz",
@@ -941,6 +984,70 @@
"node": ">=14"
}
},
+ "node_modules/@protobufjs/aspromise": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/aspromise/-/aspromise-1.1.2.tgz",
+ "integrity": "sha512-j+gKExEuLmKwvz3OgROXtrJ2UG2x8Ch2YZUxahh+s1F2HZ+wAceUNLkvy6zKCPVRkU++ZWQrdxsUeQXmcg4uoQ==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/base64": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/base64/-/base64-1.1.2.tgz",
+ "integrity": "sha512-AZkcAA5vnN/v4PDqKyMR5lx7hZttPDgClv83E//FMNhR2TMcLUhfRUBHCmSl0oi9zMgDDqRUJkSxO3wm85+XLg==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/codegen": {
+ "version": "2.0.4",
+ "resolved": "https://registry.npmjs.org/@protobufjs/codegen/-/codegen-2.0.4.tgz",
+ "integrity": "sha512-YyFaikqM5sH0ziFZCN3xDC7zeGaB/d0IUb9CATugHWbd1FRFwWwt4ld4OYMPWu5a3Xe01mGAULCdqhMlPl29Jg==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/eventemitter": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/eventemitter/-/eventemitter-1.1.0.tgz",
+ "integrity": "sha512-j9ednRT81vYJ9OfVuXG6ERSTdEL1xVsNgqpkxMsbIabzSo3goCjDIveeGv5d03om39ML71RdmrGNjG5SReBP/Q==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/fetch": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/fetch/-/fetch-1.1.0.tgz",
+ "integrity": "sha512-lljVXpqXebpsijW71PZaCYeIcE5on1w5DlQy5WH6GLbFryLUrBD4932W/E2BSpfRJWseIL4v/KPgBFxDOIdKpQ==",
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "@protobufjs/aspromise": "^1.1.1",
+ "@protobufjs/inquire": "^1.1.0"
+ }
+ },
+ "node_modules/@protobufjs/float": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/float/-/float-1.0.2.tgz",
+ "integrity": "sha512-Ddb+kVXlXst9d+R9PfTIxh1EdNkgoRe5tOX6t01f1lYWOvJnSPDBlG241QLzcyPdoNTsblLUdujGSE4RzrTZGQ==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/inquire": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/inquire/-/inquire-1.1.0.tgz",
+ "integrity": "sha512-kdSefcPdruJiFMVSbn801t4vFK7KB/5gd2fYvrxhuJYg8ILrmn9SKSX2tZdV6V+ksulWqS7aXjBcRXl3wHoD9Q==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/path": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmjs.org/@protobufjs/path/-/path-1.1.2.tgz",
+ "integrity": "sha512-6JOcJ5Tm08dOHAbdR3GrvP+yUUfkjG5ePsHYczMFLq3ZmMkAD98cDgcT2iA1lJ9NVwFd4tH/iSSoe44YWkltEA==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/pool": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/pool/-/pool-1.1.0.tgz",
+ "integrity": "sha512-0kELaGSIDBKvcgS4zkjz1PeddatrjYcmMWOlAuAPwAeccUrPHdUqo/J6LiymHHEiJT5NrF1UVwxY14f+fy4WQw==",
+ "license": "BSD-3-Clause"
+ },
+ "node_modules/@protobufjs/utf8": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmjs.org/@protobufjs/utf8/-/utf8-1.1.0.tgz",
+ "integrity": "sha512-Vvn3zZrhQZkkBE8LSuW3em98c0FwgO4nxzv6OdSxPKJIEKY2bGbHn+mhGIPerzI4twdxaP8/0+06HBpwf345Lw==",
+ "license": "BSD-3-Clause"
+ },
"node_modules/@radix-ui/colors": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/@radix-ui/colors/-/colors-3.0.0.tgz",
@@ -2174,6 +2281,89 @@
"resolved": "https://registry.npmjs.org/classnames/-/classnames-2.3.2.tgz",
"integrity": "sha512-CSbhY4cFEJRe6/GQzIk5qXZ4Jeg5pcsP7b5peFSDpffpe1cqjASH/n9UTjBwOp6XpMSTwQ8Za2K5V02ueA7Tmw=="
},
+ "node_modules/@react-aria/focus": {
+ "version": "3.18.2",
+ "resolved": "https://registry.npmjs.org/@react-aria/focus/-/focus-3.18.2.tgz",
+ "integrity": "sha512-Jc/IY+StjA3uqN73o6txKQ527RFU7gnG5crEl5Xy3V+gbYp2O5L3ezAo/E0Ipi2cyMbG6T5Iit1IDs7hcGu8aw==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@react-aria/interactions": "^3.22.2",
+ "@react-aria/utils": "^3.25.2",
+ "@react-types/shared": "^3.24.1",
+ "@swc/helpers": "^0.5.0",
+ "clsx": "^2.0.0"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/@react-aria/interactions": {
+ "version": "3.22.2",
+ "resolved": "https://registry.npmjs.org/@react-aria/interactions/-/interactions-3.22.2.tgz",
+ "integrity": "sha512-xE/77fRVSlqHp2sfkrMeNLrqf2amF/RyuAS6T5oDJemRSgYM3UoxTbWjucPhfnoW7r32pFPHHgz4lbdX8xqD/g==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@react-aria/ssr": "^3.9.5",
+ "@react-aria/utils": "^3.25.2",
+ "@react-types/shared": "^3.24.1",
+ "@swc/helpers": "^0.5.0"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/@react-aria/ssr": {
+ "version": "3.9.5",
+ "resolved": "https://registry.npmjs.org/@react-aria/ssr/-/ssr-3.9.5.tgz",
+ "integrity": "sha512-xEwGKoysu+oXulibNUSkXf8itW0npHHTa6c4AyYeZIJyRoegeteYuFpZUBPtIDE8RfHdNsSmE1ssOkxRnwbkuQ==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@swc/helpers": "^0.5.0"
+ },
+ "engines": {
+ "node": ">= 12"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/@react-aria/utils": {
+ "version": "3.25.2",
+ "resolved": "https://registry.npmjs.org/@react-aria/utils/-/utils-3.25.2.tgz",
+ "integrity": "sha512-GdIvG8GBJJZygB4L2QJP1Gabyn2mjFsha73I2wSe+o4DYeGWoJiMZRM06PyTIxLH4S7Sn7eVDtsSBfkc2VY/NA==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@react-aria/ssr": "^3.9.5",
+ "@react-stately/utils": "^3.10.3",
+ "@react-types/shared": "^3.24.1",
+ "@swc/helpers": "^0.5.0",
+ "clsx": "^2.0.0"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/@react-stately/utils": {
+ "version": "3.10.3",
+ "resolved": "https://registry.npmjs.org/@react-stately/utils/-/utils-3.10.3.tgz",
+ "integrity": "sha512-moClv7MlVSHpbYtQIkm0Cx+on8Pgt1XqtPx6fy9rQFb2DNc9u1G3AUVnqA17buOkH1vLxAtX4MedlxMWyRCYYA==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@swc/helpers": "^0.5.0"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/@react-types/shared": {
+ "version": "3.24.1",
+ "resolved": "https://registry.npmjs.org/@react-types/shared/-/shared-3.24.1.tgz",
+ "integrity": "sha512-AUQeGYEm/zDTN6zLzdXolDxz3Jk5dDL7f506F07U8tBwxNNI3WRdhU84G0/AaFikOZzDXhOZDr3MhQMzyE7Ydw==",
+ "license": "Apache-2.0",
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0-rc.1 || ^18.0.0 || ^19.0.0"
+ }
+ },
"node_modules/@rollup/rollup-android-arm-eabi": {
"version": "4.21.3",
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.21.3.tgz",
@@ -2654,6 +2844,15 @@
"integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==",
"dev": true
},
+ "node_modules/@swc/helpers": {
+ "version": "0.5.13",
+ "resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.13.tgz",
+ "integrity": "sha512-UoKGxQ3r5kYI9dALKJapMmuK+1zWM/H17Z1+iwnNmzcJRnfFuevZs375TA5rW31pu4BS4NoSy1fRsexDXfWn5w==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "tslib": "^2.4.0"
+ }
+ },
"node_modules/@swc/types": {
"version": "0.1.12",
"resolved": "https://registry.npmjs.org/@swc/types/-/types-0.1.12.tgz",
@@ -2699,6 +2898,33 @@
"react": "^18 || ^19"
}
},
+ "node_modules/@tanstack/react-virtual": {
+ "version": "3.10.8",
+ "resolved": "https://registry.npmjs.org/@tanstack/react-virtual/-/react-virtual-3.10.8.tgz",
+ "integrity": "sha512-VbzbVGSsZlQktyLrP5nxE+vE1ZR+U0NFAWPbJLoG2+DKPwd2D7dVICTVIIaYlJqX1ZCEnYDbaOpmMwbsyhBoIA==",
+ "license": "MIT",
+ "dependencies": {
+ "@tanstack/virtual-core": "3.10.8"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ },
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0",
+ "react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0"
+ }
+ },
+ "node_modules/@tanstack/virtual-core": {
+ "version": "3.10.8",
+ "resolved": "https://registry.npmjs.org/@tanstack/virtual-core/-/virtual-core-3.10.8.tgz",
+ "integrity": "sha512-PBu00mtt95jbKFi6Llk9aik8bnR3tR/oQP1o3TSi+iG//+Q2RTIzCEgKkHG8BB86kxMNW6O8wku+Lmi+QFR6jA==",
+ "license": "MIT",
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ }
+ },
"node_modules/@tokenizer/token": {
"version": "0.3.0",
"resolved": "https://registry.npmjs.org/@tokenizer/token/-/token-0.3.0.tgz",
@@ -2811,6 +3037,12 @@
"@types/node": "*"
}
},
+ "node_modules/@types/long": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/@types/long/-/long-4.0.2.tgz",
+ "integrity": "sha512-MqTGEo5bj5t157U6fA/BiDynNkn0YknVdh48CMPkTSpFTVmvao5UQmm7uEF6xBEo7qIMAlY/JSleYaE6VOdpaA==",
+ "license": "MIT"
+ },
"node_modules/@types/node": {
"version": "22.5.5",
"resolved": "https://registry.npmjs.org/@types/node/-/node-22.5.5.tgz",
@@ -3489,6 +3721,20 @@
"vite": "^4 || ^5"
}
},
+ "node_modules/@xenova/transformers": {
+ "version": "2.17.2",
+ "resolved": "https://registry.npmjs.org/@xenova/transformers/-/transformers-2.17.2.tgz",
+ "integrity": "sha512-lZmHqzrVIkSvZdKZEx7IYY51TK0WDrC8eR0c5IMnBsO8di8are1zzw8BlLhyO2TklZKLN5UffNGs1IJwT6oOqQ==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "@huggingface/jinja": "^0.2.2",
+ "onnxruntime-web": "1.14.0",
+ "sharp": "^0.32.0"
+ },
+ "optionalDependencies": {
+ "onnxruntime-node": "1.14.0"
+ }
+ },
"node_modules/acorn": {
"version": "8.12.1",
"resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.1.tgz",
@@ -3738,12 +3984,65 @@
"node": ">= 0.4"
}
},
+ "node_modules/b4a": {
+ "version": "1.6.7",
+ "resolved": "https://registry.npmjs.org/b4a/-/b4a-1.6.7.tgz",
+ "integrity": "sha512-OnAYlL5b7LEkALw87fUVafQw5rVR9RjwGd4KUwNQ6DrrNmaVaUCgLipfVlzrPQ4tWOR9P0IXGNOx50jYCCdSJg==",
+ "license": "Apache-2.0"
+ },
"node_modules/balanced-match": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-1.0.2.tgz",
"integrity": "sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw==",
"dev": true
},
+ "node_modules/bare-events": {
+ "version": "2.5.0",
+ "resolved": "https://registry.npmjs.org/bare-events/-/bare-events-2.5.0.tgz",
+ "integrity": "sha512-/E8dDe9dsbLyh2qrZ64PEPadOQ0F4gbl1sUJOrmph7xOiIxfY8vwab/4bFLh4Y88/Hk/ujKcrQKc+ps0mv873A==",
+ "license": "Apache-2.0",
+ "optional": true
+ },
+ "node_modules/bare-fs": {
+ "version": "2.3.5",
+ "resolved": "https://registry.npmjs.org/bare-fs/-/bare-fs-2.3.5.tgz",
+ "integrity": "sha512-SlE9eTxifPDJrT6YgemQ1WGFleevzwY+XAP1Xqgl56HtcrisC2CHCZ2tq6dBpcH2TnNxwUEUGhweo+lrQtYuiw==",
+ "license": "Apache-2.0",
+ "optional": true,
+ "dependencies": {
+ "bare-events": "^2.0.0",
+ "bare-path": "^2.0.0",
+ "bare-stream": "^2.0.0"
+ }
+ },
+ "node_modules/bare-os": {
+ "version": "2.4.4",
+ "resolved": "https://registry.npmjs.org/bare-os/-/bare-os-2.4.4.tgz",
+ "integrity": "sha512-z3UiI2yi1mK0sXeRdc4O1Kk8aOa/e+FNWZcTiPB/dfTWyLypuE99LibgRaQki914Jq//yAWylcAt+mknKdixRQ==",
+ "license": "Apache-2.0",
+ "optional": true
+ },
+ "node_modules/bare-path": {
+ "version": "2.1.3",
+ "resolved": "https://registry.npmjs.org/bare-path/-/bare-path-2.1.3.tgz",
+ "integrity": "sha512-lh/eITfU8hrj9Ru5quUp0Io1kJWIk1bTjzo7JH1P5dWmQ2EL4hFUlfI8FonAhSlgIfhn63p84CDY/x+PisgcXA==",
+ "license": "Apache-2.0",
+ "optional": true,
+ "dependencies": {
+ "bare-os": "^2.1.0"
+ }
+ },
+ "node_modules/bare-stream": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmjs.org/bare-stream/-/bare-stream-2.3.0.tgz",
+ "integrity": "sha512-pVRWciewGUeCyKEuRxwv06M079r+fRjAQjBEK2P6OYGrO43O+Z0LrPZZEjlc4mB6C2RpZ9AxJ1s7NLEtOHO6eA==",
+ "license": "Apache-2.0",
+ "optional": true,
+ "dependencies": {
+ "b4a": "^1.6.6",
+ "streamx": "^2.20.0"
+ }
+ },
"node_modules/base64-arraybuffer": {
"version": "1.0.2",
"resolved": "https://registry.npmjs.org/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz",
@@ -3752,6 +4051,26 @@
"node": ">= 0.6.0"
}
},
+ "node_modules/base64-js": {
+ "version": "1.5.1",
+ "resolved": "https://registry.npmjs.org/base64-js/-/base64-js-1.5.1.tgz",
+ "integrity": "sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT"
+ },
"node_modules/before-after-hook": {
"version": "2.2.3",
"resolved": "https://registry.npmjs.org/before-after-hook/-/before-after-hook-2.2.3.tgz",
@@ -3874,6 +4193,17 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/bl": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/bl/-/bl-4.1.0.tgz",
+ "integrity": "sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==",
+ "license": "MIT",
+ "dependencies": {
+ "buffer": "^5.5.0",
+ "inherits": "^2.0.4",
+ "readable-stream": "^3.4.0"
+ }
+ },
"node_modules/blurhash": {
"version": "2.0.5",
"resolved": "https://registry.npmjs.org/blurhash/-/blurhash-2.0.5.tgz",
@@ -3937,6 +4267,30 @@
"node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
}
},
+ "node_modules/buffer": {
+ "version": "5.7.1",
+ "resolved": "https://registry.npmjs.org/buffer/-/buffer-5.7.1.tgz",
+ "integrity": "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "base64-js": "^1.3.1",
+ "ieee754": "^1.1.13"
+ }
+ },
"node_modules/buffer-crc32": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/buffer-crc32/-/buffer-crc32-1.0.0.tgz",
@@ -4133,6 +4487,12 @@
"node": ">= 6"
}
},
+ "node_modules/chownr": {
+ "version": "1.1.4",
+ "resolved": "https://registry.npmjs.org/chownr/-/chownr-1.1.4.tgz",
+ "integrity": "sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg==",
+ "license": "ISC"
+ },
"node_modules/classnames": {
"version": "2.5.1",
"resolved": "https://registry.npmjs.org/classnames/-/classnames-2.5.1.tgz",
@@ -4213,6 +4573,19 @@
"resolved": "plugins/cms-export",
"link": true
},
+ "node_modules/color": {
+ "version": "4.2.3",
+ "resolved": "https://registry.npmjs.org/color/-/color-4.2.3.tgz",
+ "integrity": "sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A==",
+ "license": "MIT",
+ "dependencies": {
+ "color-convert": "^2.0.1",
+ "color-string": "^1.9.0"
+ },
+ "engines": {
+ "node": ">=12.5.0"
+ }
+ },
"node_modules/color-convert": {
"version": "2.0.1",
"resolved": "https://registry.npmjs.org/color-convert/-/color-convert-2.0.1.tgz",
@@ -4233,6 +4606,16 @@
"resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.4.tgz",
"integrity": "sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA=="
},
+ "node_modules/color-string": {
+ "version": "1.9.1",
+ "resolved": "https://registry.npmjs.org/color-string/-/color-string-1.9.1.tgz",
+ "integrity": "sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg==",
+ "license": "MIT",
+ "dependencies": {
+ "color-name": "^1.0.0",
+ "simple-swizzle": "^0.2.2"
+ }
+ },
"node_modules/colorjs.io": {
"version": "0.5.2",
"resolved": "https://registry.npmjs.org/colorjs.io/-/colorjs.io-0.5.2.tgz",
@@ -4576,7 +4959,6 @@
"version": "6.0.0",
"resolved": "https://registry.npmjs.org/decompress-response/-/decompress-response-6.0.0.tgz",
"integrity": "sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ==",
- "dev": true,
"dependencies": {
"mimic-response": "^3.1.0"
},
@@ -4591,7 +4973,6 @@
"version": "3.1.0",
"resolved": "https://registry.npmjs.org/mimic-response/-/mimic-response-3.1.0.tgz",
"integrity": "sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ==",
- "dev": true,
"engines": {
"node": ">=10"
},
@@ -4599,6 +4980,15 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/deep-extend": {
+ "version": "0.6.0",
+ "resolved": "https://registry.npmjs.org/deep-extend/-/deep-extend-0.6.0.tgz",
+ "integrity": "sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=4.0.0"
+ }
+ },
"node_modules/deep-is": {
"version": "0.1.4",
"resolved": "https://registry.npmjs.org/deep-is/-/deep-is-0.1.4.tgz",
@@ -4768,7 +5158,6 @@
"version": "1.4.4",
"resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.4.tgz",
"integrity": "sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q==",
- "dev": true,
"dependencies": {
"once": "^1.4.0"
}
@@ -5129,6 +5518,15 @@
"node": ">=4"
}
},
+ "node_modules/expand-template": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/expand-template/-/expand-template-2.0.3.tgz",
+ "integrity": "sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg==",
+ "license": "(MIT OR WTFPL)",
+ "engines": {
+ "node": ">=6"
+ }
+ },
"node_modules/ext-list": {
"version": "2.2.2",
"resolved": "https://registry.npmjs.org/ext-list/-/ext-list-2.2.2.tgz",
@@ -5173,6 +5571,12 @@
"node": ">=6.0.0"
}
},
+ "node_modules/fast-fifo": {
+ "version": "1.3.2",
+ "resolved": "https://registry.npmjs.org/fast-fifo/-/fast-fifo-1.3.2.tgz",
+ "integrity": "sha512-/d9sfos4yxzpwkDkuN7k2SqFKtYNmCTzgfEpz82x34IM9/zc8KGxQoXg1liNC/izpRM/MBdt44Nmx41ZWqk+FQ==",
+ "license": "MIT"
+ },
"node_modules/fast-glob": {
"version": "3.3.2",
"resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.2.tgz",
@@ -5348,6 +5752,12 @@
"node": "^10.12.0 || >=12.0.0"
}
},
+ "node_modules/flatbuffers": {
+ "version": "1.12.0",
+ "resolved": "https://registry.npmjs.org/flatbuffers/-/flatbuffers-1.12.0.tgz",
+ "integrity": "sha512-c7CZADjRcl6j0PlvFy0ZqXQ67qSEZfrVPynmnL+2zPc+NtMvrF8Y0QceMo7QqnSPc7+uWjUIAbvCQ5WIKlMVdQ==",
+ "license": "SEE LICENSE IN LICENSE.txt"
+ },
"node_modules/flatted": {
"version": "3.3.1",
"resolved": "https://registry.npmjs.org/flatted/-/flatted-3.3.1.tgz",
@@ -5464,6 +5874,12 @@
"react-dom": "^18.2.0"
}
},
+ "node_modules/fs-constants": {
+ "version": "1.0.0",
+ "resolved": "https://registry.npmjs.org/fs-constants/-/fs-constants-1.0.0.tgz",
+ "integrity": "sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow==",
+ "license": "MIT"
+ },
"node_modules/fs.realpath": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz",
@@ -5533,6 +5949,12 @@
"node": ">=4"
}
},
+ "node_modules/github-from-package": {
+ "version": "0.0.0",
+ "resolved": "https://registry.npmjs.org/github-from-package/-/github-from-package-0.0.0.tgz",
+ "integrity": "sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw==",
+ "license": "MIT"
+ },
"node_modules/glob": {
"version": "7.2.3",
"resolved": "https://registry.npmjs.org/glob/-/glob-7.2.3.tgz",
@@ -5667,6 +6089,12 @@
"integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==",
"dev": true
},
+ "node_modules/guid-typescript": {
+ "version": "1.0.9",
+ "resolved": "https://registry.npmjs.org/guid-typescript/-/guid-typescript-1.0.9.tgz",
+ "integrity": "sha512-Y8T4vYhEfwJOTbouREvG+3XDsjr8E3kIr7uf+JZ0BYloFsttiHU0WfvANVsR7TxNUJa/WpCnw/Ino/p+DeBhBQ==",
+ "license": "ISC"
+ },
"node_modules/hachure-fill": {
"version": "0.5.2",
"resolved": "https://registry.npmjs.org/hachure-fill/-/hachure-fill-0.5.2.tgz",
@@ -5746,7 +6174,6 @@
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/ieee754/-/ieee754-1.2.1.tgz",
"integrity": "sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA==",
- "dev": true,
"funding": [
{
"type": "github",
@@ -5820,6 +6247,12 @@
"resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.4.tgz",
"integrity": "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="
},
+ "node_modules/ini": {
+ "version": "1.3.8",
+ "resolved": "https://registry.npmjs.org/ini/-/ini-1.3.8.tgz",
+ "integrity": "sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew==",
+ "license": "ISC"
+ },
"node_modules/internmap": {
"version": "2.0.3",
"resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
@@ -6396,6 +6829,12 @@
"resolved": "https://registry.npmjs.org/lodash.some/-/lodash.some-4.6.0.tgz",
"integrity": "sha512-j7MJE+TuT51q9ggt4fSgVqro163BEFjAt3u97IqU+JA2DkWl80nFTrowzLpZ/BnpN7rrl0JA/593NAdd8p/scQ=="
},
+ "node_modules/long": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmjs.org/long/-/long-4.0.0.tgz",
+ "integrity": "sha512-XsP+KhQif4bjX1kbuSiySJFNAehNxgLb6hPRGJ9QsUr8ajHkuXGdrHmFUTUUXhDwVX2R5bY4JNZEwbUiMhV+MA==",
+ "license": "Apache-2.0"
+ },
"node_modules/loose-envify": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz",
@@ -6438,6 +6877,10 @@
"yallist": "^2.1.2"
}
},
+ "node_modules/magic-alt": {
+ "resolved": "plugins/magic-alt",
+ "link": true
+ },
"node_modules/magic-string": {
"version": "0.30.11",
"resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.30.11.tgz",
@@ -6628,6 +7071,12 @@
"mkdirp": "bin/cmd.js"
}
},
+ "node_modules/mkdirp-classic": {
+ "version": "0.5.3",
+ "resolved": "https://registry.npmjs.org/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz",
+ "integrity": "sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A==",
+ "license": "MIT"
+ },
"node_modules/mri": {
"version": "1.2.0",
"resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz",
@@ -6670,6 +7119,12 @@
"node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
}
},
+ "node_modules/napi-build-utils": {
+ "version": "1.0.2",
+ "resolved": "https://registry.npmjs.org/napi-build-utils/-/napi-build-utils-1.0.2.tgz",
+ "integrity": "sha512-ONmRUqK7zj7DWX0D9ADe03wbwOBZxNAfF20PlGfCWQcD3+/MakShIHrMqx9YwPTfxDdF1zLeL+RGZiR9kGMLdg==",
+ "license": "MIT"
+ },
"node_modules/natural-compare": {
"version": "1.4.0",
"resolved": "https://registry.npmjs.org/natural-compare/-/natural-compare-1.4.0.tgz",
@@ -6691,6 +7146,18 @@
"node-gyp-build": "^4.2.2"
}
},
+ "node_modules/node-abi": {
+ "version": "3.68.0",
+ "resolved": "https://registry.npmjs.org/node-abi/-/node-abi-3.68.0.tgz",
+ "integrity": "sha512-7vbj10trelExNjFSBm5kTvZXXa7pZyKWx9RCKIyqe6I9Ev3IzGpQoqBP3a+cOdxY+pWj6VkP28n/2wWysBHD/A==",
+ "license": "MIT",
+ "dependencies": {
+ "semver": "^7.3.5"
+ },
+ "engines": {
+ "node": ">=10"
+ }
+ },
"node_modules/node-addon-api": {
"version": "3.2.1",
"resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-3.2.1.tgz",
@@ -6862,6 +7329,50 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/onnx-proto": {
+ "version": "4.0.4",
+ "resolved": "https://registry.npmjs.org/onnx-proto/-/onnx-proto-4.0.4.tgz",
+ "integrity": "sha512-aldMOB3HRoo6q/phyB6QRQxSt895HNNw82BNyZ2CMh4bjeKv7g/c+VpAFtJuEMVfYLMbRx61hbuqnKceLeDcDA==",
+ "license": "MIT",
+ "dependencies": {
+ "protobufjs": "^6.8.8"
+ }
+ },
+ "node_modules/onnxruntime-common": {
+ "version": "1.14.0",
+ "resolved": "https://registry.npmjs.org/onnxruntime-common/-/onnxruntime-common-1.14.0.tgz",
+ "integrity": "sha512-3LJpegM2iMNRX2wUmtYfeX/ytfOzNwAWKSq1HbRrKc9+uqG/FsEA0bbKZl1btQeZaXhC26l44NWpNUeXPII7Ew==",
+ "license": "MIT"
+ },
+ "node_modules/onnxruntime-node": {
+ "version": "1.14.0",
+ "resolved": "https://registry.npmjs.org/onnxruntime-node/-/onnxruntime-node-1.14.0.tgz",
+ "integrity": "sha512-5ba7TWomIV/9b6NH/1x/8QEeowsb+jBEvFzU6z0T4mNsFwdPqXeFUM7uxC6QeSRkEbWu3qEB0VMjrvzN/0S9+w==",
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32",
+ "darwin",
+ "linux"
+ ],
+ "dependencies": {
+ "onnxruntime-common": "~1.14.0"
+ }
+ },
+ "node_modules/onnxruntime-web": {
+ "version": "1.14.0",
+ "resolved": "https://registry.npmjs.org/onnxruntime-web/-/onnxruntime-web-1.14.0.tgz",
+ "integrity": "sha512-Kcqf43UMfW8mCydVGcX9OMXI2VN17c0p6XvR7IPSZzBf/6lteBzXHvcEVWDPmCKuGombl997HgLqj91F11DzXw==",
+ "license": "MIT",
+ "dependencies": {
+ "flatbuffers": "^1.12.0",
+ "guid-typescript": "^1.0.9",
+ "long": "^4.0.0",
+ "onnx-proto": "^4.0.4",
+ "onnxruntime-common": "~1.14.0",
+ "platform": "^1.3.6"
+ }
+ },
"node_modules/optionator": {
"version": "0.9.4",
"resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
@@ -7110,6 +7621,12 @@
"nice-napi": "^1.0.2"
}
},
+ "node_modules/platform": {
+ "version": "1.3.6",
+ "resolved": "https://registry.npmjs.org/platform/-/platform-1.3.6.tgz",
+ "integrity": "sha512-fnWVljUchTro6RiCFvCXBbNhJc2NijN7oIQxbwsyL0buWJPG85v81ehlHI9fXrJsMNgTofEoWIQeClKpgxFLrg==",
+ "license": "MIT"
+ },
"node_modules/points-on-curve": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/points-on-curve/-/points-on-curve-0.2.0.tgz",
@@ -7282,13 +7799,76 @@
"integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==",
"dev": true
},
- "node_modules/prelude-ls": {
- "version": "1.2.1",
- "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
- "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
- "dev": true,
+ "node_modules/prebuild-install": {
+ "version": "7.1.2",
+ "resolved": "https://registry.npmjs.org/prebuild-install/-/prebuild-install-7.1.2.tgz",
+ "integrity": "sha512-UnNke3IQb6sgarcZIDU3gbMeTp/9SSU1DAIkil7PrqG1vZlBtY5msYccSKSHDqa3hNg436IXK+SNImReuA1wEQ==",
+ "license": "MIT",
+ "dependencies": {
+ "detect-libc": "^2.0.0",
+ "expand-template": "^2.0.3",
+ "github-from-package": "0.0.0",
+ "minimist": "^1.2.3",
+ "mkdirp-classic": "^0.5.3",
+ "napi-build-utils": "^1.0.1",
+ "node-abi": "^3.3.0",
+ "pump": "^3.0.0",
+ "rc": "^1.2.7",
+ "simple-get": "^4.0.0",
+ "tar-fs": "^2.0.0",
+ "tunnel-agent": "^0.6.0"
+ },
+ "bin": {
+ "prebuild-install": "bin.js"
+ },
"engines": {
- "node": ">= 0.8.0"
+ "node": ">=10"
+ }
+ },
+ "node_modules/prebuild-install/node_modules/detect-libc": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz",
+ "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==",
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/prebuild-install/node_modules/tar-fs": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-2.1.1.tgz",
+ "integrity": "sha512-V0r2Y9scmbDRLCNex/+hYzvp/zyYjvFbHPNgVTKfQvVrb6guiE/fxP+XblDNR011utopbkex2nM4dHNV6GDsng==",
+ "license": "MIT",
+ "dependencies": {
+ "chownr": "^1.1.1",
+ "mkdirp-classic": "^0.5.2",
+ "pump": "^3.0.0",
+ "tar-stream": "^2.1.4"
+ }
+ },
+ "node_modules/prebuild-install/node_modules/tar-stream": {
+ "version": "2.2.0",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-2.2.0.tgz",
+ "integrity": "sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ==",
+ "license": "MIT",
+ "dependencies": {
+ "bl": "^4.0.3",
+ "end-of-stream": "^1.4.1",
+ "fs-constants": "^1.0.0",
+ "inherits": "^2.0.3",
+ "readable-stream": "^3.1.1"
+ },
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/prelude-ls": {
+ "version": "1.2.1",
+ "resolved": "https://registry.npmjs.org/prelude-ls/-/prelude-ls-1.2.1.tgz",
+ "integrity": "sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g==",
+ "dev": true,
+ "engines": {
+ "node": ">= 0.8.0"
}
},
"node_modules/prettier": {
@@ -7355,6 +7935,32 @@
"react-is": "^16.13.1"
}
},
+ "node_modules/protobufjs": {
+ "version": "6.11.4",
+ "resolved": "https://registry.npmjs.org/protobufjs/-/protobufjs-6.11.4.tgz",
+ "integrity": "sha512-5kQWPaJHi1WoCpjTGszzQ32PG2F4+wRY6BmAT4Vfw56Q2FZ4YZzK20xUYQH4YkfehY1e6QSICrJquM6xXZNcrw==",
+ "hasInstallScript": true,
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "@protobufjs/aspromise": "^1.1.2",
+ "@protobufjs/base64": "^1.1.2",
+ "@protobufjs/codegen": "^2.0.4",
+ "@protobufjs/eventemitter": "^1.1.0",
+ "@protobufjs/fetch": "^1.1.0",
+ "@protobufjs/float": "^1.0.2",
+ "@protobufjs/inquire": "^1.1.0",
+ "@protobufjs/path": "^1.1.2",
+ "@protobufjs/pool": "^1.1.0",
+ "@protobufjs/utf8": "^1.1.0",
+ "@types/long": "^4.0.1",
+ "@types/node": ">=13.7.0",
+ "long": "^4.0.0"
+ },
+ "bin": {
+ "pbjs": "bin/pbjs",
+ "pbts": "bin/pbts"
+ }
+ },
"node_modules/proxy-from-env": {
"version": "1.1.0",
"resolved": "https://registry.npmjs.org/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
@@ -7370,7 +7976,6 @@
"version": "3.0.2",
"resolved": "https://registry.npmjs.org/pump/-/pump-3.0.2.tgz",
"integrity": "sha512-tUPXtzlGM8FE3P0ZL6DVs/3P58k9nk8/jZeQCurTJylQA8qFYzHFfhBJkuqyE0FifOsQ0uKWekiZ5g8wtr28cw==",
- "dev": true,
"dependencies": {
"end-of-stream": "^1.1.0",
"once": "^1.3.1"
@@ -7405,6 +8010,12 @@
}
]
},
+ "node_modules/queue-tick": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/queue-tick/-/queue-tick-1.0.1.tgz",
+ "integrity": "sha512-kJt5qhMxoszgU/62PLP1CJytzd2NKetjSRnyuj31fDd3Rlcz3fzlFdFLD1SItunPwyqEOkca6GbV612BWfaBag==",
+ "license": "MIT"
+ },
"node_modules/quick-lru": {
"version": "5.1.1",
"resolved": "https://registry.npmjs.org/quick-lru/-/quick-lru-5.1.1.tgz",
@@ -7417,6 +8028,30 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/rc": {
+ "version": "1.2.8",
+ "resolved": "https://registry.npmjs.org/rc/-/rc-1.2.8.tgz",
+ "integrity": "sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw==",
+ "license": "(BSD-2-Clause OR MIT OR Apache-2.0)",
+ "dependencies": {
+ "deep-extend": "^0.6.0",
+ "ini": "~1.3.0",
+ "minimist": "^1.2.0",
+ "strip-json-comments": "~2.0.1"
+ },
+ "bin": {
+ "rc": "cli.js"
+ }
+ },
+ "node_modules/rc/node_modules/strip-json-comments": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-2.0.1.tgz",
+ "integrity": "sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
"node_modules/react": {
"version": "18.3.1",
"resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz",
@@ -8094,7 +8729,6 @@
"version": "7.6.3",
"resolved": "https://registry.npmjs.org/semver/-/semver-7.6.3.tgz",
"integrity": "sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A==",
- "dev": true,
"bin": {
"semver": "bin/semver.js"
},
@@ -8129,6 +8763,44 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/sharp": {
+ "version": "0.32.6",
+ "resolved": "https://registry.npmjs.org/sharp/-/sharp-0.32.6.tgz",
+ "integrity": "sha512-KyLTWwgcR9Oe4d9HwCwNM2l7+J0dUQwn/yf7S0EnTtb0eVS4RxO0eUSvxPtzT4F3SY+C4K6fqdv/DO27sJ/v/w==",
+ "hasInstallScript": true,
+ "license": "Apache-2.0",
+ "dependencies": {
+ "color": "^4.2.3",
+ "detect-libc": "^2.0.2",
+ "node-addon-api": "^6.1.0",
+ "prebuild-install": "^7.1.1",
+ "semver": "^7.5.4",
+ "simple-get": "^4.0.1",
+ "tar-fs": "^3.0.4",
+ "tunnel-agent": "^0.6.0"
+ },
+ "engines": {
+ "node": ">=14.15.0"
+ },
+ "funding": {
+ "url": "https://opencollective.com/libvips"
+ }
+ },
+ "node_modules/sharp/node_modules/detect-libc": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.0.3.tgz",
+ "integrity": "sha512-bwy0MGW55bG41VqxxypOsdSdGqLwXPI/focwgTYCFMbdUiBAxLg9CFzG08sz2aqzknwiX7Hkl0bQENjg8iLByw==",
+ "license": "Apache-2.0",
+ "engines": {
+ "node": ">=8"
+ }
+ },
+ "node_modules/sharp/node_modules/node-addon-api": {
+ "version": "6.1.0",
+ "resolved": "https://registry.npmjs.org/node-addon-api/-/node-addon-api-6.1.0.tgz",
+ "integrity": "sha512-+eawOlIgy680F0kBzPUNFhMZGtJ1YmqM6l4+Crf4IkImjYrO/mqPwRMh352g23uIaQKFItcQ64I7KMaJxHgAVA==",
+ "license": "MIT"
+ },
"node_modules/shebang-command": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
@@ -8155,6 +8827,66 @@
"resolved": "https://registry.npmjs.org/signal-exit/-/signal-exit-3.0.7.tgz",
"integrity": "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ=="
},
+ "node_modules/simple-concat": {
+ "version": "1.0.1",
+ "resolved": "https://registry.npmjs.org/simple-concat/-/simple-concat-1.0.1.tgz",
+ "integrity": "sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT"
+ },
+ "node_modules/simple-get": {
+ "version": "4.0.1",
+ "resolved": "https://registry.npmjs.org/simple-get/-/simple-get-4.0.1.tgz",
+ "integrity": "sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA==",
+ "funding": [
+ {
+ "type": "github",
+ "url": "https://github.com/sponsors/feross"
+ },
+ {
+ "type": "patreon",
+ "url": "https://www.patreon.com/feross"
+ },
+ {
+ "type": "consulting",
+ "url": "https://feross.org/support"
+ }
+ ],
+ "license": "MIT",
+ "dependencies": {
+ "decompress-response": "^6.0.0",
+ "once": "^1.3.1",
+ "simple-concat": "^1.0.0"
+ }
+ },
+ "node_modules/simple-swizzle": {
+ "version": "0.2.2",
+ "resolved": "https://registry.npmjs.org/simple-swizzle/-/simple-swizzle-0.2.2.tgz",
+ "integrity": "sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg==",
+ "license": "MIT",
+ "dependencies": {
+ "is-arrayish": "^0.3.1"
+ }
+ },
+ "node_modules/simple-swizzle/node_modules/is-arrayish": {
+ "version": "0.3.2",
+ "resolved": "https://registry.npmjs.org/is-arrayish/-/is-arrayish-0.3.2.tgz",
+ "integrity": "sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ==",
+ "license": "MIT"
+ },
"node_modules/sitemap-urls": {
"version": "3.0.0",
"resolved": "https://registry.npmjs.org/sitemap-urls/-/sitemap-urls-3.0.0.tgz",
@@ -8264,6 +8996,20 @@
"resolved": "https://registry.npmjs.org/spdx-license-ids/-/spdx-license-ids-3.0.20.tgz",
"integrity": "sha512-jg25NiDV/1fLtSgEgyvVyDunvaNHbuwF9lfNV17gSmPFAlYzdfNBlLtLzXTevwkPj7DhGbmN9VnmJIgLnhvaBw=="
},
+ "node_modules/streamx": {
+ "version": "2.20.1",
+ "resolved": "https://registry.npmjs.org/streamx/-/streamx-2.20.1.tgz",
+ "integrity": "sha512-uTa0mU6WUC65iUvzKH4X9hEdvSW7rbPxPtwfWiLMSj3qTdQbAiUboZTxauKfpFuGIGa1C2BYijZ7wgdUXICJhA==",
+ "license": "MIT",
+ "dependencies": {
+ "fast-fifo": "^1.3.2",
+ "queue-tick": "^1.0.1",
+ "text-decoder": "^1.1.0"
+ },
+ "optionalDependencies": {
+ "bare-events": "^2.2.0"
+ }
+ },
"node_modules/string_decoder": {
"version": "1.3.0",
"resolved": "https://registry.npmjs.org/string_decoder/-/string_decoder-1.3.0.tgz",
@@ -8759,6 +9505,12 @@
"url": "https://github.com/fb55/nth-check?sponsor=1"
}
},
+ "node_modules/tabbable": {
+ "version": "6.2.0",
+ "resolved": "https://registry.npmjs.org/tabbable/-/tabbable-6.2.0.tgz",
+ "integrity": "sha512-Cat63mxsVJlzYvN51JmVXIgNoUokrIaT2zLclCXjRd8boZ0004U4KCs/sToJ75C6sdlByWxpYnb5Boif1VSFew==",
+ "license": "MIT"
+ },
"node_modules/tailwindcss": {
"version": "3.4.12",
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.12.tgz",
@@ -8796,6 +9548,40 @@
"node": ">=14.0.0"
}
},
+ "node_modules/tar-fs": {
+ "version": "3.0.6",
+ "resolved": "https://registry.npmjs.org/tar-fs/-/tar-fs-3.0.6.tgz",
+ "integrity": "sha512-iokBDQQkUyeXhgPYaZxmczGPhnhXZ0CmrqI+MOb/WFGS9DW5wnfrLgtjUJBvz50vQ3qfRwJ62QVoCFu8mPVu5w==",
+ "license": "MIT",
+ "dependencies": {
+ "pump": "^3.0.0",
+ "tar-stream": "^3.1.5"
+ },
+ "optionalDependencies": {
+ "bare-fs": "^2.1.1",
+ "bare-path": "^2.1.0"
+ }
+ },
+ "node_modules/tar-stream": {
+ "version": "3.1.7",
+ "resolved": "https://registry.npmjs.org/tar-stream/-/tar-stream-3.1.7.tgz",
+ "integrity": "sha512-qJj60CXt7IU1Ffyc3NJMjh6EkuCFej46zUqJ4J7pqYlThyd9bO0XBTmcOIhSzZJVWfsLks0+nle/j538YAW9RQ==",
+ "license": "MIT",
+ "dependencies": {
+ "b4a": "^1.6.4",
+ "fast-fifo": "^1.2.0",
+ "streamx": "^2.15.0"
+ }
+ },
+ "node_modules/text-decoder": {
+ "version": "1.2.0",
+ "resolved": "https://registry.npmjs.org/text-decoder/-/text-decoder-1.2.0.tgz",
+ "integrity": "sha512-n1yg1mOj9DNpk3NeZOx7T6jchTbyJS3i3cucbNN6FcdPriMZx7NsgrGpWWdWZZGxD7ES1XB+3uoqHMgOKaN+fg==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "b4a": "^1.6.4"
+ }
+ },
"node_modules/text-table": {
"version": "0.2.0",
"resolved": "https://registry.npmjs.org/text-table/-/text-table-0.2.0.tgz",
@@ -8930,6 +9716,18 @@
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.7.0.tgz",
"integrity": "sha512-gLXCKdN1/j47AiHiOkJN69hJmcbGTHI0ImLmbYLHykhgeN0jVGola9yVjFgzCUklsZQMW55o+dW7IXv3RCXDzA=="
},
+ "node_modules/tunnel-agent": {
+ "version": "0.6.0",
+ "resolved": "https://registry.npmjs.org/tunnel-agent/-/tunnel-agent-0.6.0.tgz",
+ "integrity": "sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "safe-buffer": "^5.0.1"
+ },
+ "engines": {
+ "node": "*"
+ }
+ },
"node_modules/turbo": {
"version": "2.1.2",
"resolved": "https://registry.npmjs.org/turbo/-/turbo-2.1.2.tgz",
@@ -10901,6 +11699,229 @@
"react-dom": "^18.2.0"
}
},
+ "plugins/magic-alt": {
+ "version": "0.0.0",
+ "dependencies": {
+ "@headlessui/react": "^2.1.8",
+ "@xenova/transformers": "^2.17.2",
+ "classnames": "^2.5.1",
+ "framer-motion": "^11.5.4",
+ "framer-plugin": "^0.3.1",
+ "react": "^18",
+ "react-dom": "^18",
+ "react-intersection-observer": "^9.13.1",
+ "vite-plugin-mkcert": "^1"
+ },
+ "devDependencies": {
+ "@types/react": "^18",
+ "@types/react-dom": "^18",
+ "@typescript-eslint/eslint-plugin": "^7",
+ "@typescript-eslint/parser": "^7",
+ "@vitejs/plugin-react-swc": "^3",
+ "autoprefixer": "^10.4.19",
+ "eslint": "^8",
+ "eslint-plugin-react-hooks": "^4",
+ "eslint-plugin-react-refresh": "^0",
+ "postcss": "^8.4.38",
+ "tailwindcss": "^3.4.3",
+ "typescript": "^5.3.3",
+ "vite": "^5",
+ "vite-plugin-framer": "^0"
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/eslint-plugin": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/eslint-plugin/-/eslint-plugin-7.18.0.tgz",
+ "integrity": "sha512-94EQTWZ40mzBc42ATNIBimBEDltSJ9RQHCC8vc/PDbxi4k8dVwUAv4o98dk50M1zB+JGFxp43FP7f8+FP8R6Sw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/regexpp": "^4.10.0",
+ "@typescript-eslint/scope-manager": "7.18.0",
+ "@typescript-eslint/type-utils": "7.18.0",
+ "@typescript-eslint/utils": "7.18.0",
+ "@typescript-eslint/visitor-keys": "7.18.0",
+ "graphemer": "^1.4.0",
+ "ignore": "^5.3.1",
+ "natural-compare": "^1.4.0",
+ "ts-api-utils": "^1.3.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "@typescript-eslint/parser": "^7.0.0",
+ "eslint": "^8.56.0"
+ },
+ "peerDependenciesMeta": {
+ "typescript": {
+ "optional": true
+ }
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/parser": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/parser/-/parser-7.18.0.tgz",
+ "integrity": "sha512-4Z+L8I2OqhZV8qA132M4wNL30ypZGYOQVBfMgxDH/K5UX0PNqTu1c6za9ST5r9+tavvHiTWmBnKzpCJ/GlVFtg==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "@typescript-eslint/scope-manager": "7.18.0",
+ "@typescript-eslint/types": "7.18.0",
+ "@typescript-eslint/typescript-estree": "7.18.0",
+ "@typescript-eslint/visitor-keys": "7.18.0",
+ "debug": "^4.3.4"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.56.0"
+ },
+ "peerDependenciesMeta": {
+ "typescript": {
+ "optional": true
+ }
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/scope-manager": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/scope-manager/-/scope-manager-7.18.0.tgz",
+ "integrity": "sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/types": "7.18.0",
+ "@typescript-eslint/visitor-keys": "7.18.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/type-utils": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/type-utils/-/type-utils-7.18.0.tgz",
+ "integrity": "sha512-XL0FJXuCLaDuX2sYqZUUSOJ2sG5/i1AAze+axqmLnSkNEVMVYLF+cbwlB2w8D1tinFuSikHmFta+P+HOofrLeA==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/typescript-estree": "7.18.0",
+ "@typescript-eslint/utils": "7.18.0",
+ "debug": "^4.3.4",
+ "ts-api-utils": "^1.3.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.56.0"
+ },
+ "peerDependenciesMeta": {
+ "typescript": {
+ "optional": true
+ }
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/types": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/types/-/types-7.18.0.tgz",
+ "integrity": "sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/typescript-estree": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/typescript-estree/-/typescript-estree-7.18.0.tgz",
+ "integrity": "sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA==",
+ "dev": true,
+ "license": "BSD-2-Clause",
+ "dependencies": {
+ "@typescript-eslint/types": "7.18.0",
+ "@typescript-eslint/visitor-keys": "7.18.0",
+ "debug": "^4.3.4",
+ "globby": "^11.1.0",
+ "is-glob": "^4.0.3",
+ "minimatch": "^9.0.4",
+ "semver": "^7.6.0",
+ "ts-api-utils": "^1.3.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependenciesMeta": {
+ "typescript": {
+ "optional": true
+ }
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/utils": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/utils/-/utils-7.18.0.tgz",
+ "integrity": "sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@eslint-community/eslint-utils": "^4.4.0",
+ "@typescript-eslint/scope-manager": "7.18.0",
+ "@typescript-eslint/types": "7.18.0",
+ "@typescript-eslint/typescript-estree": "7.18.0"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ },
+ "peerDependencies": {
+ "eslint": "^8.56.0"
+ }
+ },
+ "plugins/magic-alt/node_modules/@typescript-eslint/visitor-keys": {
+ "version": "7.18.0",
+ "resolved": "https://registry.npmjs.org/@typescript-eslint/visitor-keys/-/visitor-keys-7.18.0.tgz",
+ "integrity": "sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@typescript-eslint/types": "7.18.0",
+ "eslint-visitor-keys": "^3.4.3"
+ },
+ "engines": {
+ "node": "^18.18.0 || >=20.0.0"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/typescript-eslint"
+ }
+ },
"plugins/notion": {
"version": "0.0.0",
"dependencies": {
diff --git a/plugins/magic-alt/.eslintrc.cjs b/plugins/magic-alt/.eslintrc.cjs
new file mode 100644
index 00000000..9358989c
--- /dev/null
+++ b/plugins/magic-alt/.eslintrc.cjs
@@ -0,0 +1,11 @@
+module.exports = {
+ root: true,
+ env: { browser: true, es2020: true },
+ extends: ["eslint:recommended", "plugin:@typescript-eslint/recommended", "plugin:react-hooks/recommended"],
+ ignorePatterns: ["dist", ".eslintrc.cjs"],
+ parser: "@typescript-eslint/parser",
+ plugins: ["react-refresh"],
+ rules: {
+ "react-refresh/only-export-components": ["warn", { allowConstantExport: true }],
+ },
+}
diff --git a/plugins/magic-alt/.gitignore b/plugins/magic-alt/.gitignore
new file mode 100644
index 00000000..94469931
--- /dev/null
+++ b/plugins/magic-alt/.gitignore
@@ -0,0 +1,24 @@
+# See https://help.github.com/articles/ignoring-files/ for more about ignoring files.
+
+# dependencies
+/node_modules
+/.pnp
+.pnp.js
+.yarn
+
+# misc
+.DS_Store
+*.pem
+
+# files
+my-plugin
+dev-plugin
+dist
+
+# debug
+npm-debug.log*
+yarn-debug.log*
+yarn-error.log*
+
+# local env files
+.env*.local
diff --git a/plugins/magic-alt/README.md b/plugins/magic-alt/README.md
new file mode 100644
index 00000000..9daba0c0
--- /dev/null
+++ b/plugins/magic-alt/README.md
@@ -0,0 +1,21 @@
+# Framer Plugin Template
+
+This is a template for using the Framer Plugin API in a TypeScript project.
+
+## Quickstart
+
+First, run the development server:
+
+```bash
+npm run dev
+# or
+yarn dev
+# or
+pnpm dev
+# or
+bun dev
+```
+
+To see the results, open Framer and ensure the "Plugin Beta" preference is enabled. Then open the plugins menu from the toolbar, and select "Open Development Plugin"
+
+Learn more: https://developers.framer.wiki/
diff --git a/plugins/magic-alt/framer.json b/plugins/magic-alt/framer.json
new file mode 100644
index 00000000..9a48ab00
--- /dev/null
+++ b/plugins/magic-alt/framer.json
@@ -0,0 +1,6 @@
+{
+ "id": "a414e2",
+ "name": "Magic Alt",
+ "modes": ["canvas", "editImage"],
+ "icon": "/icon.svg"
+}
diff --git a/plugins/magic-alt/index.html b/plugins/magic-alt/index.html
new file mode 100644
index 00000000..17896d4f
--- /dev/null
+++ b/plugins/magic-alt/index.html
@@ -0,0 +1,14 @@
+
+
+
+
+
+
+ Magic Alt
+
+
+
+
+
+
+
diff --git a/plugins/magic-alt/package.json b/plugins/magic-alt/package.json
new file mode 100644
index 00000000..539877f5
--- /dev/null
+++ b/plugins/magic-alt/package.json
@@ -0,0 +1,39 @@
+{
+ "name": "magic-alt",
+ "private": true,
+ "version": "0.0.0",
+ "type": "module",
+ "scripts": {
+ "dev": "vite",
+ "build": "vite build --base=${PREFIX_BASE_PATH:+/$npm_package_name}/",
+ "lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0",
+ "preview": "vite preview"
+ },
+ "dependencies": {
+ "@headlessui/react": "^2.1.8",
+ "@xenova/transformers": "^2.17.2",
+ "classnames": "^2.5.1",
+ "framer-motion": "^11.5.4",
+ "framer-plugin": "^0.3.1",
+ "react": "^18",
+ "react-dom": "^18",
+ "react-intersection-observer": "^9.13.1",
+ "vite-plugin-mkcert": "^1"
+ },
+ "devDependencies": {
+ "@types/react": "^18",
+ "@types/react-dom": "^18",
+ "@typescript-eslint/eslint-plugin": "^7",
+ "@typescript-eslint/parser": "^7",
+ "@vitejs/plugin-react-swc": "^3",
+ "autoprefixer": "^10.4.19",
+ "eslint": "^8",
+ "eslint-plugin-react-hooks": "^4",
+ "eslint-plugin-react-refresh": "^0",
+ "postcss": "^8.4.38",
+ "tailwindcss": "^3.4.3",
+ "typescript": "^5.3.3",
+ "vite": "^5",
+ "vite-plugin-framer": "^0"
+ }
+}
diff --git a/plugins/magic-alt/postcss.config.js b/plugins/magic-alt/postcss.config.js
new file mode 100644
index 00000000..d41ad635
--- /dev/null
+++ b/plugins/magic-alt/postcss.config.js
@@ -0,0 +1,6 @@
+export default {
+ plugins: {
+ tailwindcss: {},
+ autoprefixer: {},
+ },
+}
diff --git a/plugins/magic-alt/public/icon.svg b/plugins/magic-alt/public/icon.svg
new file mode 100644
index 00000000..cf637da4
--- /dev/null
+++ b/plugins/magic-alt/public/icon.svg
@@ -0,0 +1,5 @@
+
+
+
+
+
diff --git a/plugins/magic-alt/src/App.tsx b/plugins/magic-alt/src/App.tsx
new file mode 100644
index 00000000..c722f6c8
--- /dev/null
+++ b/plugins/magic-alt/src/App.tsx
@@ -0,0 +1,319 @@
+import { framer } from "framer-plugin"
+import { useState, useMemo, useCallback, useEffect } from "react"
+import { useInView } from "react-intersection-observer"
+import cx from "classnames"
+import { useTranslation } from "./hooks/useTranslation"
+import { FrameNodeWithImage, useNoAltImages } from "./hooks/useNoAltImages"
+import { generateCaptions } from "./api"
+import { removeItemAtIndex } from "./utils"
+import { LANGUAGES } from "./constants"
+
+import { Spinner } from "./components/Spinner"
+import { EllipsisIcon, MagicIcon } from "./components/Icons"
+import { DropdownMenu } from "./components/DropdownMenu"
+import { IconButton } from "./components/IconButton"
+import { LanguageSelector } from "./components/LanguageSelector"
+
+interface ImageCaptionItem {
+ image: FrameNodeWithImage
+ caption: string
+ isLoading: boolean
+}
+
+const CAPTION_GENERATION_BATCH_SIZE = 5
+const TRANSLATION_BATCH_SIZE = 2
+
+const ENGLISH_LANGUAGE_INDEX = 24
+const DEFAULT_LANGUAGE = LANGUAGES[ENGLISH_LANGUAGE_INDEX]
+
+export function App() {
+ const { ref: scrollRef, inView: isAtBottom } = useInView({ threshold: 1 })
+ const { loadModel, modelLoadProgress, isModelLoading, isModelReady, translate, isTranslating } = useTranslation()
+
+ const [images] = useNoAltImages()
+ const [imageCaptionItems, setImageCaptionItems] = useState([])
+ const isGeneratingCaptions = useMemo(() => imageCaptionItems.some(item => item.isLoading), [imageCaptionItems])
+ const allCaptionsBlank = useMemo(() => imageCaptionItems.every(item => item.caption === ""), [imageCaptionItems])
+
+ const [newlyTranslatedIndices, setNewlyTranslatedIndices] = useState([])
+ const [translatingIndices, setTranslatingIndices] = useState([])
+
+ const [targetLanguage, setTargetLanguage] = useState(DEFAULT_LANGUAGE)
+ const [sourceLanguage, setSourceLanguage] = useState(DEFAULT_LANGUAGE)
+ const requiresTranslation = useMemo(
+ () => sourceLanguage.value !== targetLanguage.value,
+ [sourceLanguage, targetLanguage]
+ )
+
+ useEffect(() => {
+ // Initialize image caption items when the images array changes
+ const items = images.map(image => ({
+ image,
+ caption: "",
+ isLoading: false,
+ }))
+
+ setImageCaptionItems(items)
+ }, [images])
+
+ useEffect(() => {
+ if (requiresTranslation && !isModelLoading && !isModelReady) {
+ loadModel()
+ framer.notify("Loading translation model", { variant: "info" })
+ }
+ }, [requiresTranslation, isModelLoading, isModelReady, loadModel])
+
+ const translateCaptions = useCallback(async () => {
+ if (!requiresTranslation || !isModelReady || isTranslating) return
+
+ for (let i = 0; i < imageCaptionItems.length; i += TRANSLATION_BATCH_SIZE) {
+ const batch = imageCaptionItems.slice(i, i + TRANSLATION_BATCH_SIZE).map(item => item.caption)
+ const batchIndices = Array.from({ length: TRANSLATION_BATCH_SIZE }, (_, index) => i + index).filter(
+ index => index < imageCaptionItems.length && imageCaptionItems[index].caption.trim() !== ""
+ )
+
+ if (batch.length === 0) continue
+
+ // Update UI for captions being translated
+ setTranslatingIndices(prevIndices => [...prevIndices, ...batchIndices])
+
+ try {
+ const translatedBatch = await translate(batch, sourceLanguage.value, targetLanguage.value)
+
+ // Update captions with their translated text
+ setImageCaptionItems(prevItems => {
+ return prevItems.map((item, index) =>
+ batchIndices.includes(index)
+ ? { ...item, caption: translatedBatch[batchIndices.indexOf(index)] }
+ : item
+ )
+ })
+
+ setNewlyTranslatedIndices(prev => [...prev, ...batchIndices])
+ } catch (e) {
+ framer.notify(e instanceof Error ? e.message : String(e), { variant: "error" })
+ } finally {
+ // Captions are now translated, remove them
+ setTranslatingIndices(prevIndices => prevIndices.filter(index => !batchIndices.includes(index)))
+ }
+ }
+
+ setSourceLanguage(targetLanguage)
+ }, [imageCaptionItems, requiresTranslation, isModelReady, isTranslating, translate, sourceLanguage, targetLanguage])
+
+ useEffect(() => {
+ // Check for newly translated indices
+ if (newlyTranslatedIndices.length > 0) {
+ // Allow for animation to occur then remove
+ const timer = setTimeout(() => {
+ setNewlyTranslatedIndices([])
+ }, 500)
+
+ return () => clearTimeout(timer)
+ }
+ }, [newlyTranslatedIndices])
+
+ const handleTranslate = useCallback(async () => {
+ if (!requiresTranslation) {
+ framer.notify("Please select a different language", { variant: "warning" })
+ return
+ }
+
+ if (!isModelReady) {
+ framer.notify("Loading translation model", { variant: "info" })
+ return
+ }
+
+ if (allCaptionsBlank) return
+
+ await translateCaptions()
+ }, [requiresTranslation, isModelReady, translateCaptions, allCaptionsBlank])
+
+ const handleCaptionsGeneration = useCallback(async () => {
+ const publishInfo = await framer.getPublishInfo()
+ const siteUrl = publishInfo.staging?.url
+
+ if (!siteUrl) {
+ framer.notify("Please publish your site to staging", { variant: "error" })
+ return
+ }
+
+ setNewlyTranslatedIndices([])
+ setSourceLanguage(DEFAULT_LANGUAGE)
+ setImageCaptionItems(prevItems =>
+ prevItems.map(item => ({
+ image: item.image,
+ isLoading: true,
+ caption: "",
+ }))
+ )
+
+ for (let i = 0; i < images.length; i += CAPTION_GENERATION_BATCH_SIZE) {
+ const endIndex = Math.min(i + CAPTION_GENERATION_BATCH_SIZE, images.length)
+ const imageBatch = images.slice(i, endIndex).map(img => img.backgroundImage.url)
+
+ try {
+ const captionBatch = await generateCaptions(siteUrl, imageBatch)
+ setImageCaptionItems(prevItems => {
+ return prevItems.map((item, index) => {
+ // Update items in current batch with new captions
+ if (index >= i && index < i + captionBatch.data.length) {
+ return {
+ ...item,
+ caption: captionBatch.data[index - i].caption,
+ isLoading: false,
+ }
+ }
+
+ return item
+ })
+ })
+ } catch (e) {
+ framer.notify(e instanceof Error ? e.message : String(e), { variant: "error" })
+ }
+ }
+ }, [images])
+
+ const handleSaveCaptions = async () => {
+ await Promise.all(
+ imageCaptionItems.map(({ image }, i) =>
+ image.setAttributes({
+ backgroundImage: image.backgroundImage.cloneWithAttributes({
+ altText: imageCaptionItems[i].caption,
+ }),
+ })
+ )
+ )
+
+ await framer.closePlugin("Alt text applied", { variant: "success" })
+ }
+
+ const handleUpdateCaption = (index: number, value: string) => {
+ setImageCaptionItems(prevItems =>
+ prevItems.map((item, i) => (i === index ? { ...item, caption: value } : item))
+ )
+ }
+
+ if (images.length === 0) {
+ return (
+
+
All images have Alt Text
+
+ )
+ }
+
+ return (
+
+
+
setTargetLanguage(lang)}
+ disabled={isTranslating || isGeneratingCaptions}
+ className="flex-1"
+ />
+
+
+ Translate
+
+
+
+
+ {imageCaptionItems.map(({ image, isLoading, caption }, i) => (
+
+
framer.zoomIntoView([image.id], { maxZoom: 2 })}
+ >
+
+
+
+
+ {isLoading ? (
+
+
+
+ ) : null}
+
handleUpdateCaption(i, e.target.value)}
+ disabled={isTranslating || isGeneratingCaptions}
+ />
+
+
+
setImageCaptionItems(prevItems => removeItemAtIndex(prevItems, i)),
+ },
+ ]}
+ >
+ {({ isOpen }) => (
+
+
+
+ )}
+
+
+ ))}
+
+
+ {!isAtBottom &&
}
+
+
+
+
+ Apply
+
+
+ Generate
+
+
+
+
+ )
+}
diff --git a/plugins/magic-alt/src/api.ts b/plugins/magic-alt/src/api.ts
new file mode 100644
index 00000000..7546f127
--- /dev/null
+++ b/plugins/magic-alt/src/api.ts
@@ -0,0 +1,34 @@
+interface Generation {
+ caption: string
+ image: string
+}
+
+interface GeneratedCaptionResponse {
+ data: Generation[]
+ error: {
+ message: string
+ } | null
+}
+
+const isLocal = () => window.location.hostname.includes("localhost")
+
+const API_URL = isLocal() ? "http://localhost:8787" : "https://magic-alt-plugin-api.sakibulislam25800.workers.dev"
+
+export async function generateCaptions(siteUrl: string, images: string[]): Promise {
+ const res = await fetch(`${API_URL}/generate-alt-text`, {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({ siteUrl, images }),
+ })
+
+ const json = await res.json()
+ const error = json.error
+
+ if (error) {
+ throw new Error(error.message)
+ }
+
+ return json
+}
diff --git a/plugins/magic-alt/src/components/DropdownMenu.tsx b/plugins/magic-alt/src/components/DropdownMenu.tsx
new file mode 100644
index 00000000..6cec3302
--- /dev/null
+++ b/plugins/magic-alt/src/components/DropdownMenu.tsx
@@ -0,0 +1,51 @@
+import React from "react"
+import { Menu, MenuButton, MenuItem, MenuItems } from "@headlessui/react"
+
+interface MenuItemType {
+ label: string
+ onClick: () => void
+ icon?: React.ReactNode
+}
+
+interface DropdownMenuProps {
+ menuItems: MenuItemType[]
+ children: (props: { isOpen: boolean }) => React.ReactNode
+}
+
+export const DropdownMenu: React.FC = ({ children, menuItems }) => {
+ return (
+
+ {({ open }) => (
+ <>
+ {children({ isOpen: open })}
+
+ {menuItems.map((item, index) => (
+
+ {({ close }) => (
+ {
+ item.onClick()
+ close()
+ }}
+ >
+ {item.icon && {item.icon} }
+
+ {item.label}
+
+
+ )}
+
+ ))}
+
+ >
+ )}
+
+ )
+}
diff --git a/plugins/magic-alt/src/components/IconButton.tsx b/plugins/magic-alt/src/components/IconButton.tsx
new file mode 100644
index 00000000..31d1bf1e
--- /dev/null
+++ b/plugins/magic-alt/src/components/IconButton.tsx
@@ -0,0 +1,15 @@
+import React, { forwardRef } from "react"
+import cx from "classnames"
+
+interface Props extends React.ButtonHTMLAttributes {
+ children: React.ReactNode
+ className?: string
+}
+
+export const IconButton = forwardRef(({ className, children, ...rest }, ref) => {
+ return (
+
+ {children}
+
+ )
+})
diff --git a/plugins/magic-alt/src/components/Icons.tsx b/plugins/magic-alt/src/components/Icons.tsx
new file mode 100644
index 00000000..a48df471
--- /dev/null
+++ b/plugins/magic-alt/src/components/Icons.tsx
@@ -0,0 +1,29 @@
+export const EllipsisIcon = () => (
+
+
+
+
+
+)
+
+export const MagicIcon = () => (
+
+
+
+
+)
diff --git a/plugins/magic-alt/src/components/LanguageSelector.tsx b/plugins/magic-alt/src/components/LanguageSelector.tsx
new file mode 100644
index 00000000..8fdf35e6
--- /dev/null
+++ b/plugins/magic-alt/src/components/LanguageSelector.tsx
@@ -0,0 +1,117 @@
+import { useState, useEffect } from "react"
+import { Combobox, ComboboxInput, ComboboxOptions, ComboboxOption, ComboboxButton } from "@headlessui/react"
+import cx from "classnames"
+import { LANGUAGES } from "../constants"
+
+interface Language {
+ value: string
+ label: string
+ code: string
+}
+
+interface Props {
+ language: Language
+ onChange: (language: Language) => void
+ className?: string
+ disabled?: boolean
+}
+
+const LanguageCodeBadge = ({ language, className }: { language: Language; className?: string }) => (
+
+
{language.code.toUpperCase()}
+
+)
+
+export const LanguageSelector = ({ language, onChange, className, disabled }: Props) => {
+ const [query, setQuery] = useState(language.label)
+ const [filteredLanguages, setFilteredLanguages] = useState(LANGUAGES)
+
+ useEffect(() => {
+ const filtered =
+ query === ""
+ ? LANGUAGES
+ : LANGUAGES.filter(lang => {
+ const lowercaseQuery = query.toLowerCase()
+ const lowercaseLabel = lang.label.toLowerCase()
+
+ // Show all the languages when there's an exact match
+ if (lowercaseLabel === lowercaseQuery) {
+ return true
+ }
+
+ return lowercaseLabel.includes(lowercaseQuery)
+ })
+
+ setFilteredLanguages(filtered)
+ }, [query])
+
+ return (
+
+
{
+ if (inputLanguage) {
+ onChange(inputLanguage)
+ } else {
+ // Keep the currently selected language if the
+ // input is blank
+ onChange(language)
+ }
+ }}
+ >
+
+
+ lang.label}
+ onChange={e => setQuery(e.target.value)}
+ onFocus={() => setFilteredLanguages(LANGUAGES)}
+ className="w-full !pl-[31px]"
+ />
+
+
+
+
+
+ {filteredLanguages.length > 0 && (
+
+
+
+ {filteredLanguages.map(lang => (
+
+ cx("cursor-pointer rounded-[5px] text-xs", {
+ "bg-white dark:bg-tertiary": !focus,
+ "bg-tint text-white": focus,
+ })
+ }
+ >
+ {({ focus }) => (
+
+
+ {lang.label}
+
+ {lang.code}
+
+ )}
+
+ ))}
+
+
+
+ )}
+
+
+ )
+}
diff --git a/plugins/magic-alt/src/components/Spinner.tsx b/plugins/magic-alt/src/components/Spinner.tsx
new file mode 100644
index 00000000..594ad9d9
--- /dev/null
+++ b/plugins/magic-alt/src/components/Spinner.tsx
@@ -0,0 +1,40 @@
+import cx from "classnames"
+import styles from "./spinner.module.css"
+
+export interface SpinnerProps {
+ /** Size of the spinner */
+ size?: "normal" | "medium" | "large"
+ /** Set the spinner to have a static position inline with other content */
+ inline?: boolean
+ className?: string
+ inheritColor?: boolean
+}
+
+function styleForSize(size: SpinnerProps["size"]) {
+ switch (size) {
+ case "normal":
+ return styles.normalStyle
+ case "medium":
+ return styles.mediumStyle
+ case "large":
+ return styles.largeStyle
+ }
+}
+
+function spinnerClassNames(size: SpinnerProps["size"] = "normal") {
+ return cx(styles.spin, styles.baseStyle, styleForSize(size))
+}
+
+export const Spinner = ({ size, inline = false, inheritColor, className, ...rest }: SpinnerProps) => {
+ return (
+
+ )
+}
diff --git a/plugins/magic-alt/src/components/spinner.module.css b/plugins/magic-alt/src/components/spinner.module.css
new file mode 100644
index 00000000..a0d1a7ac
--- /dev/null
+++ b/plugins/magic-alt/src/components/spinner.module.css
@@ -0,0 +1,60 @@
+.baseStyle {
+ --spinner-translate: 0;
+ background-color: #fff;
+}
+
+.buttonWithDepthSpinner {
+ background-color: currentColor;
+}
+
+.normalStyle {
+ width: 12px;
+ height: 12px;
+ -webkit-mask: url("");
+ mask: url("");
+ -webkit-mask-size: 12px;
+ mask-size: 12px;
+}
+
+.mediumStyle {
+ width: 24px;
+ height: 24px;
+ -webkit-mask: url("");
+ mask: url("");
+ -webkit-mask-size: 24px;
+ mask-size: 24px;
+}
+
+.largeStyle {
+ width: 30px;
+ height: 30px;
+ -webkit-mask: url("");
+ mask: url("");
+ -webkit-mask-size: 30px;
+ mask-size: 30px;
+}
+
+.centeredStyle {
+ --spinner-translate: -50%;
+ position: absolute;
+ top: 50%;
+ left: 50%;
+ transform: translate(var(--spinner-translate), var(--spinner-translate));
+}
+
+.spin {
+ animation-duration: 800ms;
+ animation-iteration-count: infinite;
+ animation-name: spin;
+ animation-timing-function: linear;
+}
+
+@keyframes spin {
+ 0% {
+ transform: translate(var(--spinner-translate), var(--spinner-translate)) rotate(0deg);
+ }
+
+ 100% {
+ transform: translate(var(--spinner-translate), var(--spinner-translate)) rotate(360deg);
+ }
+}
diff --git a/plugins/magic-alt/src/constants.ts b/plugins/magic-alt/src/constants.ts
new file mode 100644
index 00000000..a4cbed92
--- /dev/null
+++ b/plugins/magic-alt/src/constants.ts
@@ -0,0 +1,159 @@
+// List of languages in FLORES-200 is available here:
+// https://github.com/facebookresearch/flores/blob/main/flores200/README.md#languages-in-flores-200
+
+export interface Language {
+ label: string
+ value: string
+ code: string
+}
+
+export const LANGUAGES: Language[] = [
+ { label: "Afrikaans", value: "afr_Latn", code: "af" },
+ { label: "Akan", value: "aka_Latn", code: "ak" },
+ { label: "Amharic", value: "amh_Ethi", code: "am" },
+ { label: "Armenian", value: "hye_Armn", code: "hy" },
+ { label: "Assamese", value: "asm_Beng", code: "as" },
+ { label: "Bambara", value: "bam_Latn", code: "bm" },
+ { label: "Bashkir", value: "bak_Cyrl", code: "ba" },
+ { label: "Basque", value: "eus_Latn", code: "eu" },
+ { label: "Belarusian", value: "bel_Cyrl", code: "be" },
+ { label: "Bengali", value: "ben_Beng", code: "bn" },
+ { label: "Bosnian", value: "bos_Latn", code: "bs" },
+ { label: "Bulgarian", value: "bul_Cyrl", code: "bg" },
+ { label: "Burmese", value: "mya_Mymr", code: "my" },
+ { label: "Catalan", value: "cat_Latn", code: "ca" },
+ { label: "Central Aymara", value: "ayr_Latn", code: "ay" },
+ { label: "Chinese (Simplified)", value: "zho_Hans", code: "zh" },
+ { label: "Chinese (Traditional)", value: "zho_Hant", code: "zh" },
+ { label: "Croatian", value: "hrv_Latn", code: "hr" },
+ { label: "Czech", value: "ces_Latn", code: "cs" },
+ { label: "Danish", value: "dan_Latn", code: "da" },
+ { label: "Dutch", value: "nld_Latn", code: "nl" },
+ { label: "Dzongkha", value: "dzo_Tibt", code: "dz" },
+ { label: "Eastern Panjabi", value: "pan_Guru", code: "pa" },
+ { label: "Eastern Yiddish", value: "ydd_Hebr", code: "yi" },
+ { label: "English", value: "eng_Latn", code: "en" },
+ { label: "Esperanto", value: "epo_Latn", code: "eo" },
+ { label: "Estonian", value: "est_Latn", code: "et" },
+ { label: "Ewe", value: "ewe_Latn", code: "ee" },
+ { label: "Faroese", value: "fao_Latn", code: "fo" },
+ { label: "Fijian", value: "fij_Latn", code: "fj" },
+ { label: "Finnish", value: "fin_Latn", code: "fi" },
+ { label: "Fon", value: "fon_Latn", code: "fon" },
+ { label: "French", value: "fra_Latn", code: "fr" },
+ { label: "Galician", value: "glg_Latn", code: "gl" },
+ { label: "Ganda", value: "lug_Latn", code: "lg" },
+ { label: "Georgian", value: "kat_Geor", code: "ka" },
+ { label: "German", value: "deu_Latn", code: "de" },
+ { label: "Greek", value: "ell_Grek", code: "el" },
+ { label: "Guarani", value: "grn_Latn", code: "gn" },
+ { label: "Gujarati", value: "guj_Gujr", code: "gu" },
+ { label: "Haitian Creole", value: "hat_Latn", code: "ht" },
+ { label: "Halh Mongolian", value: "khk_Cyrl", code: "mn" },
+ { label: "Hausa", value: "hau_Latn", code: "ha" },
+ { label: "Hebrew", value: "heb_Hebr", code: "he" },
+ { label: "Hindi", value: "hin_Deva", code: "hi" },
+ { label: "Hungarian", value: "hun_Latn", code: "hu" },
+ { label: "Icelandic", value: "isl_Latn", code: "is" },
+ { label: "Igbo", value: "ibo_Latn", code: "ig" },
+ { label: "Indonesian", value: "ind_Latn", code: "id" },
+ { label: "Irish", value: "gle_Latn", code: "ga" },
+ { label: "Italian", value: "ita_Latn", code: "it" },
+ { label: "Japanese", value: "jpn_Jpan", code: "ja" },
+ { label: "Javanese", value: "jav_Latn", code: "jv" },
+ { label: "Kannada", value: "kan_Knda", code: "kn" },
+ { label: "Kashmiri (Arabic script)", value: "kas_Arab", code: "ks" },
+ { label: "Kashmiri (Devanagari script)", value: "kas_Deva", code: "ks" },
+ { label: "Kazakh", value: "kaz_Cyrl", code: "kk" },
+ { label: "Khmer", value: "khm_Khmr", code: "km" },
+ { label: "Kikongo", value: "kon_Latn", code: "kg" },
+ { label: "Kikuyu", value: "kik_Latn", code: "ki" },
+ { label: "Kinyarwanda", value: "kin_Latn", code: "rw" },
+ { label: "Korean", value: "kor_Hang", code: "ko" },
+ { label: "Kyrgyz", value: "kir_Cyrl", code: "ky" },
+ { label: "Lao", value: "lao_Laoo", code: "lo" },
+ { label: "Latgalian", value: "ltg_Latn", code: "ltg" },
+ { label: "Ligurian", value: "lij_Latn", code: "lij" },
+ { label: "Limburgish", value: "lim_Latn", code: "li" },
+ { label: "Lingala", value: "lin_Latn", code: "ln" },
+ { label: "Lithuanian", value: "lit_Latn", code: "lt" },
+ { label: "Luo", value: "luo_Latn", code: "luo" },
+ { label: "Luxembourgish", value: "ltz_Latn", code: "lb" },
+ { label: "Macedonian", value: "mkd_Cyrl", code: "mk" },
+ { label: "Magahi", value: "mag_Deva", code: "mag" },
+ { label: "Maithili", value: "mai_Deva", code: "mai" },
+ { label: "Malayalam", value: "mal_Mlym", code: "ml" },
+ { label: "Maltese", value: "mlt_Latn", code: "mt" },
+ { label: "Maori", value: "mri_Latn", code: "mi" },
+ { label: "Marathi", value: "mar_Deva", code: "mr" },
+ { label: "Mizo", value: "lus_Latn", code: "lus" },
+ { label: "Modern Standard Arabic (Romanized)", value: "arb_Latn", code: "ar" },
+ { label: "Modern Standard Arabic", value: "arb_Arab", code: "ar" },
+ { label: "Moroccan Arabic", value: "ary_Arab", code: "ary" },
+ { label: "Nepali", value: "npi_Deva", code: "ne" },
+ { label: "North Azerbaijani", value: "azj_Latn", code: "az" },
+ { label: "Northern Sotho", value: "nso_Latn", code: "nso" },
+ { label: "Northern Uzbek", value: "uzn_Latn", code: "uz" },
+ { label: "Norwegian Bokmål", value: "nob_Latn", code: "nb" },
+ { label: "Norwegian Nynorsk", value: "nno_Latn", code: "nn" },
+ { label: "Nuer", value: "nus_Latn", code: "nus" },
+ { label: "Nyanja", value: "nya_Latn", code: "ny" },
+ { label: "Occitan", value: "oci_Latn", code: "oc" },
+ { label: "Odia", value: "ory_Orya", code: "or" },
+ { label: "Plateau Malagasy", value: "plt_Latn", code: "mg" },
+ { label: "Polish", value: "pol_Latn", code: "pl" },
+ { label: "Portuguese", value: "por_Latn", code: "pt" },
+ { label: "Romanian", value: "ron_Latn", code: "ro" },
+ { label: "Rundi", value: "run_Latn", code: "rn" },
+ { label: "Russian", value: "rus_Cyrl", code: "ru" },
+ { label: "Samoan", value: "smo_Latn", code: "sm" },
+ { label: "Sango", value: "sag_Latn", code: "sg" },
+ { label: "Sanskrit", value: "san_Deva", code: "sa" },
+ { label: "Sardinian", value: "srd_Latn", code: "sc" },
+ { label: "Scottish Gaelic", value: "gla_Latn", code: "gd" },
+ { label: "Serbian", value: "srp_Cyrl", code: "sr" },
+ { label: "Shan", value: "shn_Mymr", code: "shn" },
+ { label: "Shona", value: "sna_Latn", code: "sn" },
+ { label: "Sindhi", value: "snd_Arab", code: "sd" },
+ { label: "Sinhala", value: "sin_Sinh", code: "si" },
+ { label: "Slovak", value: "slk_Latn", code: "sk" },
+ { label: "Slovenian", value: "slv_Latn", code: "sl" },
+ { label: "Somali", value: "som_Latn", code: "so" },
+ { label: "Southern Pashto", value: "pbt_Arab", code: "ps" },
+ { label: "Southern Sotho", value: "sot_Latn", code: "st" },
+ { label: "Spanish", value: "spa_Latn", code: "es" },
+ { label: "Standard Latvian", value: "lvs_Latn", code: "lv" },
+ { label: "Standard Malay", value: "zsm_Latn", code: "ms" },
+ { label: "Standard Tibetan", value: "bod_Tibt", code: "bo" },
+ { label: "Sundanese", value: "sun_Latn", code: "su" },
+ { label: "Swahili", value: "swh_Latn", code: "sw" },
+ { label: "Swati", value: "ssw_Latn", code: "ss" },
+ { label: "Swedish", value: "swe_Latn", code: "sv" },
+ { label: "Tagalog", value: "tgl_Latn", code: "tl" },
+ { label: "Tajik", value: "tgk_Cyrl", code: "tg" },
+ { label: "Tamil", value: "tam_Taml", code: "ta" },
+ { label: "Tatar", value: "tat_Cyrl", code: "tt" },
+ { label: "Telugu", value: "tel_Telu", code: "te" },
+ { label: "Thai", value: "tha_Thai", code: "th" },
+ { label: "Tigrinya", value: "tir_Ethi", code: "ti" },
+ { label: "Tok Pisin", value: "tpi_Latn", code: "tpi" },
+ { label: "Tosk Albanian", value: "als_Latn", code: "sq" },
+ { label: "Tsonga", value: "tso_Latn", code: "ts" },
+ { label: "Tswana", value: "tsn_Latn", code: "tn" },
+ { label: "Tumbuka", value: "tum_Latn", code: "tum" },
+ { label: "Turkish", value: "tur_Latn", code: "tr" },
+ { label: "Turkmen", value: "tuk_Latn", code: "tk" },
+ { label: "Twi", value: "twi_Latn", code: "tw" },
+ { label: "Ukrainian", value: "ukr_Cyrl", code: "uk" },
+ { label: "Urdu", value: "urd_Arab", code: "ur" },
+ { label: "Uyghur", value: "uig_Arab", code: "ug" },
+ { label: "Vietnamese", value: "vie_Latn", code: "vi" },
+ { label: "Welsh", value: "cym_Latn", code: "cy" },
+ { label: "West Central Oromo", value: "gaz_Latn", code: "om" },
+ { label: "Western Persian", value: "pes_Arab", code: "fa" },
+ { label: "Wolof", value: "wol_Latn", code: "wo" },
+ { label: "Xhosa", value: "xho_Latn", code: "xh" },
+ { label: "Yoruba", value: "yor_Latn", code: "yo" },
+ { label: "Yue Chinese", value: "yue_Hant", code: "yue" },
+ { label: "Zulu", value: "zul_Latn", code: "zu" },
+]
diff --git a/plugins/magic-alt/src/globals.css b/plugins/magic-alt/src/globals.css
new file mode 100644
index 00000000..cf0a7486
--- /dev/null
+++ b/plugins/magic-alt/src/globals.css
@@ -0,0 +1,81 @@
+@tailwind base;
+@tailwind components;
+@tailwind utilities;
+
+@layer utilities {
+ /* Chrome, Safari and Opera */
+ .no-scrollbar::-webkit-scrollbar {
+ display: none;
+ }
+
+ .no-scrollbar {
+ -ms-overflow-style: none; /* IE and Edge */
+ scrollbar-width: none; /* Firefox */
+ }
+}
+
+@layer components {
+ .row {
+ display: flex;
+ flex-direction: row;
+ gap: 10px;
+ }
+
+ .col {
+ display: flex;
+ flex-direction: column;
+ gap: 10px;
+ }
+
+ .row-lg {
+ display: flex;
+ flex-direction: row;
+ gap: 15px;
+ }
+
+ .col-lg {
+ display: flex;
+ flex-direction: column;
+ gap: 15px;
+ }
+
+ .combo-bg {
+ @apply bg-white dark:bg-tertiary;
+ }
+
+ .menu-shadow {
+ box-shadow: 0 10px 20px 0 rgba(0, 0, 0, 0.1);
+ }
+
+ .scroll-fade {
+ flex-shrink: 0;
+ height: 45px;
+ width: 100%;
+ display: block;
+ overflow: visible;
+ z-index: 10;
+ position: absolute;
+ left: 0;
+ bottom: 61px;
+ pointer-events: none;
+ }
+
+ [data-framer-theme="light"] .scroll-fade {
+ background: linear-gradient(to bottom, transparent, white);
+ }
+
+ [data-framer-theme="dark"] .scroll-fade {
+ background: linear-gradient(180deg, rgba(18, 18, 18, 0) 0%, rgb(17, 17, 17) 97.8%);
+ }
+
+ .loading-transition {
+ transition:
+ width 150ms ease-in-out,
+ opacity 150ms ease-in-out 150ms;
+ }
+}
+
+#root {
+ display: flex;
+ flex-direction: column;
+}
diff --git a/plugins/magic-alt/src/hooks/useNoAltImages.tsx b/plugins/magic-alt/src/hooks/useNoAltImages.tsx
new file mode 100644
index 00000000..5283a0d4
--- /dev/null
+++ b/plugins/magic-alt/src/hooks/useNoAltImages.tsx
@@ -0,0 +1,32 @@
+import { FrameNode, framer } from "framer-plugin"
+import { useEffect, useState } from "react"
+
+export type FrameNodeWithImage = FrameNode & {
+ backgroundImage: NonNullable
+}
+
+export function useNoAltImages() {
+ const [images, setImages] = useState([])
+
+ useEffect(() => {
+ let active = true
+
+ async function run() {
+ const imageNodes = await framer.getNodesWithAttributeSet("backgroundImage")
+ const imagesWithoutAlt = imageNodes.filter(
+ node => node.backgroundImage != null && !node.backgroundImage.altText
+ )
+
+ if (!active) return
+ setImages(imagesWithoutAlt as FrameNodeWithImage[])
+ }
+
+ run()
+
+ return () => {
+ active = false
+ }
+ }, [])
+
+ return [images, setImages] as const
+}
diff --git a/plugins/magic-alt/src/hooks/useTranslation.ts b/plugins/magic-alt/src/hooks/useTranslation.ts
new file mode 100644
index 00000000..016bace1
--- /dev/null
+++ b/plugins/magic-alt/src/hooks/useTranslation.ts
@@ -0,0 +1,161 @@
+import { useEffect, useMemo, useRef, useState } from "react"
+import { framer } from "framer-plugin"
+import { Outputs, ProgressData, WorkerRequest } from "../translation.worker"
+
+interface InitPipelineEvent {
+ status: "initiate"
+ file: string
+}
+
+interface InitPipelineProgressEvent extends ProgressData {
+ status: "progress"
+}
+
+interface InitPipelineDoneEvent extends ProgressData {
+ status: "done"
+}
+
+interface PipelineReadyEvent {
+ status: "ready"
+}
+
+interface TranslateUpdateEvent {
+ status: "update"
+ outputs: Outputs
+}
+
+interface TranslateCompleteEvent {
+ status: "complete"
+ outputs: Outputs
+}
+
+interface ErrorEvent {
+ status: "error"
+ message: string
+}
+
+type WorkerResponse =
+ | InitPipelineEvent
+ | InitPipelineProgressEvent
+ | InitPipelineDoneEvent
+ | PipelineReadyEvent
+ | TranslateUpdateEvent
+ | TranslateCompleteEvent
+ | ErrorEvent
+
+export const useTranslation = () => {
+ const worker = useRef(null)
+
+ const [outputs, setOutputs] = useState([])
+ const [isModelReady, setIsModelReady] = useState(false)
+ const [isModelLoading, setIsModelLoading] = useState(false)
+ const [isTranslating, setIsTranslating] = useState(false)
+ const [progressItems, setProgressItems] = useState([])
+
+ const translationResolve = useRef<((outputs: Outputs) => void) | null>(null)
+ const translationReject = useRef<((reason: Error) => void) | null>(null)
+
+ const postMessage = (request: WorkerRequest) => {
+ return worker.current?.postMessage(request)
+ }
+
+ const modelLoadProgress = useMemo(() => {
+ const hasDeterminedFileTotals = progressItems.every(item => item.total !== 0)
+ if (progressItems.length === 0 || !hasDeterminedFileTotals) return 0
+
+ const totalBytes = progressItems.reduce((sum, item) => sum + item.total, 0)
+ const loadedBytes = progressItems.reduce((sum, item) => sum + item.loaded, 0)
+
+ if (totalBytes === 0) return 0
+
+ return Math.round((loadedBytes / totalBytes) * 100)
+ }, [progressItems])
+
+ useEffect(() => {
+ if (!worker.current) {
+ worker.current = new Worker(new URL("../translation.worker.ts", import.meta.url), {
+ type: "module",
+ })
+ }
+
+ const onMessageReceived = (e: MessageEvent) => {
+ const eventData = e.data
+
+ switch (eventData.status) {
+ case "initiate":
+ // Model file start load: add to progress items for tracking
+ setProgressItems(prev => [...prev, { ...eventData, total: 0, loaded: 0, progress: 0 }])
+ setIsModelLoading(true)
+ break
+
+ case "progress":
+ // Model file progress: update progress items
+ setProgressItems(items => items.map(item => (item.file === eventData.file ? eventData : item)))
+ break
+
+ case "update":
+ // Generation update: update the real-time outputs
+ setOutputs(eventData.outputs)
+ break
+
+ case "ready":
+ // Pipeline ready: worker ready to accept messages
+ setIsModelLoading(false)
+ setIsModelReady(true)
+ break
+
+ case "complete":
+ // Translation complete
+ setIsTranslating(false)
+
+ if (translationResolve.current) {
+ translationResolve.current(eventData.outputs)
+ translationResolve.current = null
+ translationReject.current = null
+ }
+
+ break
+
+ case "error":
+ framer.notify(eventData.message, { variant: "error" })
+
+ if (translationReject.current) {
+ translationReject.current(new Error(eventData.message))
+ translationResolve.current = null
+ translationReject.current = null
+ }
+
+ break
+ }
+ }
+
+ worker.current?.addEventListener("message", onMessageReceived)
+
+ return () => worker.current?.removeEventListener("message", onMessageReceived)
+ }, [])
+
+ const loadModel = () => {
+ postMessage({ action: "load_model" })
+ }
+
+ const translate = async (texts: string[], src_lang: string, tgt_lang: string): Promise => {
+ setIsTranslating(true)
+
+ return new Promise((resolve: (outputs: Outputs) => void, reject) => {
+ translationResolve.current = resolve
+ translationReject.current = reject
+
+ postMessage({ action: "translate", texts, src_lang, tgt_lang })
+ }).finally(() => setIsTranslating(false))
+ }
+
+ return {
+ loadModel,
+ modelLoadProgress,
+ translate,
+ isTranslating,
+ isModelLoading,
+ isModelReady,
+ outputs,
+ }
+}
diff --git a/plugins/magic-alt/src/main.tsx b/plugins/magic-alt/src/main.tsx
new file mode 100644
index 00000000..4f141d2f
--- /dev/null
+++ b/plugins/magic-alt/src/main.tsx
@@ -0,0 +1,68 @@
+import "./globals.css"
+import "framer-plugin/framer.css"
+
+import React from "react"
+import ReactDOM from "react-dom/client"
+import { App } from "./App"
+import { framer } from "framer-plugin"
+import { generateCaptions } from "./api"
+
+function renderPlugin() {
+ const root = document.getElementById("root")
+ if (!root) throw new Error("Root element not found")
+
+ framer.showUI({
+ position: "top right",
+ minWidth: 240,
+ minHeight: 300,
+ resizable: true,
+ })
+
+ ReactDOM.createRoot(root).render(
+
+
+
+ )
+}
+
+async function runPlugin() {
+ const mode = framer.mode
+
+ if (mode === "editImage") {
+ try {
+ const image = await framer.getImage()
+
+ if (!image) {
+ await framer.closePlugin("No Image was selected.", { variant: "error" })
+ return
+ }
+
+ const siteInfo = await framer.getPublishInfo()
+ const siteUrl = siteInfo.staging?.url
+
+ if (!siteUrl) {
+ await framer.closePlugin("Please publish your site to staging", { variant: "error" })
+ return
+ }
+
+ const captions = await generateCaptions(siteUrl, [image.url])
+
+ await framer.setImage({
+ image: {
+ bytes: (await image.getData()).bytes,
+ mimeType: (await image.getData()).mimeType,
+ },
+ altText: captions.data[0].caption,
+ })
+
+ await framer.closePlugin("Alt text generated", { variant: "success" })
+ } catch (e) {
+ const message = e instanceof Error ? e.message : String(e)
+ framer.closePlugin("An unexpected error occurred: " + message, { variant: "error" })
+ }
+ }
+
+ renderPlugin()
+}
+
+runPlugin()
diff --git a/plugins/magic-alt/src/translation.worker.ts b/plugins/magic-alt/src/translation.worker.ts
new file mode 100644
index 00000000..ad9bee05
--- /dev/null
+++ b/plugins/magic-alt/src/translation.worker.ts
@@ -0,0 +1,113 @@
+import { env, pipeline, TranslationOutput, TranslationPipeline } from "@xenova/transformers"
+import { GenerationConfigType } from "@xenova/transformers/types/utils/generation"
+
+// See: https://github.com/xenova/transformers.js/issues/366
+env.allowLocalModels = false
+env.allowRemoteModels = true
+env.useBrowserCache = false
+
+// See: https://github.com/xenova/transformers.js/issues/270
+const originalWarn = console.warn
+
+console.warn = function (...args) {
+ if (!args[0].includes("onnxruntime")) {
+ originalWarn.apply(console, args)
+ }
+}
+
+export interface TranslationRequest {
+ action: "translate"
+ texts: string[]
+ src_lang: string
+ tgt_lang: string
+}
+
+export interface LoadModelRequest {
+ action: "load_model"
+}
+
+export type WorkerRequest = TranslationRequest | LoadModelRequest
+
+export interface ProgressData {
+ file: string
+ progress: number
+ loaded: number
+ total: number
+}
+
+export type Outputs = string[]
+
+interface ModelCallbackOutput {
+ output_token_ids: number[]
+}
+
+class TranslationSingleton {
+ static instance: Promise | null = null
+
+ static async getInstance(progress_callback: (data: ProgressData) => void) {
+ if (this.instance === null) {
+ this.instance = pipeline("translation", "Xenova/nllb-200-distilled-600M", {
+ progress_callback,
+ quantized: true,
+ })
+ }
+
+ return this.instance
+ }
+}
+
+let translator: TranslationPipeline | null = null
+
+self.addEventListener("message", async (event: MessageEvent) => {
+ const eventData = event.data
+
+ switch (eventData.action) {
+ case "load_model": {
+ translator = await TranslationSingleton.getInstance(x => {
+ // Track model loading progress
+ self.postMessage(x)
+ })
+
+ break
+ }
+
+ case "translate": {
+ if (!translator) {
+ self.postMessage({ status: "error", message: "Please load the model before translating" })
+ return
+ }
+
+ const { texts, src_lang, tgt_lang } = eventData
+
+ try {
+ const outputs = (await translator(texts, {
+ tgt_lang,
+ src_lang,
+ // Allow for partial output
+ callback_function: (x: ModelCallbackOutput[]) => {
+ const decodedOutputs = x.map(output =>
+ translator!.tokenizer.decode(output.output_token_ids, { skip_special_tokens: true })
+ )
+
+ self.postMessage({
+ status: "update",
+ outputs: decodedOutputs,
+ })
+ },
+ } as GenerationConfigType)) as TranslationOutput
+
+ self.postMessage({
+ status: "complete",
+ outputs: outputs.map(output => output.translation_text),
+ })
+ } catch (e) {
+ self.postMessage({ status: "error", message: e instanceof Error ? e.message : String(e) })
+ }
+
+ break
+ }
+
+ default:
+ self.postMessage({ status: "error", message: "Unknown action" })
+ }
+})
diff --git a/plugins/magic-alt/src/utils.ts b/plugins/magic-alt/src/utils.ts
new file mode 100644
index 00000000..3812e6ff
--- /dev/null
+++ b/plugins/magic-alt/src/utils.ts
@@ -0,0 +1,4 @@
+export const removeItemAtIndex = (array: T[] | null, index: number): T[] => {
+ if (!array) return []
+ return [...array.slice(0, index), ...array.slice(index + 1)]
+}
diff --git a/plugins/magic-alt/src/vite-env.d.ts b/plugins/magic-alt/src/vite-env.d.ts
new file mode 100644
index 00000000..11f02fe2
--- /dev/null
+++ b/plugins/magic-alt/src/vite-env.d.ts
@@ -0,0 +1 @@
+///
diff --git a/plugins/magic-alt/tailwind.config.js b/plugins/magic-alt/tailwind.config.js
new file mode 100644
index 00000000..d225cb35
--- /dev/null
+++ b/plugins/magic-alt/tailwind.config.js
@@ -0,0 +1,61 @@
+/** @type {import('tailwindcss').Config} */
+export default {
+ darkMode: ["class", '[data-framer-theme="dark"]'],
+ content: ["./index.html", "./src/**/*.{js,ts,jsx,tsx}"],
+ theme: {
+ extend: {
+ backgroundColor: {
+ primary: "var(--framer-color-bg)",
+ secondary: "var(--framer-color-bg-secondary)",
+ tertiary: "var(--framer-color-bg-tertiary)",
+ tertiaryDimmedLight: "rgba(243, 243, 243, 0.75)",
+ tertiaryDimmedDark: "rgba(43, 43, 43, 0.75)",
+ divider: "var(--framer-color-divider)",
+ tint: "var(--framer-color-tint)",
+ tintDimmed: "var(--framer-color-tint-dimmed)",
+ tintDark: "var(--framer-color-tint-dark)",
+ blackDimmed: "rgba(0, 0, 0, 0.5)",
+ },
+ colors: {
+ primary: "var(--framer-color-text)",
+ secondary: "var(--framer-color-text-secondary)",
+ tertiary: "var(--framer-color-text-tertiary)",
+ inverted: "var(--framer-color-text-inverted)",
+ "framer-red": "#FF3366",
+ "framer-yellow": "#FFC25A",
+ "framer-blue": "#0099FF",
+ "framer-green": "#00c25a",
+ },
+ borderColor: {
+ divider: "var(--framer-color-divider)",
+ },
+ fontSize: {
+ "2xs": "10px",
+ },
+ spacing: {
+ 30: "30px",
+ 15: "15px",
+ },
+ boxShadow: {
+ "bg-image": "inset 0 0 0 1px rgba(0, 0, 0, 0.1)",
+ },
+ dropShadow: {
+ "locale-code": "0 1px 1px rgba(0, 0, 0, 0.2)",
+ },
+ keyframes: {
+ pulse: {
+ "0%, 100%": { opacity: "1" },
+ "50%": { opacity: "0.5" },
+ },
+ expand: {
+ from: { width: "0" },
+ to: { width: "100%" },
+ },
+ },
+ animation: {
+ pulse: "pulse 2s cubic-bezier(0.4, 0, 0.6, 1) infinite",
+ expand: "expand 0.5s ease-out forwards",
+ },
+ },
+ },
+}
diff --git a/plugins/magic-alt/tsconfig.json b/plugins/magic-alt/tsconfig.json
new file mode 100644
index 00000000..325606c6
--- /dev/null
+++ b/plugins/magic-alt/tsconfig.json
@@ -0,0 +1,23 @@
+{
+ "compilerOptions": {
+ "target": "ES2022",
+ "useDefineForClassFields": true,
+ "lib": ["ES2022", "DOM", "DOM.Iterable"],
+ "module": "ES2022",
+
+ /* Bundler mode */
+ "moduleResolution": "bundler",
+ "allowImportingTsExtensions": true,
+ "resolveJsonModule": true,
+ "isolatedModules": true,
+ "noEmit": true,
+ "jsx": "react-jsx",
+
+ /* Linting */
+ "strict": true,
+ "noUnusedLocals": true,
+ "noUnusedParameters": true,
+ "noFallthroughCasesInSwitch": true
+ },
+ "include": ["src"]
+}
diff --git a/plugins/magic-alt/vite.config.ts b/plugins/magic-alt/vite.config.ts
new file mode 100644
index 00000000..7efb6ef8
--- /dev/null
+++ b/plugins/magic-alt/vite.config.ts
@@ -0,0 +1,12 @@
+import { defineConfig } from "vite"
+import react from "@vitejs/plugin-react-swc"
+import mkcert from "vite-plugin-mkcert"
+import framer from "vite-plugin-framer"
+
+// https://vitejs.dev/config/
+export default defineConfig({
+ plugins: [react(), mkcert(), framer()],
+ build: {
+ target: "ES2022",
+ },
+})