diff --git a/package-lock.json b/package-lock.json
index 55412b1..a6c48e8 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -14,6 +14,7 @@
"@testing-library/react": "^16.3.0",
"@testing-library/user-event": "^13.5.0",
"axios": "^0.27.2",
+ "d3": "^7.9.0",
"react": "^19.1.0",
"react-dom": "^19.1.0",
"react-icons": "^5.5.0",
@@ -6414,6 +6415,416 @@
"integrity": "sha512-b0tGHbfegbhPJpxpiBPU2sCkigAqtM9O121le6bbOlgyV+NyGyCmVfJ6QW9eRjz8CpNfWEOYBIMIGRYkLwsIYg==",
"license": "MIT"
},
+ "node_modules/d3": {
+ "version": "7.9.0",
+ "resolved": "https://registry.npmjs.org/d3/-/d3-7.9.0.tgz",
+ "integrity": "sha512-e1U46jVP+w7Iut8Jt8ri1YsPOvFpg46k+K8TpCb0P+zjCkjkPnV7WzfDJzMHy1LnA+wj5pLT1wjO901gLXeEhA==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-array": "3",
+ "d3-axis": "3",
+ "d3-brush": "3",
+ "d3-chord": "3",
+ "d3-color": "3",
+ "d3-contour": "4",
+ "d3-delaunay": "6",
+ "d3-dispatch": "3",
+ "d3-drag": "3",
+ "d3-dsv": "3",
+ "d3-ease": "3",
+ "d3-fetch": "3",
+ "d3-force": "3",
+ "d3-format": "3",
+ "d3-geo": "3",
+ "d3-hierarchy": "3",
+ "d3-interpolate": "3",
+ "d3-path": "3",
+ "d3-polygon": "3",
+ "d3-quadtree": "3",
+ "d3-random": "3",
+ "d3-scale": "4",
+ "d3-scale-chromatic": "3",
+ "d3-selection": "3",
+ "d3-shape": "3",
+ "d3-time": "3",
+ "d3-time-format": "4",
+ "d3-timer": "3",
+ "d3-transition": "3",
+ "d3-zoom": "3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-array": {
+ "version": "3.2.4",
+ "resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz",
+ "integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==",
+ "license": "ISC",
+ "dependencies": {
+ "internmap": "1 - 2"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-axis": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-axis/-/d3-axis-3.0.0.tgz",
+ "integrity": "sha512-IH5tgjV4jE/GhHkRV0HiVYPDtvfjHQlQfJHs0usq7M30XcSBvOotpmH1IgkcXsO/5gEQZD43B//fc7SRT5S+xw==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-brush": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-brush/-/d3-brush-3.0.0.tgz",
+ "integrity": "sha512-ALnjWlVYkXsVIGlOsuWH1+3udkYFI48Ljihfnh8FZPF2QS9o+PzGLBslO0PjzVoHLZ2KCVgAM8NVkXPJB2aNnQ==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-dispatch": "1 - 3",
+ "d3-drag": "2 - 3",
+ "d3-interpolate": "1 - 3",
+ "d3-selection": "3",
+ "d3-transition": "3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-chord": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-chord/-/d3-chord-3.0.1.tgz",
+ "integrity": "sha512-VE5S6TNa+j8msksl7HwjxMHDM2yNK3XCkusIlpX5kwauBfXuyLAtNg9jCp/iHH61tgI4sb6R/EIMWCqEIdjT/g==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-path": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-color": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz",
+ "integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-contour": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/d3-contour/-/d3-contour-4.0.2.tgz",
+ "integrity": "sha512-4EzFTRIikzs47RGmdxbeUvLWtGedDUNkTcmzoeyg4sP/dvCexO47AaQL7VKy/gul85TOxw+IBgA8US2xwbToNA==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-array": "^3.2.0"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-delaunay": {
+ "version": "6.0.4",
+ "resolved": "https://registry.npmjs.org/d3-delaunay/-/d3-delaunay-6.0.4.tgz",
+ "integrity": "sha512-mdjtIZ1XLAM8bm/hx3WwjfHt6Sggek7qH043O8KEjDXN40xi3vx/6pYSVTwLjEgiXQTbvaouWKynLBiUZ6SK6A==",
+ "license": "ISC",
+ "dependencies": {
+ "delaunator": "5"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-dispatch": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-dispatch/-/d3-dispatch-3.0.1.tgz",
+ "integrity": "sha512-rzUyPU/S7rwUflMyLc1ETDeBj0NRuHKKAcvukozwhshr6g6c5d8zh4c2gQjY2bZ0dXeGLWc1PF174P2tVvKhfg==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-drag": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-drag/-/d3-drag-3.0.0.tgz",
+ "integrity": "sha512-pWbUJLdETVA8lQNJecMxoXfH6x+mO2UQo8rSmZ+QqxcbyA3hfeprFgIT//HW2nlHChWeIIMwS2Fq+gEARkhTkg==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-dispatch": "1 - 3",
+ "d3-selection": "3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-dsv": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-dsv/-/d3-dsv-3.0.1.tgz",
+ "integrity": "sha512-UG6OvdI5afDIFP9w4G0mNq50dSOsXHJaRE8arAS5o9ApWnIElp8GZw1Dun8vP8OyHOZ/QJUKUJwxiiCCnUwm+Q==",
+ "license": "ISC",
+ "dependencies": {
+ "commander": "7",
+ "iconv-lite": "0.6",
+ "rw": "1"
+ },
+ "bin": {
+ "csv2json": "bin/dsv2json.js",
+ "csv2tsv": "bin/dsv2dsv.js",
+ "dsv2dsv": "bin/dsv2dsv.js",
+ "dsv2json": "bin/dsv2json.js",
+ "json2csv": "bin/json2dsv.js",
+ "json2dsv": "bin/json2dsv.js",
+ "json2tsv": "bin/json2dsv.js",
+ "tsv2csv": "bin/dsv2dsv.js",
+ "tsv2json": "bin/dsv2json.js"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-dsv/node_modules/commander": {
+ "version": "7.2.0",
+ "resolved": "https://registry.npmjs.org/commander/-/commander-7.2.0.tgz",
+ "integrity": "sha512-QrWXB+ZQSVPmIWIhtEO9H+gwHaMGYiF5ChvoJ+K9ZGHG/sVsa6yiesAD1GC/x46sET00Xlwo1u49RVVVzvcSkw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">= 10"
+ }
+ },
+ "node_modules/d3-ease": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz",
+ "integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==",
+ "license": "BSD-3-Clause",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-fetch": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-fetch/-/d3-fetch-3.0.1.tgz",
+ "integrity": "sha512-kpkQIM20n3oLVBKGg6oHrUchHM3xODkTzjMoj7aWQFq5QEM+R6E4WkzT5+tojDY7yjez8KgCBRoj4aEr99Fdqw==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-dsv": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-force": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-force/-/d3-force-3.0.0.tgz",
+ "integrity": "sha512-zxV/SsA+U4yte8051P4ECydjD/S+qeYtnaIyAs9tgHCqfguma/aAQDjo85A9Z6EKhBirHRJHXIgJUlffT4wdLg==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-dispatch": "1 - 3",
+ "d3-quadtree": "1 - 3",
+ "d3-timer": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-format": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.0.tgz",
+ "integrity": "sha512-YyUI6AEuY/Wpt8KWLgZHsIU86atmikuoOmCfommt0LYHiQSPjvX2AcFc38PX0CBpr2RCyZhjex+NS/LPOv6YqA==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-geo": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmjs.org/d3-geo/-/d3-geo-3.1.1.tgz",
+ "integrity": "sha512-637ln3gXKXOwhalDzinUgY83KzNWZRKbYubaG+fGVuc/dxO64RRljtCTnf5ecMyE1RIdtqpkVcq0IbtU2S8j2Q==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-array": "2.5.0 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-hierarchy": {
+ "version": "3.1.2",
+ "resolved": "https://registry.npmjs.org/d3-hierarchy/-/d3-hierarchy-3.1.2.tgz",
+ "integrity": "sha512-FX/9frcub54beBdugHjDCdikxThEqjnR93Qt7PvQTOHxyiNCAlvMrHhclk3cD5VeAaq9fxmfRp+CnWw9rEMBuA==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-interpolate": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz",
+ "integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-color": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-path": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz",
+ "integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-polygon": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-polygon/-/d3-polygon-3.0.1.tgz",
+ "integrity": "sha512-3vbA7vXYwfe1SYhED++fPUQlWSYTTGmFmQiany/gdbiWgU/iEyQzyymwL9SkJjFFuCS4902BSzewVGsHHmHtXg==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-quadtree": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-quadtree/-/d3-quadtree-3.0.1.tgz",
+ "integrity": "sha512-04xDrxQTDTCFwP5H6hRhsRcb9xxv2RzkcsygFzmkSIOJy3PeRJP7sNk3VRIbKXcog561P9oU0/rVH6vDROAgUw==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-random": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-random/-/d3-random-3.0.1.tgz",
+ "integrity": "sha512-FXMe9GfxTxqd5D6jFsQ+DJ8BJS4E/fT5mqqdjovykEB2oFbTMDVdg1MGFxfQW+FBOGoB++k8swBrgwSHT1cUXQ==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-scale": {
+ "version": "4.0.2",
+ "resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz",
+ "integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-array": "2.10.0 - 3",
+ "d3-format": "1 - 3",
+ "d3-interpolate": "1.2.0 - 3",
+ "d3-time": "2.1.1 - 3",
+ "d3-time-format": "2 - 4"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-scale-chromatic": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/d3-scale-chromatic/-/d3-scale-chromatic-3.1.0.tgz",
+ "integrity": "sha512-A3s5PWiZ9YCXFye1o246KoscMWqf8BsD9eRiJ3He7C9OBaxKhAd5TFCdEx/7VbKtxxTsu//1mMJFrEt572cEyQ==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-color": "1 - 3",
+ "d3-interpolate": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-selection": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-selection/-/d3-selection-3.0.0.tgz",
+ "integrity": "sha512-fmTRWbNMmsmWq6xJV8D19U/gw/bwrHfNXxrIN+HfZgnzqTHp9jOmKMhsTUjXOJnZOdZY9Q28y4yebKzqDKlxlQ==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-shape": {
+ "version": "3.2.0",
+ "resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz",
+ "integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-path": "^3.1.0"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-time": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz",
+ "integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-array": "2 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-time-format": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz",
+ "integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-time": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-timer": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz",
+ "integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
+ "node_modules/d3-transition": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmjs.org/d3-transition/-/d3-transition-3.0.1.tgz",
+ "integrity": "sha512-ApKvfjsSR6tg06xrL434C0WydLr7JewBB3V+/39RMHsaXTOG0zmt/OAXeng5M5LBm0ojmxJrpomQVZ1aPvBL4w==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-color": "1 - 3",
+ "d3-dispatch": "1 - 3",
+ "d3-ease": "1 - 3",
+ "d3-interpolate": "1 - 3",
+ "d3-timer": "1 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ },
+ "peerDependencies": {
+ "d3-selection": "2 - 3"
+ }
+ },
+ "node_modules/d3-zoom": {
+ "version": "3.0.0",
+ "resolved": "https://registry.npmjs.org/d3-zoom/-/d3-zoom-3.0.0.tgz",
+ "integrity": "sha512-b8AmV3kfQaqWAuacbPuNbL6vahnOJflOhexLzMMNLga62+/nh0JzvJ0aO/5a5MVgUFGS7Hu1P9P03o3fJkDCyw==",
+ "license": "ISC",
+ "dependencies": {
+ "d3-dispatch": "1 - 3",
+ "d3-drag": "2 - 3",
+ "d3-interpolate": "1 - 3",
+ "d3-selection": "2 - 3",
+ "d3-transition": "2 - 3"
+ },
+ "engines": {
+ "node": ">=12"
+ }
+ },
"node_modules/damerau-levenshtein": {
"version": "1.0.8",
"resolved": "https://registry.npmjs.org/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz",
@@ -6584,6 +6995,15 @@
"url": "https://github.com/sponsors/ljharb"
}
},
+ "node_modules/delaunator": {
+ "version": "5.0.1",
+ "resolved": "https://registry.npmjs.org/delaunator/-/delaunator-5.0.1.tgz",
+ "integrity": "sha512-8nvh+XBe96aCESrGOqMp/84b13H9cdKbG5P2ejQCh4d4sK9RL4371qou9drQjMhvnPmhWl5hnmqbEE0fXr9Xnw==",
+ "license": "ISC",
+ "dependencies": {
+ "robust-predicates": "^3.0.2"
+ }
+ },
"node_modules/delayed-stream": {
"version": "1.0.0",
"resolved": "https://registry.npmjs.org/delayed-stream/-/delayed-stream-1.0.0.tgz",
@@ -9318,6 +9738,15 @@
"node": ">= 0.4"
}
},
+ "node_modules/internmap": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
+ "integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==",
+ "license": "ISC",
+ "engines": {
+ "node": ">=12"
+ }
+ },
"node_modules/ipaddr.js": {
"version": "2.2.0",
"resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-2.2.0.tgz",
@@ -14502,6 +14931,12 @@
"url": "https://github.com/sponsors/isaacs"
}
},
+ "node_modules/robust-predicates": {
+ "version": "3.0.2",
+ "resolved": "https://registry.npmjs.org/robust-predicates/-/robust-predicates-3.0.2.tgz",
+ "integrity": "sha512-IXgzBWvWQwE6PrDI05OvmXUIruQTcoMDzRsOd5CDvHCVLcLHMTSYvOK5Cm46kWqlV3yAbuSpBZdJ5oP5OUoStg==",
+ "license": "Unlicense"
+ },
"node_modules/rollup": {
"version": "2.79.2",
"resolved": "https://registry.npmjs.org/rollup/-/rollup-2.79.2.tgz",
@@ -14579,6 +15014,12 @@
"queue-microtask": "^1.2.2"
}
},
+ "node_modules/rw": {
+ "version": "1.3.3",
+ "resolved": "https://registry.npmjs.org/rw/-/rw-1.3.3.tgz",
+ "integrity": "sha512-PdhdWy89SiZogBLaw42zdeqtRJ//zFd2PgQavcICDUgJT5oW10QCRKbJ6bg4r0/UY2M6BWd5tkxuGFRvCkgfHQ==",
+ "license": "BSD-3-Clause"
+ },
"node_modules/safe-array-concat": {
"version": "1.1.3",
"resolved": "https://registry.npmjs.org/safe-array-concat/-/safe-array-concat-1.1.3.tgz",
diff --git a/package.json b/package.json
index e696477..4cfad09 100644
--- a/package.json
+++ b/package.json
@@ -10,6 +10,7 @@
"@testing-library/react": "^16.3.0",
"@testing-library/user-event": "^13.5.0",
"axios": "^0.27.2",
+ "d3": "^7.9.0",
"react": "^19.1.0",
"react-dom": "^19.1.0",
"react-icons": "^5.5.0",
diff --git a/src/components/ide/AnimationFactory.js b/src/components/ide/AnimationFactory.js
new file mode 100644
index 0000000..84a5920
--- /dev/null
+++ b/src/components/ide/AnimationFactory.js
@@ -0,0 +1,288 @@
+import React from 'react';
+
+// ๐จ ์์ ํ๋ ์ด์คํ๋ ์ปดํฌ๋ํธ (ํ์ผ์ด ์์ ๊ฒฝ์ฐ ๋๋น)
+const FallbackPlaceholder = ({ type = 'unknown', data = null, currentStep = 0, totalSteps = 0 }) => (
+
+
๐ง
+
+ {type} ์ ๋๋ฉ์ด์
๊ฐ๋ฐ ์ค
+
+
+ ๊ณง ์์ฑ๋ ์์ ์
๋๋ค!
+
+
+
+
ํ์ฌ ๋จ๊ณ
+
+ {currentStep + 1} / {totalSteps}
+
+
+
+
์ ๋๋ฉ์ด์
+
{type}
+
+
+
+);
+
+// PlaceholderAnimation import ์๋ (์คํจ์ FallbackPlaceholder ์ฌ์ฉ)
+let PlaceholderAnimation;
+try {
+ PlaceholderAnimation = require('./animations/PlaceholderAnimation').default;
+} catch (error) {
+ console.warn('PlaceholderAnimation ์ปดํฌ๋ํธ๋ฅผ ๋ถ๋ฌ์ฌ ์ ์์ต๋๋ค. Fallback์ ์ฌ์ฉํฉ๋๋ค.', error);
+ PlaceholderAnimation = FallbackPlaceholder;
+}
+
+/**
+ * ๐ญ AnimationFactory ํด๋์ค (์์ ๊ฐ๋ฐ ๋ฒ์ )
+ * ์ค์ ์ ๋๋ฉ์ด์
์ปดํฌ๋ํธ๊ฐ ์ค๋น๋๋ฉด ๊ต์ฒด๋ ์์
+ */
+export class AnimationFactory {
+ // ๐ ์ ๋๋ฉ์ด์
ํ์
โ ์ปดํฌ๋ํธ ๋งคํ ํ
์ด๋ธ (๋ชจ๋ ํ๋ ์ด์คํ๋)
+ static animations = {
+ // ๐ ์ฌ๊ท ์ ๋๋ฉ์ด์
๋ค
+ 'fibonacci-recursion': PlaceholderAnimation,
+ 'factorial-recursion': PlaceholderAnimation,
+ 'hanoi-tower': PlaceholderAnimation,
+ 'recursion-tree': PlaceholderAnimation,
+
+ // ๐ข ์ ๋ ฌ ์ ๋๋ฉ์ด์
๋ค
+ 'bubble-sort': PlaceholderAnimation,
+ 'quick-sort': PlaceholderAnimation,
+ 'merge-sort': PlaceholderAnimation,
+ 'insertion-sort': PlaceholderAnimation,
+ 'selection-sort': PlaceholderAnimation,
+ 'heap-sort': PlaceholderAnimation,
+
+ // ๐ ์๋ฃ๊ตฌ์กฐ ์ ๋๋ฉ์ด์
๋ค
+ 'array': PlaceholderAnimation,
+ 'array-manipulation': PlaceholderAnimation,
+ 'linked-list': PlaceholderAnimation,
+ 'stack': PlaceholderAnimation,
+ 'queue': PlaceholderAnimation,
+ 'tree': PlaceholderAnimation,
+ 'binary-tree': PlaceholderAnimation,
+ 'bst': PlaceholderAnimation,
+
+ // ๐ ํ์ ์ ๋๋ฉ์ด์
๋ค
+ 'binary-search': PlaceholderAnimation,
+ 'linear-search': PlaceholderAnimation,
+ 'breadth-first-search': PlaceholderAnimation,
+ 'depth-first-search': PlaceholderAnimation,
+ 'bfs': PlaceholderAnimation,
+ 'dfs': PlaceholderAnimation,
+
+ // ๐ ๋ณ์ ์ถ์ ์ ๋๋ฉ์ด์
๋ค
+ 'variables': PlaceholderAnimation,
+ 'variable-tracking': PlaceholderAnimation,
+ 'basic-algorithm': PlaceholderAnimation,
+
+ // ๐ฏ ๊ธฐ๋ณธ ์ ๋๋ฉ์ด์
(fallback)
+ 'default': PlaceholderAnimation,
+ 'unknown': PlaceholderAnimation
+ };
+
+ // ๐ ์ ๋๋ฉ์ด์
์นดํ
๊ณ ๋ฆฌ๋ณ ๋ถ๋ฅ
+ static categories = {
+ recursion: [
+ 'fibonacci-recursion', 'factorial-recursion',
+ 'hanoi-tower', 'recursion-tree'
+ ],
+ sorting: [
+ 'bubble-sort', 'quick-sort', 'merge-sort',
+ 'insertion-sort', 'selection-sort', 'heap-sort'
+ ],
+ dataStructures: [
+ 'array', 'array-manipulation', 'linked-list',
+ 'stack', 'queue', 'tree', 'binary-tree', 'bst'
+ ],
+ searching: [
+ 'binary-search', 'linear-search', 'breadth-first-search',
+ 'depth-first-search', 'bfs', 'dfs'
+ ],
+ variables: [
+ 'variables', 'variable-tracking', 'basic-algorithm'
+ ]
+ };
+
+ /**
+ * ๐จ ์ ๋๋ฉ์ด์
์ปดํฌ๋ํธ ์์ฑ
+ * @param {string} type - ์ ๋๋ฉ์ด์
ํ์
+ * @param {Object} props - ์ปดํฌ๋ํธ์ ์ ๋ฌํ props
+ * @returns {React.Component} ์์ฑ๋ ์ ๋๋ฉ์ด์
์ปดํฌ๋ํธ
+ */
+ static createAnimation(type, props = {}) {
+ console.log(`๐ญ Creating animation: ${type} (๊ฐ๋ฐ ์ค ๋ฒ์ )`);
+ console.log('Props:', props);
+
+ // 1. ์ ๋๋ฉ์ด์
ํ์
์ ๊ทํ
+ const normalizedType = this.normalizeType(type);
+ console.log('Normalized type:', normalizedType);
+
+ // 2. ์ปดํฌ๋ํธ ์ฐพ๊ธฐ (ํ์ฌ๋ ๋ชจ๋ ํ๋ ์ด์คํ๋)
+ const AnimationComponent = this.animations[normalizedType] || PlaceholderAnimation;
+ console.log('Selected component:', AnimationComponent);
+
+ // 3. ์ปดํฌ๋ํธ ์์ฑ ๋ฐ ๋ฐํ
+ const element = React.createElement(AnimationComponent, {
+ key: `animation-${normalizedType}-${Date.now()}`,
+ type: normalizedType,
+ animationType: normalizedType,
+ ...props
+ });
+
+ console.log('Created element:', element);
+ return element;
+ }
+
+ /**
+ * ๐ง ์ ๋๋ฉ์ด์
ํ์
์ ๊ทํ
+ * @param {string} type - ์๋ณธ ํ์
+ * @returns {string} ์ ๊ทํ๋ ํ์
+ */
+ static normalizeType(type) {
+ if (!type) return 'unknown';
+
+ // ์๋ฌธ์๋ก ๋ณํํ๊ณ ๊ณต๋ฐฑ/์ธ๋์ค์ฝ์ด๋ฅผ ํ์ดํ์ผ๋ก ๋ณํ
+ return type.toLowerCase()
+ .replace(/[\s_]+/g, '-')
+ .replace(/[^a-z0-9-]/g, '');
+ }
+
+ /**
+ * ๐ ์ฌ์ฉ ๊ฐ๋ฅํ ์ ๋๋ฉ์ด์
ํ์
๋ชฉ๋ก
+ * @returns {Array} ์ ๋๋ฉ์ด์
ํ์
๋ฐฐ์ด
+ */
+ static getAvailableTypes() {
+ return Object.keys(this.animations).filter(type =>
+ type !== 'unknown' && type !== 'default'
+ );
+ }
+
+ /**
+ * ๐ท๏ธ ์ ๋๋ฉ์ด์
ํ์
์ด ์ ํจํ์ง ํ์ธ
+ * @param {string} type - ํ์ธํ ํ์
+ * @returns {boolean} ์ ํจ์ฑ ์ฌ๋ถ
+ */
+ static isValidType(type) {
+ const normalizedType = this.normalizeType(type);
+ return normalizedType in this.animations;
+ }
+
+ /**
+ * ๐ ์นดํ
๊ณ ๋ฆฌ๋ณ ์ ๋๋ฉ์ด์
ํ์
์กฐํ
+ * @param {string} category - ์นดํ
๊ณ ๋ฆฌ๋ช
+ * @returns {Array} ํด๋น ์นดํ
๊ณ ๋ฆฌ์ ์ ๋๋ฉ์ด์
ํ์
๋ค
+ */
+ static getTypesByCategory(category) {
+ return this.categories[category] || [];
+ }
+
+ /**
+ * ๐ ์ ๋๋ฉ์ด์
ํ์
์ ์นดํ
๊ณ ๋ฆฌ ์ฐพ๊ธฐ
+ * @param {string} type - ์ ๋๋ฉ์ด์
ํ์
+ * @returns {string|null} ์นดํ
๊ณ ๋ฆฌ๋ช
+ */
+ static getCategoryByType(type) {
+ const normalizedType = this.normalizeType(type);
+
+ for (const [category, types] of Object.entries(this.categories)) {
+ if (types.includes(normalizedType)) {
+ return category;
+ }
+ }
+ return null;
+ }
+
+ /**
+ * ๐ฏ ์๋ก์ด ์ ๋๋ฉ์ด์
ํ์
๋ฑ๋ก (์ค์ ์ปดํฌ๋ํธ ์์ฑ ํ ์ฌ์ฉ)
+ * @param {string} type - ์ ๋๋ฉ์ด์
ํ์
+ * @param {React.Component} component - ์ปดํฌ๋ํธ
+ * @param {string} category - ์นดํ
๊ณ ๋ฆฌ (์ ํ์ฌํญ)
+ */
+ static registerAnimation(type, component, category = null) {
+ const normalizedType = this.normalizeType(type);
+
+ // ์ ๋๋ฉ์ด์
๋ฑ๋ก
+ this.animations[normalizedType] = component;
+
+ // ์นดํ
๊ณ ๋ฆฌ์ ์ถ๊ฐ (์ง์ ๋ ๊ฒฝ์ฐ)
+ if (category && this.categories[category]) {
+ if (!this.categories[category].includes(normalizedType)) {
+ this.categories[category].push(normalizedType);
+ }
+ }
+
+ console.log(`โ
Registered animation: ${normalizedType}`);
+ }
+
+ /**
+ * ๐ ํฉํ ๋ฆฌ ์ํ ์ ๋ณด
+ * @returns {Object} ํฉํ ๋ฆฌ ์ํ
+ */
+ static getFactoryInfo() {
+ return {
+ version: 'development',
+ mode: 'placeholder-only',
+ totalAnimations: Object.keys(this.animations).length,
+ categories: Object.keys(this.categories),
+ availableTypes: this.getAvailableTypes(),
+ categoriesInfo: Object.entries(this.categories).map(([name, types]) => ({
+ name,
+ count: types.length,
+ types
+ })),
+ note: '๋ชจ๋ ์ ๋๋ฉ์ด์
์ด ๊ฐ๋ฐ ์ค์
๋๋ค. ์ค์ ์ปดํฌ๋ํธ๋ ๊ณง ์ถ๊ฐ๋ ์์ ์
๋๋ค.'
+ };
+ }
+}
+
+// ๐จ ํธ์ ํจ์๋ค
+export const createAnimation = (type, props) =>
+ AnimationFactory.createAnimation(type, props);
+
+export const isValidAnimationType = (type) =>
+ AnimationFactory.isValidType(type);
+
+export const getAnimationTypes = () =>
+ AnimationFactory.getAvailableTypes();
+
+export const registerAnimation = (type, component, category) =>
+ AnimationFactory.registerAnimation(type, component, category);
+
+// ๊ธฐ๋ณธ export
+export default AnimationFactory;
+
+// ๊ฐ๋ฐ ์ํ ๋ก๊ทธ
+console.log('๐ญ AnimationFactory (๊ฐ๋ฐ ์ค) ๋ก๋๋จ:', AnimationFactory.getFactoryInfo());
\ No newline at end of file
diff --git a/src/components/ide/IDE.css b/src/components/ide/IDE.css
index 58e0a6b..ec4c905 100644
--- a/src/components/ide/IDE.css
+++ b/src/components/ide/IDE.css
@@ -726,8 +726,8 @@ body.dark-mode .visualization-button.active {
position: absolute;
top: 0;
right: 0;
- bottom: 0;
- width: 65%; /* ํ๋ฉด์ 65% ์ฐจ์ง */
+ bottom: 20px;
+ width: 90%; /* ํ๋ฉด์ 65% ์ฐจ์ง */
background-color: var(--element);
border-left: 1px solid var(--border);
display: flex;
diff --git a/src/components/ide/IDE.jsx b/src/components/ide/IDE.jsx
index b49a651..e45d5d1 100644
--- a/src/components/ide/IDE.jsx
+++ b/src/components/ide/IDE.jsx
@@ -1,6 +1,8 @@
import React, { useState, useEffect, useRef } from 'react';
import { Link, useNavigate, useLocation, useParams } from 'react-router-dom';
import Editor from '@monaco-editor/react';
+import CodeVisualizer from './VisualizationModal';
+import VisualizationModal from './VisualizationModal'; // ์๋ก ์ถ๊ฐ
import './IDE.css';
//npm install @monaco-editor/react
@@ -77,6 +79,20 @@ const applyResizeObserverFix = () => {
const IDE = () => {
// ์ปดํฌ๋ํธ ๋ง์ดํธ ์ ResizeObserver ํจ์น ์ ์ฉ
+
+ const [isVisualizationModalOpen, setIsVisualizationModalOpen] = useState(false);
+ const handleVisualizationClick = () => {
+ if (!code.trim()) {
+ alert('์๊ฐํํ ์ฝ๋๋ฅผ ๋จผ์ ์์ฑํด์ฃผ์ธ์.');
+ return;
+ }
+ setIsVisualizationModalOpen(true);
+ };
+ const handleVisualizationClose = () => {
+ setIsVisualizationModalOpen(false);
+ };
+
+
useEffect(() => {
applyResizeObserverFix();
@@ -621,7 +637,7 @@ const IDE = () => {
// ํ์ฌ ์๋ํฐ์ ๊ฐ์ ๊ฐ์ ธ์ด
const currentCode = editorRef.current.getValue();
- // API ์์ฒญ ๋ณธ๋ฌธ ์์ฑ (์ฑ๊ณตํ ํ์๊ณผ ๋์ผํ๊ฒ)
+ // API ์์ฒญ ๋ณธ๋ฌธ ์์ฑ
const requestBody = {
code: currentCode,
input: input,
@@ -648,9 +664,32 @@ const IDE = () => {
throw new Error(`API ์์ฒญ ์คํจ: ${response.status} ${response.statusText}`);
}
- // ์๋ต์ด ํ
์คํธ ํ์์ผ ๊ฒ์ผ๋ก ์์
- const result = await response.text();
- setOutput(result || "์คํ ๊ฒฐ๊ณผ๊ฐ ์์ต๋๋ค.");
+ // ๐ฅ ์์ ๋ ๋ถ๋ถ: JSON ์๋ต ํ์ฑ ํ stdout ์ถ์ถ
+ const result = await response.json(); // text() ๋์ json() ์ฌ์ฉ
+
+ console.log('API ์๋ต ๋ฐ์ดํฐ:', result); // ๋๋ฒ๊น
์ฉ ๋ก๊ทธ
+
+ // stdout ๊ฐ๋ง ์ถ์ถํด์ ์ถ๋ ฅ
+ if (result && typeof result === 'object') {
+ // stdout, Stdout, STDOUT ๋ฑ ๋ค์ํ ์ผ์ด์ค ๋์
+ const stdout = result.stdout || result.Stdout || result.STDOUT ||
+ result.output || result.Output || result.OUTPUT;
+
+ if (stdout !== undefined) {
+ setOutput(stdout || "์คํ ์๋ฃ (์ถ๋ ฅ ์์)");
+ } else {
+ // stdout์ด ์๋ ๊ฒฝ์ฐ ์ ์ฒด ์๋ต์ ๋ณด์ฌ์ฃผ๋, ์๋ฌ ์ ๋ณด ์ฐ์
+ const errorMsg = result.stderr || result.error || result.message;
+ if (errorMsg) {
+ setOutput(`์ค๋ฅ: ${errorMsg}`);
+ } else {
+ setOutput("์คํ ์๋ฃ๋์์ง๋ง ์ถ๋ ฅ์ด ์์ต๋๋ค.");
+ }
+ }
+ } else {
+ // ์๋ต์ด ๊ฐ์ฒด๊ฐ ์๋ ๊ฒฝ์ฐ (๋ฌธ์์ด ๋ฑ)
+ setOutput(result || "์คํ ๊ฒฐ๊ณผ๊ฐ ์์ต๋๋ค.");
+ }
} catch (error) {
console.error('์ฝ๋ ์คํ ์ค ์ค๋ฅ:', error);
@@ -953,8 +992,9 @@ const IDE = () => {
์คํ ์ฝ๋