@efxlab/motion-canvas-2d 4.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (455) hide show
  1. package/editor/index.css +40 -0
  2. package/editor/index.js +525 -0
  3. package/editor/index.js.map +1 -0
  4. package/lib/code/CodeCursor.d.ts +84 -0
  5. package/lib/code/CodeCursor.d.ts.map +1 -0
  6. package/lib/code/CodeCursor.js +315 -0
  7. package/lib/code/CodeDiffer.d.ts +28 -0
  8. package/lib/code/CodeDiffer.d.ts.map +1 -0
  9. package/lib/code/CodeDiffer.js +51 -0
  10. package/lib/code/CodeFragment.d.ts +42 -0
  11. package/lib/code/CodeFragment.d.ts.map +1 -0
  12. package/lib/code/CodeFragment.js +72 -0
  13. package/lib/code/CodeHighlighter.d.ts +69 -0
  14. package/lib/code/CodeHighlighter.d.ts.map +1 -0
  15. package/lib/code/CodeHighlighter.js +2 -0
  16. package/lib/code/CodeMetrics.d.ts +11 -0
  17. package/lib/code/CodeMetrics.d.ts.map +1 -0
  18. package/lib/code/CodeMetrics.js +29 -0
  19. package/lib/code/CodeRange.d.ts +43 -0
  20. package/lib/code/CodeRange.d.ts.map +1 -0
  21. package/lib/code/CodeRange.js +185 -0
  22. package/lib/code/CodeScope.d.ts +16 -0
  23. package/lib/code/CodeScope.d.ts.map +1 -0
  24. package/lib/code/CodeScope.js +72 -0
  25. package/lib/code/CodeSelection.d.ts +6 -0
  26. package/lib/code/CodeSelection.d.ts.map +1 -0
  27. package/lib/code/CodeSelection.js +13 -0
  28. package/lib/code/CodeSignal.d.ts +63 -0
  29. package/lib/code/CodeSignal.d.ts.map +1 -0
  30. package/lib/code/CodeSignal.js +206 -0
  31. package/lib/code/CodeTokenizer.d.ts +8 -0
  32. package/lib/code/CodeTokenizer.d.ts.map +1 -0
  33. package/lib/code/CodeTokenizer.js +50 -0
  34. package/lib/code/DefaultHighlightStyle.d.ts +3 -0
  35. package/lib/code/DefaultHighlightStyle.d.ts.map +1 -0
  36. package/lib/code/DefaultHighlightStyle.js +98 -0
  37. package/lib/code/LezerHighlighter.d.ts +22 -0
  38. package/lib/code/LezerHighlighter.d.ts.map +1 -0
  39. package/lib/code/LezerHighlighter.js +91 -0
  40. package/lib/code/diff.d.ts +31 -0
  41. package/lib/code/diff.d.ts.map +1 -0
  42. package/lib/code/diff.js +236 -0
  43. package/lib/code/extractRange.d.ts +17 -0
  44. package/lib/code/extractRange.d.ts.map +1 -0
  45. package/lib/code/extractRange.js +102 -0
  46. package/lib/code/index.d.ts +14 -0
  47. package/lib/code/index.d.ts.map +1 -0
  48. package/lib/code/index.js +14 -0
  49. package/lib/components/Bezier.d.ts +22 -0
  50. package/lib/components/Bezier.d.ts.map +1 -0
  51. package/lib/components/Bezier.js +80 -0
  52. package/lib/components/Camera.d.ts +172 -0
  53. package/lib/components/Camera.d.ts.map +1 -0
  54. package/lib/components/Camera.js +239 -0
  55. package/lib/components/Circle.d.ts +191 -0
  56. package/lib/components/Circle.d.ts.map +1 -0
  57. package/lib/components/Circle.js +178 -0
  58. package/lib/components/Code.d.ts +238 -0
  59. package/lib/components/Code.d.ts.map +1 -0
  60. package/lib/components/Code.js +331 -0
  61. package/lib/components/CodeBlock.d.ts +131 -0
  62. package/lib/components/CodeBlock.d.ts.map +1 -0
  63. package/lib/components/CodeBlock.js +462 -0
  64. package/lib/components/CubicBezier.d.ts +69 -0
  65. package/lib/components/CubicBezier.d.ts.map +1 -0
  66. package/lib/components/CubicBezier.js +81 -0
  67. package/lib/components/Curve.d.ts +202 -0
  68. package/lib/components/Curve.d.ts.map +1 -0
  69. package/lib/components/Curve.js +284 -0
  70. package/lib/components/Grid.d.ts +75 -0
  71. package/lib/components/Grid.d.ts.map +1 -0
  72. package/lib/components/Grid.js +91 -0
  73. package/lib/components/Icon.d.ts +58 -0
  74. package/lib/components/Icon.d.ts.map +1 -0
  75. package/lib/components/Icon.js +58 -0
  76. package/lib/components/Img.d.ts +116 -0
  77. package/lib/components/Img.d.ts.map +1 -0
  78. package/lib/components/Img.js +233 -0
  79. package/lib/components/Knot.d.ts +89 -0
  80. package/lib/components/Knot.d.ts.map +1 -0
  81. package/lib/components/Knot.js +68 -0
  82. package/lib/components/Latex.d.ts +49 -0
  83. package/lib/components/Latex.d.ts.map +1 -0
  84. package/lib/components/Latex.js +206 -0
  85. package/lib/components/Layout.d.ts +423 -0
  86. package/lib/components/Layout.d.ts.map +1 -0
  87. package/lib/components/Layout.js +699 -0
  88. package/lib/components/Line.d.ts +158 -0
  89. package/lib/components/Line.d.ts.map +1 -0
  90. package/lib/components/Line.js +315 -0
  91. package/lib/components/Node.d.ts +843 -0
  92. package/lib/components/Node.d.ts.map +1 -0
  93. package/lib/components/Node.js +1335 -0
  94. package/lib/components/Path.d.ts +18 -0
  95. package/lib/components/Path.d.ts.map +1 -0
  96. package/lib/components/Path.js +96 -0
  97. package/lib/components/Polygon.d.ts +157 -0
  98. package/lib/components/Polygon.d.ts.map +1 -0
  99. package/lib/components/Polygon.js +183 -0
  100. package/lib/components/QuadBezier.d.ts +61 -0
  101. package/lib/components/QuadBezier.d.ts.map +1 -0
  102. package/lib/components/QuadBezier.js +76 -0
  103. package/lib/components/Ray.d.ts +60 -0
  104. package/lib/components/Ray.d.ts.map +1 -0
  105. package/lib/components/Ray.js +95 -0
  106. package/lib/components/Rect.d.ts +112 -0
  107. package/lib/components/Rect.d.ts.map +1 -0
  108. package/lib/components/Rect.js +76 -0
  109. package/lib/components/SVG.d.ts +175 -0
  110. package/lib/components/SVG.d.ts.map +1 -0
  111. package/lib/components/SVG.js +582 -0
  112. package/lib/components/Shape.d.ts +39 -0
  113. package/lib/components/Shape.d.ts.map +1 -0
  114. package/lib/components/Shape.js +134 -0
  115. package/lib/components/Spline.d.ts +87 -0
  116. package/lib/components/Spline.d.ts.map +1 -0
  117. package/lib/components/Spline.js +230 -0
  118. package/lib/components/Txt.d.ts +51 -0
  119. package/lib/components/Txt.d.ts.map +1 -0
  120. package/lib/components/Txt.js +172 -0
  121. package/lib/components/TxtLeaf.d.ts +20 -0
  122. package/lib/components/TxtLeaf.d.ts.map +1 -0
  123. package/lib/components/TxtLeaf.js +185 -0
  124. package/lib/components/Video.d.ts +110 -0
  125. package/lib/components/Video.d.ts.map +1 -0
  126. package/lib/components/Video.js +267 -0
  127. package/lib/components/View2D.d.ts +25 -0
  128. package/lib/components/View2D.d.ts.map +1 -0
  129. package/lib/components/View2D.js +85 -0
  130. package/lib/components/index.d.ts +27 -0
  131. package/lib/components/index.d.ts.map +1 -0
  132. package/lib/components/index.js +27 -0
  133. package/lib/components/types.d.ts +17 -0
  134. package/lib/components/types.d.ts.map +1 -0
  135. package/lib/components/types.js +2 -0
  136. package/lib/curves/ArcSegment.d.ts +26 -0
  137. package/lib/curves/ArcSegment.d.ts.map +1 -0
  138. package/lib/curves/ArcSegment.js +116 -0
  139. package/lib/curves/CircleSegment.d.ts +18 -0
  140. package/lib/curves/CircleSegment.d.ts.map +1 -0
  141. package/lib/curves/CircleSegment.js +60 -0
  142. package/lib/curves/CubicBezierSegment.d.ts +18 -0
  143. package/lib/curves/CubicBezierSegment.d.ts.map +1 -0
  144. package/lib/curves/CubicBezierSegment.js +60 -0
  145. package/lib/curves/CurveDrawingInfo.d.ts +11 -0
  146. package/lib/curves/CurveDrawingInfo.d.ts.map +1 -0
  147. package/lib/curves/CurveDrawingInfo.js +2 -0
  148. package/lib/curves/CurvePoint.d.ts +15 -0
  149. package/lib/curves/CurvePoint.d.ts.map +1 -0
  150. package/lib/curves/CurvePoint.js +2 -0
  151. package/lib/curves/CurveProfile.d.ts +7 -0
  152. package/lib/curves/CurveProfile.d.ts.map +1 -0
  153. package/lib/curves/CurveProfile.js +2 -0
  154. package/lib/curves/KnotInfo.d.ts +12 -0
  155. package/lib/curves/KnotInfo.d.ts.map +1 -0
  156. package/lib/curves/KnotInfo.js +2 -0
  157. package/lib/curves/LineSegment.d.ts +16 -0
  158. package/lib/curves/LineSegment.d.ts.map +1 -0
  159. package/lib/curves/LineSegment.js +51 -0
  160. package/lib/curves/Polynomial.d.ts +118 -0
  161. package/lib/curves/Polynomial.d.ts.map +1 -0
  162. package/lib/curves/Polynomial.js +263 -0
  163. package/lib/curves/Polynomial2D.d.ts +22 -0
  164. package/lib/curves/Polynomial2D.d.ts.map +1 -0
  165. package/lib/curves/Polynomial2D.js +51 -0
  166. package/lib/curves/PolynomialSegment.d.ts +39 -0
  167. package/lib/curves/PolynomialSegment.d.ts.map +1 -0
  168. package/lib/curves/PolynomialSegment.js +88 -0
  169. package/lib/curves/QuadBezierSegment.d.ts +17 -0
  170. package/lib/curves/QuadBezierSegment.d.ts.map +1 -0
  171. package/lib/curves/QuadBezierSegment.js +53 -0
  172. package/lib/curves/Segment.d.ts +9 -0
  173. package/lib/curves/Segment.d.ts.map +1 -0
  174. package/lib/curves/Segment.js +3 -0
  175. package/lib/curves/UniformPolynomialCurveSampler.d.ts +43 -0
  176. package/lib/curves/UniformPolynomialCurveSampler.d.ts.map +1 -0
  177. package/lib/curves/UniformPolynomialCurveSampler.js +74 -0
  178. package/lib/curves/createCurveProfileLerp.d.ts +32 -0
  179. package/lib/curves/createCurveProfileLerp.d.ts.map +1 -0
  180. package/lib/curves/createCurveProfileLerp.js +351 -0
  181. package/lib/curves/getBezierSplineProfile.d.ts +12 -0
  182. package/lib/curves/getBezierSplineProfile.d.ts.map +1 -0
  183. package/lib/curves/getBezierSplineProfile.js +140 -0
  184. package/lib/curves/getCircleProfile.d.ts +4 -0
  185. package/lib/curves/getCircleProfile.d.ts.map +1 -0
  186. package/lib/curves/getCircleProfile.js +44 -0
  187. package/lib/curves/getPathProfile.d.ts +3 -0
  188. package/lib/curves/getPathProfile.d.ts.map +1 -0
  189. package/lib/curves/getPathProfile.js +128 -0
  190. package/lib/curves/getPointAtDistance.d.ts +4 -0
  191. package/lib/curves/getPointAtDistance.d.ts.map +1 -0
  192. package/lib/curves/getPointAtDistance.js +15 -0
  193. package/lib/curves/getPolylineProfile.d.ts +4 -0
  194. package/lib/curves/getPolylineProfile.d.ts.map +1 -0
  195. package/lib/curves/getPolylineProfile.js +58 -0
  196. package/lib/curves/getRectProfile.d.ts +4 -0
  197. package/lib/curves/getRectProfile.d.ts.map +1 -0
  198. package/lib/curves/getRectProfile.js +57 -0
  199. package/lib/curves/index.d.ts +17 -0
  200. package/lib/curves/index.d.ts.map +1 -0
  201. package/lib/curves/index.js +17 -0
  202. package/lib/decorators/canvasStyleSignal.d.ts +5 -0
  203. package/lib/decorators/canvasStyleSignal.d.ts.map +1 -0
  204. package/lib/decorators/canvasStyleSignal.js +12 -0
  205. package/lib/decorators/colorSignal.d.ts +2 -0
  206. package/lib/decorators/colorSignal.d.ts.map +1 -0
  207. package/lib/decorators/colorSignal.js +9 -0
  208. package/lib/decorators/compound.d.ts +27 -0
  209. package/lib/decorators/compound.d.ts.map +1 -0
  210. package/lib/decorators/compound.js +49 -0
  211. package/lib/decorators/computed.d.ts +9 -0
  212. package/lib/decorators/computed.d.ts.map +1 -0
  213. package/lib/decorators/computed.js +18 -0
  214. package/lib/decorators/defaultStyle.d.ts +2 -0
  215. package/lib/decorators/defaultStyle.d.ts.map +1 -0
  216. package/lib/decorators/defaultStyle.js +13 -0
  217. package/lib/decorators/filtersSignal.d.ts +11 -0
  218. package/lib/decorators/filtersSignal.d.ts.map +1 -0
  219. package/lib/decorators/filtersSignal.js +73 -0
  220. package/lib/decorators/index.d.ts +11 -0
  221. package/lib/decorators/index.d.ts.map +1 -0
  222. package/lib/decorators/index.js +11 -0
  223. package/lib/decorators/initializers.d.ts +4 -0
  224. package/lib/decorators/initializers.d.ts.map +1 -0
  225. package/lib/decorators/initializers.js +27 -0
  226. package/lib/decorators/nodeName.d.ts +9 -0
  227. package/lib/decorators/nodeName.d.ts.map +1 -0
  228. package/lib/decorators/nodeName.js +13 -0
  229. package/lib/decorators/signal.d.ts +183 -0
  230. package/lib/decorators/signal.d.ts.map +1 -0
  231. package/lib/decorators/signal.js +285 -0
  232. package/lib/decorators/spacingSignal.d.ts +2 -0
  233. package/lib/decorators/spacingSignal.d.ts.map +1 -0
  234. package/lib/decorators/spacingSignal.js +15 -0
  235. package/lib/decorators/vector2Signal.d.ts +8 -0
  236. package/lib/decorators/vector2Signal.d.ts.map +1 -0
  237. package/lib/decorators/vector2Signal.js +15 -0
  238. package/lib/index.d.ts +9 -0
  239. package/lib/index.d.ts.map +1 -0
  240. package/lib/index.js +9 -0
  241. package/lib/jsx-dev-runtime.d.ts +3 -0
  242. package/lib/jsx-dev-runtime.d.ts.map +1 -0
  243. package/lib/jsx-dev-runtime.js +3 -0
  244. package/lib/jsx-runtime.d.ts +12 -0
  245. package/lib/jsx-runtime.d.ts.map +1 -0
  246. package/lib/jsx-runtime.js +23 -0
  247. package/lib/partials/Filter.d.ts +82 -0
  248. package/lib/partials/Filter.d.ts.map +1 -0
  249. package/lib/partials/Filter.js +137 -0
  250. package/lib/partials/Gradient.d.ts +31 -0
  251. package/lib/partials/Gradient.d.ts.map +1 -0
  252. package/lib/partials/Gradient.js +63 -0
  253. package/lib/partials/Pattern.d.ts +13 -0
  254. package/lib/partials/Pattern.d.ts.map +1 -0
  255. package/lib/partials/Pattern.js +27 -0
  256. package/lib/partials/ShaderConfig.d.ts +81 -0
  257. package/lib/partials/ShaderConfig.d.ts.map +1 -0
  258. package/lib/partials/ShaderConfig.js +25 -0
  259. package/lib/partials/index.d.ts +5 -0
  260. package/lib/partials/index.d.ts.map +1 -0
  261. package/lib/partials/index.js +5 -0
  262. package/lib/partials/types.d.ts +35 -0
  263. package/lib/partials/types.d.ts.map +1 -0
  264. package/lib/partials/types.js +2 -0
  265. package/lib/scenes/Scene2D.d.ts +23 -0
  266. package/lib/scenes/Scene2D.d.ts.map +1 -0
  267. package/lib/scenes/Scene2D.js +152 -0
  268. package/lib/scenes/index.d.ts +4 -0
  269. package/lib/scenes/index.d.ts.map +1 -0
  270. package/lib/scenes/index.js +4 -0
  271. package/lib/scenes/makeScene2D.d.ts +5 -0
  272. package/lib/scenes/makeScene2D.d.ts.map +1 -0
  273. package/lib/scenes/makeScene2D.js +12 -0
  274. package/lib/scenes/useScene2D.d.ts +3 -0
  275. package/lib/scenes/useScene2D.d.ts.map +1 -0
  276. package/lib/scenes/useScene2D.js +5 -0
  277. package/lib/tsconfig.build.tsbuildinfo +1 -0
  278. package/lib/tsconfig.tsbuildinfo +1 -0
  279. package/lib/utils/CanvasUtils.d.ts +21 -0
  280. package/lib/utils/CanvasUtils.d.ts.map +1 -0
  281. package/lib/utils/CanvasUtils.js +138 -0
  282. package/lib/utils/diff.d.ts +31 -0
  283. package/lib/utils/diff.d.ts.map +1 -0
  284. package/lib/utils/diff.js +97 -0
  285. package/lib/utils/index.d.ts +4 -0
  286. package/lib/utils/index.d.ts.map +1 -0
  287. package/lib/utils/index.js +4 -0
  288. package/lib/utils/is.d.ts +8 -0
  289. package/lib/utils/is.d.ts.map +1 -0
  290. package/lib/utils/is.js +10 -0
  291. package/lib/utils/makeSignalExtensions.d.ts +4 -0
  292. package/lib/utils/makeSignalExtensions.d.ts.map +1 -0
  293. package/lib/utils/makeSignalExtensions.js +20 -0
  294. package/lib/utils/withDefaults.d.ts +20 -0
  295. package/lib/utils/withDefaults.d.ts.map +1 -0
  296. package/lib/utils/withDefaults.js +23 -0
  297. package/package.json +54 -0
  298. package/src/editor/NodeInspectorConfig.tsx +76 -0
  299. package/src/editor/PreviewOverlayConfig.tsx +65 -0
  300. package/src/editor/Provider.tsx +109 -0
  301. package/src/editor/SceneGraphTabConfig.tsx +87 -0
  302. package/src/editor/icons/CircleIcon.tsx +7 -0
  303. package/src/editor/icons/CodeBlockIcon.tsx +8 -0
  304. package/src/editor/icons/CurveIcon.tsx +7 -0
  305. package/src/editor/icons/GridIcon.tsx +7 -0
  306. package/src/editor/icons/IconMap.ts +35 -0
  307. package/src/editor/icons/ImgIcon.tsx +8 -0
  308. package/src/editor/icons/LayoutIcon.tsx +9 -0
  309. package/src/editor/icons/LineIcon.tsx +7 -0
  310. package/src/editor/icons/NodeIcon.tsx +7 -0
  311. package/src/editor/icons/RayIcon.tsx +7 -0
  312. package/src/editor/icons/RectIcon.tsx +7 -0
  313. package/src/editor/icons/ShapeIcon.tsx +7 -0
  314. package/src/editor/icons/TxtIcon.tsx +8 -0
  315. package/src/editor/icons/VideoIcon.tsx +7 -0
  316. package/src/editor/icons/View2DIcon.tsx +10 -0
  317. package/src/editor/index.css +0 -0
  318. package/src/editor/index.ts +19 -0
  319. package/src/editor/shortcuts.ts +27 -0
  320. package/src/editor/tree/DetachedRoot.tsx +27 -0
  321. package/src/editor/tree/NodeElement.tsx +72 -0
  322. package/src/editor/tree/TreeElement.tsx +70 -0
  323. package/src/editor/tree/TreeRoot.tsx +10 -0
  324. package/src/editor/tree/ViewRoot.tsx +20 -0
  325. package/src/editor/tree/index.module.scss +45 -0
  326. package/src/editor/tree/index.ts +4 -0
  327. package/src/editor/tree/navigation.ts +145 -0
  328. package/src/editor/tsconfig.build.json +5 -0
  329. package/src/editor/tsconfig.json +12 -0
  330. package/src/editor/tsdoc.json +4 -0
  331. package/src/editor/utils/SignalSet.ts +37 -0
  332. package/src/editor/utils/index.ts +1 -0
  333. package/src/editor/vite-env.d.ts +1 -0
  334. package/src/lib/code/CodeCursor.ts +468 -0
  335. package/src/lib/code/CodeDiffer.ts +77 -0
  336. package/src/lib/code/CodeFragment.ts +96 -0
  337. package/src/lib/code/CodeHighlighter.ts +73 -0
  338. package/src/lib/code/CodeMetrics.ts +47 -0
  339. package/src/lib/code/CodeRange.test.ts +113 -0
  340. package/src/lib/code/CodeRange.ts +222 -0
  341. package/src/lib/code/CodeScope.ts +100 -0
  342. package/src/lib/code/CodeSelection.ts +28 -0
  343. package/src/lib/code/CodeSignal.ts +348 -0
  344. package/src/lib/code/CodeTokenizer.ts +54 -0
  345. package/src/lib/code/DefaultHighlightStyle.ts +98 -0
  346. package/src/lib/code/LezerHighlighter.ts +113 -0
  347. package/src/lib/code/diff.test.ts +311 -0
  348. package/src/lib/code/diff.ts +319 -0
  349. package/src/lib/code/extractRange.ts +125 -0
  350. package/src/lib/code/index.ts +13 -0
  351. package/src/lib/components/Bezier.ts +103 -0
  352. package/src/lib/components/Camera.ts +359 -0
  353. package/src/lib/components/Circle.ts +269 -0
  354. package/src/lib/components/Code.ts +532 -0
  355. package/src/lib/components/CodeBlock.ts +581 -0
  356. package/src/lib/components/CubicBezier.ts +115 -0
  357. package/src/lib/components/Curve.ts +455 -0
  358. package/src/lib/components/Grid.ts +134 -0
  359. package/src/lib/components/Icon.ts +95 -0
  360. package/src/lib/components/Img.ts +305 -0
  361. package/src/lib/components/Knot.ts +156 -0
  362. package/src/lib/components/Latex.ts +249 -0
  363. package/src/lib/components/Layout.ts +1071 -0
  364. package/src/lib/components/Line.ts +394 -0
  365. package/src/lib/components/Node.ts +1949 -0
  366. package/src/lib/components/Path.ts +132 -0
  367. package/src/lib/components/Polygon.ts +238 -0
  368. package/src/lib/components/QuadBezier.ts +103 -0
  369. package/src/lib/components/Ray.ts +126 -0
  370. package/src/lib/components/Rect.ts +186 -0
  371. package/src/lib/components/SVG.ts +788 -0
  372. package/src/lib/components/Shape.ts +146 -0
  373. package/src/lib/components/Spline.ts +318 -0
  374. package/src/lib/components/Txt.test.tsx +81 -0
  375. package/src/lib/components/Txt.ts +204 -0
  376. package/src/lib/components/TxtLeaf.ts +210 -0
  377. package/src/lib/components/Video.ts +368 -0
  378. package/src/lib/components/View2D.ts +85 -0
  379. package/src/lib/components/__logs__/image-without-source.md +17 -0
  380. package/src/lib/components/__logs__/line-without-points.md +30 -0
  381. package/src/lib/components/__logs__/reactive-playback-rate.md +21 -0
  382. package/src/lib/components/__logs__/spline-with-insufficient-knots.md +24 -0
  383. package/src/lib/components/__tests__/children.test.tsx +142 -0
  384. package/src/lib/components/__tests__/clone.test.tsx +126 -0
  385. package/src/lib/components/__tests__/generatorTest.ts +27 -0
  386. package/src/lib/components/__tests__/mockScene2D.ts +50 -0
  387. package/src/lib/components/__tests__/query.test.tsx +122 -0
  388. package/src/lib/components/__tests__/state.test.tsx +60 -0
  389. package/src/lib/components/index.ts +26 -0
  390. package/src/lib/components/types.ts +35 -0
  391. package/src/lib/curves/ArcSegment.ts +155 -0
  392. package/src/lib/curves/CircleSegment.ts +77 -0
  393. package/src/lib/curves/CubicBezierSegment.ts +78 -0
  394. package/src/lib/curves/CurveDrawingInfo.ts +11 -0
  395. package/src/lib/curves/CurvePoint.ts +15 -0
  396. package/src/lib/curves/CurveProfile.ts +7 -0
  397. package/src/lib/curves/KnotInfo.ts +10 -0
  398. package/src/lib/curves/LineSegment.ts +62 -0
  399. package/src/lib/curves/Polynomial.ts +355 -0
  400. package/src/lib/curves/Polynomial2D.ts +62 -0
  401. package/src/lib/curves/PolynomialSegment.ts +124 -0
  402. package/src/lib/curves/QuadBezierSegment.ts +64 -0
  403. package/src/lib/curves/Segment.ts +17 -0
  404. package/src/lib/curves/UniformPolynomialCurveSampler.ts +93 -0
  405. package/src/lib/curves/createCurveProfileLerp.ts +471 -0
  406. package/src/lib/curves/getBezierSplineProfile.ts +227 -0
  407. package/src/lib/curves/getCircleProfile.ts +86 -0
  408. package/src/lib/curves/getPathProfile.ts +177 -0
  409. package/src/lib/curves/getPointAtDistance.ts +21 -0
  410. package/src/lib/curves/getPolylineProfile.test.ts +21 -0
  411. package/src/lib/curves/getPolylineProfile.ts +88 -0
  412. package/src/lib/curves/getRectProfile.ts +138 -0
  413. package/src/lib/curves/index.ts +16 -0
  414. package/src/lib/decorators/canvasStyleSignal.ts +15 -0
  415. package/src/lib/decorators/colorSignal.ts +9 -0
  416. package/src/lib/decorators/compound.ts +85 -0
  417. package/src/lib/decorators/computed.ts +18 -0
  418. package/src/lib/decorators/defaultStyle.ts +15 -0
  419. package/src/lib/decorators/filtersSignal.ts +133 -0
  420. package/src/lib/decorators/index.ts +10 -0
  421. package/src/lib/decorators/initializers.ts +34 -0
  422. package/src/lib/decorators/nodeName.ts +13 -0
  423. package/src/lib/decorators/signal.test.ts +89 -0
  424. package/src/lib/decorators/signal.ts +348 -0
  425. package/src/lib/decorators/spacingSignal.ts +15 -0
  426. package/src/lib/decorators/vector2Signal.ts +35 -0
  427. package/src/lib/globals.d.ts +3 -0
  428. package/src/lib/index.ts +8 -0
  429. package/src/lib/jsx-dev-runtime.ts +2 -0
  430. package/src/lib/jsx-runtime.ts +45 -0
  431. package/src/lib/parse-svg-path.d.ts +14 -0
  432. package/src/lib/partials/Filter.ts +185 -0
  433. package/src/lib/partials/Gradient.ts +103 -0
  434. package/src/lib/partials/Pattern.ts +35 -0
  435. package/src/lib/partials/ShaderConfig.ts +122 -0
  436. package/src/lib/partials/index.ts +4 -0
  437. package/src/lib/partials/types.ts +58 -0
  438. package/src/lib/scenes/Scene2D.ts +195 -0
  439. package/src/lib/scenes/index.ts +3 -0
  440. package/src/lib/scenes/makeScene2D.ts +19 -0
  441. package/src/lib/scenes/useScene2D.ts +6 -0
  442. package/src/lib/tsconfig.build.json +12 -0
  443. package/src/lib/tsconfig.json +14 -0
  444. package/src/lib/tsdoc.json +4 -0
  445. package/src/lib/utils/CanvasUtils.ts +306 -0
  446. package/src/lib/utils/diff.test.ts +453 -0
  447. package/src/lib/utils/diff.ts +148 -0
  448. package/src/lib/utils/index.ts +3 -0
  449. package/src/lib/utils/is.ts +11 -0
  450. package/src/lib/utils/makeSignalExtensions.ts +29 -0
  451. package/src/lib/utils/withDefaults.tsx +26 -0
  452. package/src/tsconfig.base.json +18 -0
  453. package/src/tsconfig.build.json +8 -0
  454. package/src/tsconfig.json +5 -0
  455. package/tsconfig.project.json +7 -0
@@ -0,0 +1,1071 @@
1
+ import {
2
+ BBox,
3
+ boolLerp,
4
+ Direction,
5
+ InterpolationFunction,
6
+ modify,
7
+ Origin,
8
+ originToOffset,
9
+ PossibleSpacing,
10
+ PossibleVector2,
11
+ SerializedVector2,
12
+ Signal,
13
+ SignalValue,
14
+ SimpleSignal,
15
+ SimpleVector2Signal,
16
+ SpacingSignal,
17
+ threadable,
18
+ ThreadGenerator,
19
+ TimingFunction,
20
+ tween,
21
+ Vector2,
22
+ Vector2Signal,
23
+ } from '@efxlab/motion-canvas-core';
24
+ import {
25
+ addInitializer,
26
+ cloneable,
27
+ computed,
28
+ defaultStyle,
29
+ getPropertyMeta,
30
+ initial,
31
+ interpolation,
32
+ nodeName,
33
+ signal,
34
+ Vector2LengthSignal,
35
+ vector2Signal,
36
+ } from '../decorators';
37
+ import {spacingSignal} from '../decorators/spacingSignal';
38
+ import {
39
+ DesiredLength,
40
+ FlexBasis,
41
+ FlexContent,
42
+ FlexDirection,
43
+ FlexItems,
44
+ FlexWrap,
45
+ LayoutMode,
46
+ Length,
47
+ LengthLimit,
48
+ TextWrap,
49
+ } from '../partials';
50
+ import {drawLine, drawPivot, is} from '../utils';
51
+ import {Node, NodeProps} from './Node';
52
+
53
+ export interface LayoutProps extends NodeProps {
54
+ layout?: LayoutMode;
55
+ tagName?: keyof HTMLElementTagNameMap;
56
+
57
+ width?: SignalValue<Length>;
58
+ height?: SignalValue<Length>;
59
+ maxWidth?: SignalValue<LengthLimit>;
60
+ maxHeight?: SignalValue<LengthLimit>;
61
+ minWidth?: SignalValue<LengthLimit>;
62
+ minHeight?: SignalValue<LengthLimit>;
63
+ ratio?: SignalValue<number>;
64
+
65
+ marginTop?: SignalValue<number>;
66
+ marginBottom?: SignalValue<number>;
67
+ marginLeft?: SignalValue<number>;
68
+ marginRight?: SignalValue<number>;
69
+ margin?: SignalValue<PossibleSpacing>;
70
+
71
+ paddingTop?: SignalValue<number>;
72
+ paddingBottom?: SignalValue<number>;
73
+ paddingLeft?: SignalValue<number>;
74
+ paddingRight?: SignalValue<number>;
75
+ padding?: SignalValue<PossibleSpacing>;
76
+
77
+ direction?: SignalValue<FlexDirection>;
78
+ basis?: SignalValue<FlexBasis>;
79
+ grow?: SignalValue<number>;
80
+ shrink?: SignalValue<number>;
81
+ wrap?: SignalValue<FlexWrap>;
82
+
83
+ justifyContent?: SignalValue<FlexContent>;
84
+ alignContent?: SignalValue<FlexContent>;
85
+ alignItems?: SignalValue<FlexItems>;
86
+ alignSelf?: SignalValue<FlexItems>;
87
+ rowGap?: SignalValue<Length>;
88
+ columnGap?: SignalValue<Length>;
89
+ gap?: SignalValue<PossibleVector2<Length>>;
90
+
91
+ fontFamily?: SignalValue<string>;
92
+ fontSize?: SignalValue<number>;
93
+ fontStyle?: SignalValue<string>;
94
+ fontWeight?: SignalValue<number>;
95
+ lineHeight?: SignalValue<Length>;
96
+ letterSpacing?: SignalValue<number>;
97
+ textWrap?: SignalValue<TextWrap>;
98
+ textDirection?: SignalValue<CanvasDirection>;
99
+ textAlign?: SignalValue<CanvasTextAlign>;
100
+
101
+ size?: SignalValue<PossibleVector2<Length>>;
102
+ offsetX?: SignalValue<number>;
103
+ offsetY?: SignalValue<number>;
104
+ offset?: SignalValue<PossibleVector2>;
105
+ /**
106
+ * The position of the center of this node.
107
+ *
108
+ * @remarks
109
+ * This shortcut property will set the node's position so that the center ends
110
+ * up in the given place.
111
+ * If present, overrides the {@link NodeProps.position} property.
112
+ * When {@link offset} is not set, this will be the same as the
113
+ * {@link NodeProps.position}.
114
+ */
115
+ middle?: SignalValue<PossibleVector2>;
116
+ /**
117
+ * The position of the top edge of this node.
118
+ *
119
+ * @remarks
120
+ * This shortcut property will set the node's position so that the top edge
121
+ * ends up in the given place.
122
+ * If present, overrides the {@link NodeProps.position} property.
123
+ */
124
+ top?: SignalValue<PossibleVector2>;
125
+ /**
126
+ * The position of the bottom edge of this node.
127
+ *
128
+ * @remarks
129
+ * This shortcut property will set the node's position so that the bottom edge
130
+ * ends up in the given place.
131
+ * If present, overrides the {@link NodeProps.position} property.
132
+ */
133
+ bottom?: SignalValue<PossibleVector2>;
134
+ /**
135
+ * The position of the left edge of this node.
136
+ *
137
+ * @remarks
138
+ * This shortcut property will set the node's position so that the left edge
139
+ * ends up in the given place.
140
+ * If present, overrides the {@link NodeProps.position} property.
141
+ */
142
+ left?: SignalValue<PossibleVector2>;
143
+ /**
144
+ * The position of the right edge of this node.
145
+ *
146
+ * @remarks
147
+ * This shortcut property will set the node's position so that the right edge
148
+ * ends up in the given place.
149
+ * If present, overrides the {@link NodeProps.position} property.
150
+ */
151
+ right?: SignalValue<PossibleVector2>;
152
+ /**
153
+ * The position of the top left corner of this node.
154
+ *
155
+ * @remarks
156
+ * This shortcut property will set the node's position so that the top left
157
+ * corner ends up in the given place.
158
+ * If present, overrides the {@link NodeProps.position} property.
159
+ */
160
+ topLeft?: SignalValue<PossibleVector2>;
161
+ /**
162
+ * The position of the top right corner of this node.
163
+ *
164
+ * @remarks
165
+ * This shortcut property will set the node's position so that the top right
166
+ * corner ends up in the given place.
167
+ * If present, overrides the {@link NodeProps.position} property.
168
+ */
169
+ topRight?: SignalValue<PossibleVector2>;
170
+ /**
171
+ * The position of the bottom left corner of this node.
172
+ *
173
+ * @remarks
174
+ * This shortcut property will set the node's position so that the bottom left
175
+ * corner ends up in the given place.
176
+ * If present, overrides the {@link NodeProps.position} property.
177
+ */
178
+ bottomLeft?: SignalValue<PossibleVector2>;
179
+ /**
180
+ * The position of the bottom right corner of this node.
181
+ *
182
+ * @remarks
183
+ * This shortcut property will set the node's position so that the bottom
184
+ * right corner ends up in the given place.
185
+ * If present, overrides the {@link NodeProps.position} property.
186
+ */
187
+ bottomRight?: SignalValue<PossibleVector2>;
188
+ clip?: SignalValue<boolean>;
189
+ }
190
+
191
+ @nodeName('Layout')
192
+ export class Layout extends Node {
193
+ @initial(null)
194
+ @interpolation(boolLerp)
195
+ @signal()
196
+ declare public readonly layout: SimpleSignal<LayoutMode, this>;
197
+
198
+ @initial(null)
199
+ @signal()
200
+ declare public readonly maxWidth: SimpleSignal<LengthLimit, this>;
201
+ @initial(null)
202
+ @signal()
203
+ declare public readonly maxHeight: SimpleSignal<LengthLimit, this>;
204
+ @initial(null)
205
+ @signal()
206
+ declare public readonly minWidth: SimpleSignal<LengthLimit, this>;
207
+ @initial(null)
208
+ @signal()
209
+ declare public readonly minHeight: SimpleSignal<LengthLimit, this>;
210
+ @initial(null)
211
+ @signal()
212
+ declare public readonly ratio: SimpleSignal<number | null, this>;
213
+
214
+ @spacingSignal('margin')
215
+ declare public readonly margin: SpacingSignal<this>;
216
+
217
+ @spacingSignal('padding')
218
+ declare public readonly padding: SpacingSignal<this>;
219
+
220
+ @initial('row')
221
+ @signal()
222
+ declare public readonly direction: SimpleSignal<FlexDirection, this>;
223
+ @initial(null)
224
+ @signal()
225
+ declare public readonly basis: SimpleSignal<FlexBasis, this>;
226
+ @initial(0)
227
+ @signal()
228
+ declare public readonly grow: SimpleSignal<number, this>;
229
+ @initial(1)
230
+ @signal()
231
+ declare public readonly shrink: SimpleSignal<number, this>;
232
+ @initial('nowrap')
233
+ @signal()
234
+ declare public readonly wrap: SimpleSignal<FlexWrap, this>;
235
+
236
+ @initial('start')
237
+ @signal()
238
+ declare public readonly justifyContent: SimpleSignal<FlexContent, this>;
239
+ @initial('normal')
240
+ @signal()
241
+ declare public readonly alignContent: SimpleSignal<FlexContent, this>;
242
+ @initial('stretch')
243
+ @signal()
244
+ declare public readonly alignItems: SimpleSignal<FlexItems, this>;
245
+ @initial('auto')
246
+ @signal()
247
+ declare public readonly alignSelf: SimpleSignal<FlexItems, this>;
248
+ @initial(0)
249
+ @vector2Signal({x: 'columnGap', y: 'rowGap'})
250
+ declare public readonly gap: Vector2LengthSignal<this>;
251
+ public get columnGap(): Signal<Length, number, this> {
252
+ return this.gap.x;
253
+ }
254
+ public get rowGap(): Signal<Length, number, this> {
255
+ return this.gap.y;
256
+ }
257
+
258
+ @defaultStyle('Roboto')
259
+ @signal()
260
+ declare public readonly fontFamily: SimpleSignal<string, this>;
261
+ @defaultStyle(48)
262
+ @signal()
263
+ declare public readonly fontSize: SimpleSignal<number, this>;
264
+ @defaultStyle('normal')
265
+ @signal()
266
+ declare public readonly fontStyle: SimpleSignal<string, this>;
267
+ @defaultStyle(500)
268
+ @signal()
269
+ declare public readonly fontWeight: SimpleSignal<number, this>;
270
+ @defaultStyle('120%')
271
+ @signal()
272
+ declare public readonly lineHeight: SimpleSignal<Length, this>;
273
+ @defaultStyle(0)
274
+ @signal()
275
+ declare public readonly letterSpacing: SimpleSignal<number, this>;
276
+
277
+ @defaultStyle(false)
278
+ @signal()
279
+ declare public readonly textWrap: SimpleSignal<TextWrap, this>;
280
+ @initial('ltr')
281
+ @signal()
282
+ declare public readonly textDirection: SimpleSignal<CanvasDirection, this>;
283
+ @defaultStyle('start')
284
+ @signal()
285
+ declare public readonly textAlign: SimpleSignal<CanvasTextAlign, this>;
286
+
287
+ protected getX(): number {
288
+ if (this.isLayoutRoot()) {
289
+ return this.x.context.getter();
290
+ }
291
+
292
+ return this.computedPosition().x;
293
+ }
294
+ protected setX(value: SignalValue<number>) {
295
+ this.x.context.setter(value);
296
+ }
297
+
298
+ protected getY(): number {
299
+ if (this.isLayoutRoot()) {
300
+ return this.y.context.getter();
301
+ }
302
+
303
+ return this.computedPosition().y;
304
+ }
305
+ protected setY(value: SignalValue<number>) {
306
+ this.y.context.setter(value);
307
+ }
308
+
309
+ /**
310
+ * Represents the size of this node.
311
+ *
312
+ * @remarks
313
+ * A size is a two-dimensional vector, where `x` represents the `width`, and `y`
314
+ * represents the `height`.
315
+ *
316
+ * The value of both x and y is of type {@link partials.Length} which is
317
+ * either:
318
+ * - `number` - the desired length in pixels
319
+ * - `${number}%` - a string with the desired length in percents, for example
320
+ * `'50%'`
321
+ * - `null` - an automatic length
322
+ *
323
+ * When retrieving the size, all units are converted to pixels, using the
324
+ * current state of the layout. For example, retrieving the width set to
325
+ * `'50%'`, while the parent has a width of `200px` will result in the number
326
+ * `100` being returned.
327
+ *
328
+ * When the node is not part of the layout, setting its size using percents
329
+ * refers to the size of the entire scene.
330
+ *
331
+ * @example
332
+ * Initializing the size:
333
+ * ```tsx
334
+ * // with a possible vector:
335
+ * <Node size={['50%', 200]} />
336
+ * // with individual components:
337
+ * <Node width={'50%'} height={200} />
338
+ * ```
339
+ *
340
+ * Accessing the size:
341
+ * ```tsx
342
+ * // retrieving the vector:
343
+ * const size = node.size();
344
+ * // retrieving an individual component:
345
+ * const width = node.size.x();
346
+ * ```
347
+ *
348
+ * Setting the size:
349
+ * ```tsx
350
+ * // with a possible vector:
351
+ * node.size(['50%', 200]);
352
+ * node.size(() => ['50%', 200]);
353
+ * // with individual components:
354
+ * node.size.x('50%');
355
+ * node.size.x(() => '50%');
356
+ * ```
357
+ */
358
+ @initial({x: null, y: null})
359
+ @vector2Signal({x: 'width', y: 'height'})
360
+ declare public readonly size: Vector2LengthSignal<this>;
361
+ public get width(): Signal<Length, number, this> {
362
+ return this.size.x;
363
+ }
364
+ public get height(): Signal<Length, number, this> {
365
+ return this.size.y;
366
+ }
367
+
368
+ protected getWidth(): number {
369
+ return this.computedSize().width;
370
+ }
371
+ protected setWidth(value: SignalValue<Length>) {
372
+ this.width.context.setter(value);
373
+ }
374
+
375
+ @threadable()
376
+ protected *tweenWidth(
377
+ value: SignalValue<Length>,
378
+ time: number,
379
+ timingFunction: TimingFunction,
380
+ interpolationFunction: InterpolationFunction<Length>,
381
+ ): ThreadGenerator {
382
+ const width = this.desiredSize().x;
383
+ const lock = typeof width !== 'number' || typeof value !== 'number';
384
+ let from: number;
385
+ if (lock) {
386
+ from = this.size.x();
387
+ } else {
388
+ from = width;
389
+ }
390
+
391
+ let to: number;
392
+ if (lock) {
393
+ this.size.x(value);
394
+ to = this.size.x();
395
+ } else {
396
+ to = value;
397
+ }
398
+
399
+ this.size.x(from);
400
+ lock && this.lockSize();
401
+ yield* tween(time, value =>
402
+ this.size.x(interpolationFunction(from, to, timingFunction(value))),
403
+ );
404
+ this.size.x(value);
405
+ lock && this.releaseSize();
406
+ }
407
+
408
+ protected getHeight(): number {
409
+ return this.computedSize().height;
410
+ }
411
+ protected setHeight(value: SignalValue<Length>) {
412
+ this.height.context.setter(value);
413
+ }
414
+
415
+ @threadable()
416
+ protected *tweenHeight(
417
+ value: SignalValue<Length>,
418
+ time: number,
419
+ timingFunction: TimingFunction,
420
+ interpolationFunction: InterpolationFunction<Length>,
421
+ ): ThreadGenerator {
422
+ const height = this.desiredSize().y;
423
+ const lock = typeof height !== 'number' || typeof value !== 'number';
424
+
425
+ let from: number;
426
+ if (lock) {
427
+ from = this.size.y();
428
+ } else {
429
+ from = height;
430
+ }
431
+
432
+ let to: number;
433
+ if (lock) {
434
+ this.size.y(value);
435
+ to = this.size.y();
436
+ } else {
437
+ to = value;
438
+ }
439
+
440
+ this.size.y(from);
441
+ lock && this.lockSize();
442
+ yield* tween(time, value =>
443
+ this.size.y(interpolationFunction(from, to, timingFunction(value))),
444
+ );
445
+ this.size.y(value);
446
+ lock && this.releaseSize();
447
+ }
448
+
449
+ /**
450
+ * Get the desired size of this node.
451
+ *
452
+ * @remarks
453
+ * This method can be used to control the size using external factors.
454
+ * By default, the returned size is the same as the one declared by the user.
455
+ */
456
+ @computed()
457
+ protected desiredSize(): SerializedVector2<DesiredLength> {
458
+ return {
459
+ x: this.width.context.getter(),
460
+ y: this.height.context.getter(),
461
+ };
462
+ }
463
+
464
+ @threadable()
465
+ protected *tweenSize(
466
+ value: SignalValue<SerializedVector2<Length>>,
467
+ time: number,
468
+ timingFunction: TimingFunction,
469
+ interpolationFunction: InterpolationFunction<Vector2>,
470
+ ): ThreadGenerator {
471
+ const size = this.desiredSize();
472
+ let from: Vector2;
473
+ if (typeof size.x !== 'number' || typeof size.y !== 'number') {
474
+ from = this.size();
475
+ } else {
476
+ from = new Vector2(<Vector2>size);
477
+ }
478
+
479
+ let to: Vector2;
480
+ if (
481
+ typeof value === 'object' &&
482
+ typeof value.x === 'number' &&
483
+ typeof value.y === 'number'
484
+ ) {
485
+ to = new Vector2(<Vector2>value);
486
+ } else {
487
+ this.size(value);
488
+ to = this.size();
489
+ }
490
+
491
+ this.size(from);
492
+ this.lockSize();
493
+ yield* tween(time, value =>
494
+ this.size(interpolationFunction(from, to, timingFunction(value))),
495
+ );
496
+ this.releaseSize();
497
+ this.size(value);
498
+ }
499
+
500
+ /**
501
+ * Represents the offset of this node's origin.
502
+ *
503
+ * @remarks
504
+ * By default, the origin of a node is located at its center. The origin
505
+ * serves as the pivot point when rotating and scaling a node, but it doesn't
506
+ * affect the placement of its children.
507
+ *
508
+ * The value is relative to the size of this node. A value of `1` means as far
509
+ * to the right/bottom as possible. Here are a few examples of offsets:
510
+ * - `[-1, -1]` - top left corner
511
+ * - `[1, -1]` - top right corner
512
+ * - `[0, 1]` - bottom edge
513
+ * - `[-1, 1]` - bottom left corner
514
+ */
515
+ @vector2Signal('offset')
516
+ declare public readonly offset: Vector2Signal<this>;
517
+
518
+ /**
519
+ * The position of the center of this node.
520
+ *
521
+ * @remarks
522
+ * When set, this shortcut property will modify the node's position so that
523
+ * the center ends up in the given place.
524
+ *
525
+ * If the {@link offset} has not been changed, this will be the same as the
526
+ * {@link position}.
527
+ *
528
+ * When retrieved, it will return the position of the center in the parent
529
+ * space.
530
+ */
531
+ @originSignal(Origin.Middle)
532
+ declare public readonly middle: SimpleVector2Signal<this>;
533
+
534
+ /**
535
+ * The position of the top edge of this node.
536
+ *
537
+ * @remarks
538
+ * When set, this shortcut property will modify the node's position so that
539
+ * the top edge ends up in the given place.
540
+ *
541
+ * When retrieved, it will return the position of the top edge in the parent
542
+ * space.
543
+ */
544
+ @originSignal(Origin.Top)
545
+ declare public readonly top: SimpleVector2Signal<this>;
546
+ /**
547
+ * The position of the bottom edge of this node.
548
+ *
549
+ * @remarks
550
+ * When set, this shortcut property will modify the node's position so that
551
+ * the bottom edge ends up in the given place.
552
+ *
553
+ * When retrieved, it will return the position of the bottom edge in the
554
+ * parent space.
555
+ */
556
+ @originSignal(Origin.Bottom)
557
+ declare public readonly bottom: SimpleVector2Signal<this>;
558
+ /**
559
+ * The position of the left edge of this node.
560
+ *
561
+ * @remarks
562
+ * When set, this shortcut property will modify the node's position so that
563
+ * the left edge ends up in the given place.
564
+ *
565
+ * When retrieved, it will return the position of the left edge in the parent
566
+ * space.
567
+ */
568
+ @originSignal(Origin.Left)
569
+ declare public readonly left: SimpleVector2Signal<this>;
570
+ /**
571
+ * The position of the right edge of this node.
572
+ *
573
+ * @remarks
574
+ * When set, this shortcut property will modify the node's position so that
575
+ * the right edge ends up in the given place.
576
+ *
577
+ * When retrieved, it will return the position of the right edge in the parent
578
+ * space.
579
+ */
580
+ @originSignal(Origin.Right)
581
+ declare public readonly right: SimpleVector2Signal<this>;
582
+ /**
583
+ * The position of the top left corner of this node.
584
+ *
585
+ * @remarks
586
+ * When set, this shortcut property will modify the node's position so that
587
+ * the top left corner ends up in the given place.
588
+ *
589
+ * When retrieved, it will return the position of the top left corner in the
590
+ * parent space.
591
+ */
592
+ @originSignal(Origin.TopLeft)
593
+ declare public readonly topLeft: SimpleVector2Signal<this>;
594
+ /**
595
+ * The position of the top right corner of this node.
596
+ *
597
+ * @remarks
598
+ * When set, this shortcut property will modify the node's position so that
599
+ * the top right corner ends up in the given place.
600
+ *
601
+ * When retrieved, it will return the position of the top right corner in the
602
+ * parent space.
603
+ */
604
+ @originSignal(Origin.TopRight)
605
+ declare public readonly topRight: SimpleVector2Signal<this>;
606
+ /**
607
+ * The position of the bottom left corner of this node.
608
+ *
609
+ * @remarks
610
+ * When set, this shortcut property will modify the node's position so that
611
+ * the bottom left corner ends up in the given place.
612
+ *
613
+ * When retrieved, it will return the position of the bottom left corner in
614
+ * the parent space.
615
+ */
616
+ @originSignal(Origin.BottomLeft)
617
+ declare public readonly bottomLeft: SimpleVector2Signal<this>;
618
+ /**
619
+ * The position of the bottom right corner of this node.
620
+ *
621
+ * @remarks
622
+ * When set, this shortcut property will modify the node's position so that
623
+ * the bottom right corner ends up in the given place.
624
+ *
625
+ * When retrieved, it will return the position of the bottom right corner in
626
+ * the parent space.
627
+ */
628
+ @originSignal(Origin.BottomRight)
629
+ declare public readonly bottomRight: SimpleVector2Signal<this>;
630
+
631
+ /**
632
+ * Get the cardinal point corresponding to the given origin.
633
+ *
634
+ * @param origin - The origin or direction of the point.
635
+ */
636
+ public cardinalPoint(origin: Origin | Direction): SimpleVector2Signal<this> {
637
+ switch (origin) {
638
+ case Origin.TopLeft:
639
+ return this.topLeft;
640
+ case Origin.TopRight:
641
+ return this.topRight;
642
+ case Origin.BottomLeft:
643
+ return this.bottomLeft;
644
+ case Origin.BottomRight:
645
+ return this.bottomRight;
646
+ case Origin.Top:
647
+ case Direction.Top:
648
+ return this.top;
649
+ case Origin.Bottom:
650
+ case Direction.Bottom:
651
+ return this.bottom;
652
+ case Origin.Left:
653
+ case Direction.Left:
654
+ return this.left;
655
+ case Origin.Right:
656
+ case Direction.Right:
657
+ return this.right;
658
+ default:
659
+ return this.middle;
660
+ }
661
+ }
662
+
663
+ @initial(false)
664
+ @signal()
665
+ declare public readonly clip: SimpleSignal<boolean, this>;
666
+
667
+ declare public element: HTMLElement;
668
+ declare public styles: CSSStyleDeclaration;
669
+
670
+ @initial(0)
671
+ @signal()
672
+ declare protected readonly sizeLockCounter: SimpleSignal<number, this>;
673
+
674
+ public constructor(props: LayoutProps) {
675
+ super(props);
676
+ this.element.dataset.motionCanvasKey = this.key;
677
+ }
678
+
679
+ public lockSize() {
680
+ this.sizeLockCounter(this.sizeLockCounter() + 1);
681
+ }
682
+
683
+ public releaseSize() {
684
+ this.sizeLockCounter(this.sizeLockCounter() - 1);
685
+ }
686
+
687
+ @computed()
688
+ protected parentTransform(): Layout | null {
689
+ return this.findAncestor(is(Layout));
690
+ }
691
+
692
+ @computed()
693
+ public anchorPosition() {
694
+ const size = this.computedSize();
695
+ const offset = this.offset();
696
+
697
+ return size.scale(0.5).mul(offset);
698
+ }
699
+
700
+ /**
701
+ * Get the resolved layout mode of this node.
702
+ *
703
+ * @remarks
704
+ * When the mode is `null`, its value will be inherited from the parent.
705
+ *
706
+ * Use {@link layout} to get the raw mode set for this node (without
707
+ * inheritance).
708
+ */
709
+ @computed()
710
+ public layoutEnabled(): boolean {
711
+ return this.layout() ?? this.parentTransform()?.layoutEnabled() ?? false;
712
+ }
713
+
714
+ @computed()
715
+ public isLayoutRoot(): boolean {
716
+ return !this.layoutEnabled() || !this.parentTransform()?.layoutEnabled();
717
+ }
718
+
719
+ public override localToParent(): DOMMatrix {
720
+ const matrix = super.localToParent();
721
+ const offset = this.offset();
722
+ if (!offset.exactlyEquals(Vector2.zero)) {
723
+ const translate = this.size().mul(offset).scale(-0.5);
724
+ matrix.translateSelf(translate.x, translate.y);
725
+ }
726
+
727
+ return matrix;
728
+ }
729
+
730
+ /**
731
+ * A simplified version of {@link localToParent} matrix used for transforming
732
+ * direction vectors.
733
+ *
734
+ * @internal
735
+ */
736
+ @computed()
737
+ protected scalingRotationMatrix(): DOMMatrix {
738
+ const matrix = new DOMMatrix();
739
+
740
+ matrix.rotateSelf(0, 0, this.rotation());
741
+ matrix.scaleSelf(this.scale.x(), this.scale.y());
742
+
743
+ const offset = this.offset();
744
+ if (!offset.exactlyEquals(Vector2.zero)) {
745
+ const translate = this.size().mul(offset).scale(-0.5);
746
+ matrix.translateSelf(translate.x, translate.y);
747
+ }
748
+
749
+ return matrix;
750
+ }
751
+
752
+ protected getComputedLayout(): BBox {
753
+ return new BBox(this.element.getBoundingClientRect());
754
+ }
755
+
756
+ @computed()
757
+ public computedPosition(): Vector2 {
758
+ this.requestLayoutUpdate();
759
+ const box = this.getComputedLayout();
760
+
761
+ const position = new Vector2(
762
+ box.x + (box.width / 2) * this.offset.x(),
763
+ box.y + (box.height / 2) * this.offset.y(),
764
+ );
765
+
766
+ const parent = this.parentTransform();
767
+ if (parent) {
768
+ const parentRect = parent.getComputedLayout();
769
+ position.x -= parentRect.x + (parentRect.width - box.width) / 2;
770
+ position.y -= parentRect.y + (parentRect.height - box.height) / 2;
771
+ }
772
+
773
+ return position;
774
+ }
775
+
776
+ @computed()
777
+ protected computedSize(): Vector2 {
778
+ this.requestLayoutUpdate();
779
+ return this.getComputedLayout().size;
780
+ }
781
+
782
+ /**
783
+ * Find the closest layout root and apply any new layout changes.
784
+ */
785
+ @computed()
786
+ protected requestLayoutUpdate() {
787
+ const parent = this.parentTransform();
788
+ if (this.appendedToView()) {
789
+ parent?.requestFontUpdate();
790
+ this.updateLayout();
791
+ } else {
792
+ parent!.requestLayoutUpdate();
793
+ }
794
+ }
795
+
796
+ @computed()
797
+ protected appendedToView() {
798
+ const root = this.isLayoutRoot();
799
+ if (root) {
800
+ this.view().element.append(this.element);
801
+ }
802
+
803
+ return root;
804
+ }
805
+
806
+ /**
807
+ * Apply any new layout changes to this node and its children.
808
+ */
809
+ @computed()
810
+ protected updateLayout() {
811
+ this.applyFont();
812
+ this.applyFlex();
813
+ if (this.layoutEnabled()) {
814
+ const children = this.layoutChildren();
815
+ for (const child of children) {
816
+ child.updateLayout();
817
+ }
818
+ }
819
+ }
820
+
821
+ @computed()
822
+ protected layoutChildren(): Layout[] {
823
+ const queue = [...this.children()];
824
+ const result: Layout[] = [];
825
+ const elements: HTMLElement[] = [];
826
+ while (queue.length) {
827
+ const child = queue.shift();
828
+ if (child instanceof Layout) {
829
+ if (child.layoutEnabled()) {
830
+ result.push(child);
831
+ elements.push(child.element);
832
+ }
833
+ } else if (child) {
834
+ queue.unshift(...child.children());
835
+ }
836
+ }
837
+ this.element.replaceChildren(...elements);
838
+
839
+ return result;
840
+ }
841
+
842
+ /**
843
+ * Apply any new font changes to this node and all of its ancestors.
844
+ */
845
+ @computed()
846
+ protected requestFontUpdate() {
847
+ this.appendedToView();
848
+ this.parentTransform()?.requestFontUpdate();
849
+ this.applyFont();
850
+ }
851
+
852
+ protected override getCacheBBox(): BBox {
853
+ return BBox.fromSizeCentered(this.computedSize());
854
+ }
855
+
856
+ protected override draw(context: CanvasRenderingContext2D) {
857
+ if (this.clip()) {
858
+ const size = this.computedSize();
859
+ if (size.width === 0 || size.height === 0) {
860
+ return;
861
+ }
862
+
863
+ context.beginPath();
864
+ context.rect(size.width / -2, size.height / -2, size.width, size.height);
865
+ context.closePath();
866
+ context.clip();
867
+ }
868
+
869
+ this.drawChildren(context);
870
+ }
871
+
872
+ public override drawOverlay(
873
+ context: CanvasRenderingContext2D,
874
+ matrix: DOMMatrix,
875
+ ) {
876
+ const size = this.computedSize();
877
+ const offset = size.mul(this.offset()).scale(0.5).transformAsPoint(matrix);
878
+ const box = BBox.fromSizeCentered(size);
879
+ const layout = box.transformCorners(matrix);
880
+ const padding = box
881
+ .addSpacing(this.padding().scale(-1))
882
+ .transformCorners(matrix);
883
+ const margin = box.addSpacing(this.margin()).transformCorners(matrix);
884
+
885
+ context.beginPath();
886
+ drawLine(context, margin);
887
+ drawLine(context, layout);
888
+ context.closePath();
889
+ context.fillStyle = 'rgba(255,193,125,0.6)';
890
+ context.fill('evenodd');
891
+
892
+ context.beginPath();
893
+ drawLine(context, layout);
894
+ drawLine(context, padding);
895
+ context.closePath();
896
+ context.fillStyle = 'rgba(180,255,147,0.6)';
897
+ context.fill('evenodd');
898
+
899
+ context.beginPath();
900
+ drawLine(context, layout);
901
+ context.closePath();
902
+ context.lineWidth = 1;
903
+ context.strokeStyle = 'white';
904
+ context.stroke();
905
+
906
+ context.beginPath();
907
+ drawPivot(context, offset);
908
+ context.stroke();
909
+ }
910
+
911
+ public getOriginDelta(origin: Origin) {
912
+ const size = this.computedSize().scale(0.5);
913
+ const offset = this.offset().mul(size);
914
+ if (origin === Origin.Middle) {
915
+ return offset.flipped;
916
+ }
917
+
918
+ const newOffset = originToOffset(origin).mul(size);
919
+ return newOffset.sub(offset);
920
+ }
921
+
922
+ /**
923
+ * Update the offset of this node and adjust the position to keep it in the
924
+ * same place.
925
+ *
926
+ * @param offset - The new offset.
927
+ */
928
+ public moveOffset(offset: Vector2) {
929
+ const size = this.computedSize().scale(0.5);
930
+ const oldOffset = this.offset().mul(size);
931
+ const newOffset = offset.mul(size);
932
+ this.offset(offset);
933
+ this.position(this.position().add(newOffset).sub(oldOffset));
934
+ }
935
+
936
+ protected parsePixels(value: number | null): string {
937
+ return value === null ? '' : `${value}px`;
938
+ }
939
+
940
+ protected parseLength(value: number | string | null): string {
941
+ if (value === null) {
942
+ return '';
943
+ }
944
+ if (typeof value === 'string') {
945
+ return value;
946
+ }
947
+ return `${value}px`;
948
+ }
949
+
950
+ @computed()
951
+ protected applyFlex() {
952
+ this.element.style.position = this.isLayoutRoot() ? 'absolute' : 'relative';
953
+
954
+ const size = this.desiredSize();
955
+ this.element.style.width = this.parseLength(size.x);
956
+ this.element.style.height = this.parseLength(size.y);
957
+ this.element.style.maxWidth = this.parseLength(this.maxWidth());
958
+ this.element.style.minWidth = this.parseLength(this.minWidth());
959
+ this.element.style.maxHeight = this.parseLength(this.maxHeight());
960
+ this.element.style.minHeight = this.parseLength(this.minHeight()!);
961
+ this.element.style.aspectRatio =
962
+ this.ratio() === null ? '' : this.ratio()!.toString();
963
+
964
+ this.element.style.marginTop = this.parsePixels(this.margin.top());
965
+ this.element.style.marginBottom = this.parsePixels(this.margin.bottom());
966
+ this.element.style.marginLeft = this.parsePixels(this.margin.left());
967
+ this.element.style.marginRight = this.parsePixels(this.margin.right());
968
+
969
+ this.element.style.paddingTop = this.parsePixels(this.padding.top());
970
+ this.element.style.paddingBottom = this.parsePixels(this.padding.bottom());
971
+ this.element.style.paddingLeft = this.parsePixels(this.padding.left());
972
+ this.element.style.paddingRight = this.parsePixels(this.padding.right());
973
+
974
+ this.element.style.flexDirection = this.direction();
975
+ this.element.style.flexBasis = this.parseLength(this.basis()!);
976
+ this.element.style.flexWrap = this.wrap();
977
+
978
+ this.element.style.justifyContent = this.justifyContent();
979
+ this.element.style.alignContent = this.alignContent();
980
+ this.element.style.alignItems = this.alignItems();
981
+ this.element.style.alignSelf = this.alignSelf();
982
+ this.element.style.columnGap = this.parseLength(this.gap.x());
983
+ this.element.style.rowGap = this.parseLength(this.gap.y());
984
+
985
+ if (this.sizeLockCounter() > 0) {
986
+ this.element.style.flexGrow = '0';
987
+ this.element.style.flexShrink = '0';
988
+ } else {
989
+ this.element.style.flexGrow = this.grow().toString();
990
+ this.element.style.flexShrink = this.shrink().toString();
991
+ }
992
+ }
993
+
994
+ @computed()
995
+ protected applyFont() {
996
+ this.element.style.fontFamily = this.fontFamily();
997
+ this.element.style.fontSize = `${this.fontSize()}px`;
998
+ this.element.style.fontStyle = this.fontStyle();
999
+
1000
+ const lineHeight = this.lineHeight();
1001
+ this.element.style.lineHeight =
1002
+ typeof lineHeight === 'number'
1003
+ ? `${lineHeight}px`
1004
+ : (parseFloat(lineHeight as string) / 100).toString();
1005
+
1006
+ this.element.style.fontWeight = this.fontWeight().toString();
1007
+ this.element.style.letterSpacing = `${this.letterSpacing()}px`;
1008
+ this.element.style.textAlign = this.textAlign();
1009
+
1010
+ const wrap = this.textWrap();
1011
+ if (typeof wrap === 'boolean') {
1012
+ this.element.style.whiteSpace = wrap ? 'normal' : 'nowrap';
1013
+ } else {
1014
+ this.element.style.whiteSpace = wrap;
1015
+ }
1016
+ }
1017
+
1018
+ public override dispose() {
1019
+ super.dispose();
1020
+ this.sizeLockCounter?.context.dispose();
1021
+ if (this.element) {
1022
+ this.element.remove();
1023
+ this.element.innerHTML = '';
1024
+ }
1025
+ this.element = null as unknown as HTMLElement;
1026
+ this.styles = null as unknown as CSSStyleDeclaration;
1027
+ }
1028
+
1029
+ public override hit(position: Vector2): Node | null {
1030
+ const local = position.transformAsPoint(this.localToParent().inverse());
1031
+ if (this.cacheBBox().includes(local)) {
1032
+ return super.hit(position) ?? this;
1033
+ }
1034
+
1035
+ return null;
1036
+ }
1037
+ }
1038
+
1039
+ function originSignal(origin: Origin): PropertyDecorator {
1040
+ return (target, key) => {
1041
+ signal()(target, key);
1042
+ cloneable(false)(target, key);
1043
+ const meta = getPropertyMeta<any>(target, key);
1044
+ meta!.parser = value => new Vector2(value);
1045
+ meta!.getter = function (this: Layout) {
1046
+ return this.computedSize()
1047
+ .getOriginOffset(origin)
1048
+ .transformAsPoint(this.localToParent());
1049
+ };
1050
+ meta!.setter = function (
1051
+ this: Layout,
1052
+ value: SignalValue<PossibleVector2>,
1053
+ ) {
1054
+ this.position(
1055
+ modify(value, unwrapped =>
1056
+ this.getOriginDelta(origin)
1057
+ .transform(this.scalingRotationMatrix())
1058
+ .flipped.add(unwrapped),
1059
+ ),
1060
+ );
1061
+ return this;
1062
+ };
1063
+ };
1064
+ }
1065
+
1066
+ addInitializer<Layout>(Layout.prototype, instance => {
1067
+ instance.element = document.createElement('div');
1068
+ instance.element.style.display = 'flex';
1069
+ instance.element.style.boxSizing = 'border-box';
1070
+ instance.styles = getComputedStyle(instance.element);
1071
+ });