From 1210d14d9b3390c9f7c4ed7e04bbdf1db4a02883 Mon Sep 17 00:00:00 2001 From: Jerens Lensun <54782057+jerensl@users.noreply.github.com> Date: Sat, 21 Sep 2024 15:38:38 +0800 Subject: [PATCH] feat(website): add stretchable layout for playground (#2067) Signed-off-by: jerensl <54782057+jerensl@users.noreply.github.com> Co-authored-by: Ashmit JaiSarita Gupta <43639341+devilkiller-ag@users.noreply.github.com> Co-authored-by: asyncapi-bot Co-authored-by: Jonas Lagoni --- modelina-website/next.config.js | 7 +- modelina-website/package-lock.json | 333 +++++++++++++++++- modelina-website/package.json | 4 + .../src/components/MonacoEditorWrapper.tsx | 2 + .../components/contexts/PlaygroundContext.tsx | 21 -- .../contexts/PlaygroundLayoutContext.tsx | 89 +++++ .../src/components/playground/Content.tsx | 93 ++--- .../src/components/playground/Resizable.tsx | 78 ++++ .../src/components/playground/Sidebar.tsx | 79 +---- modelina-website/src/pages/playground.tsx | 5 +- modelina-website/src/store/useLayoutStore.tsx | 153 ++++++++ modelina-website/tsconfig.json | 8 +- 12 files changed, 726 insertions(+), 146 deletions(-) create mode 100644 modelina-website/src/components/contexts/PlaygroundLayoutContext.tsx create mode 100644 modelina-website/src/components/playground/Resizable.tsx create mode 100644 modelina-website/src/store/useLayoutStore.tsx diff --git a/modelina-website/next.config.js b/modelina-website/next.config.js index 4b0135c5da..c396f7e616 100644 --- a/modelina-website/next.config.js +++ b/modelina-website/next.config.js @@ -1,5 +1,8 @@ const isGithubActions = process.env.GITHUB_ACTIONS || false; - +const withBundleAnalyzer = require('@next/bundle-analyzer')({ + enabled: process.env.ANALYZE === 'true', +}) + let assetPrefix = '/'; let basePath = ''; @@ -20,4 +23,4 @@ const nextConfig = { } }; -module.exports = nextConfig; +module.exports = withBundleAnalyzer(nextConfig); diff --git a/modelina-website/package-lock.json b/modelina-website/package-lock.json index c7f53cc436..d45625106d 100644 --- a/modelina-website/package-lock.json +++ b/modelina-website/package-lock.json @@ -8,13 +8,17 @@ "dependencies": { "@monaco-editor/react": "^4.6.0", "@netlify/functions": "^1.4.0", + "@next/bundle-analyzer": "^14.2.5", "@tailwindcss/aspect-ratio": "^0.4.0", "@tailwindcss/forms": "^0.5.2", "@tailwindcss/line-clamp": "^0.4.0", "@tailwindcss/typography": "^0.5.9", "@tippyjs/react": "^4.2.6", + "@uidotdev/usehooks": "^2.4.1", "clsx": "^2.1.1", "cssnano": "^5.1.14", + "framer-motion": "^11.2.12", + "immer": "^10.1.1", "js-base64": "^3.7.4", "lodash": "^4.17.21", "lowlight": "^1.12.1", @@ -90,6 +94,14 @@ "node": ">=6.9.0" } }, + "node_modules/@discoveryjs/json-ext": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", + "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==", + "engines": { + "node": ">=10.0.0" + } + }, "node_modules/@eslint-community/eslint-utils": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", @@ -318,6 +330,14 @@ "node": ">=8.3.0" } }, + "node_modules/@next/bundle-analyzer": { + "version": "14.2.5", + "resolved": "https://registry.npmjs.org/@next/bundle-analyzer/-/bundle-analyzer-14.2.5.tgz", + "integrity": "sha512-BtBbI8VUnB7s4m9ut6CkeJ8Hyx+aq+86mbH+uAld7ZGG0/eH4+5hcPnkHKsQM/yj74iClazS0fninI8yZbIZWA==", + "dependencies": { + "webpack-bundle-analyzer": "4.10.1" + } + }, "node_modules/@next/env": { "version": "13.4.3", "resolved": "https://registry.npmjs.org/@next/env/-/env-13.4.3.tgz", @@ -567,6 +587,11 @@ "url": "https://opencollective.com/unts" } }, + "node_modules/@polka/url": { + "version": "1.0.0-next.25", + "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.25.tgz", + "integrity": "sha512-j7P6Rgr3mmtdkeDGTe0E/aYyWEWVtc5yFXtHCRHs28/jptDEWfaVOc5T7cblqy1XKPPfCxJc/8DwQ5YgLOZOVQ==" + }, "node_modules/@popperjs/core": { "version": "2.11.8", "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz", @@ -978,6 +1003,18 @@ "url": "https://opencollective.com/typescript-eslint" } }, + "node_modules/@uidotdev/usehooks": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/@uidotdev/usehooks/-/usehooks-2.4.1.tgz", + "integrity": "sha512-1I+RwWyS+kdv3Mv0Vmc+p0dPYH0DTRAo04HLyXReYBL9AeseDWUJyi4THuksBJcu9F0Pih69Ak150VDnqbVnXg==", + "engines": { + "node": ">=16" + }, + "peerDependencies": { + "react": ">=18.0.0", + "react-dom": ">=18.0.0" + } + }, "node_modules/@ungap/structured-clone": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", @@ -988,7 +1025,6 @@ "version": "8.12.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.0.tgz", "integrity": "sha512-RTvkC4w+KNXrM39/lWCUaG0IbRkWdCv7W/IOW9oU6SawyxulvkQy5HQPVTKxEjczcUvapcrw3cFx/60VN/NRNw==", - "dev": true, "bin": { "acorn": "bin/acorn" }, @@ -1005,6 +1041,17 @@ "acorn": "^6.0.0 || ^7.0.0 || ^8.0.0" } }, + "node_modules/acorn-walk": { + "version": "8.3.3", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.3.tgz", + "integrity": "sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==", + "dependencies": { + "acorn": "^8.11.0" + }, + "engines": { + "node": ">=0.4.0" + } + }, "node_modules/ajv": { "version": "6.12.6", "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", @@ -1875,6 +1922,11 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/debounce": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/debounce/-/debounce-1.2.1.tgz", + "integrity": "sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug==" + }, "node_modules/debug": { "version": "4.3.4", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", @@ -2061,6 +2113,11 @@ "url": "https://github.com/fb55/domutils?sponsor=1" } }, + "node_modules/duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==" + }, "node_modules/eastasianwidth": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", @@ -2269,7 +2326,6 @@ "version": "4.0.0", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true, "engines": { "node": ">=10" }, @@ -3033,6 +3089,30 @@ "url": "https://github.com/sponsors/rawify" } }, + "node_modules/framer-motion": { + "version": "11.2.12", + "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-11.2.12.tgz", + "integrity": "sha512-lCjkV4nA9rWOy2bhR4RZzkp2xpB++kFmUZ6D44V9VQaxk+JDmbDd5lq+u58DjJIIllE8AZEXp9OG/TyDN4FB/w==", + "dependencies": { + "tslib": "^2.4.0" + }, + "peerDependencies": { + "@emotion/is-prop-valid": "*", + "react": "^18.0.0", + "react-dom": "^18.0.0" + }, + "peerDependenciesMeta": { + "@emotion/is-prop-valid": { + "optional": true + }, + "react": { + "optional": true + }, + "react-dom": { + "optional": true + } + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -3241,6 +3321,20 @@ "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", "dev": true }, + "node_modules/gzip-size": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-6.0.0.tgz", + "integrity": "sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==", + "dependencies": { + "duplexer": "^0.1.2" + }, + "engines": { + "node": ">=10" + }, + "funding": { + "url": "https://github.com/sponsors/sindresorhus" + } + }, "node_modules/has-bigints": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", @@ -3511,6 +3605,11 @@ "node": "*" } }, + "node_modules/html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==" + }, "node_modules/html-void-elements": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/html-void-elements/-/html-void-elements-2.0.1.tgz", @@ -3529,6 +3628,15 @@ "node": ">= 4" } }, + "node_modules/immer": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/immer/-/immer-10.1.1.tgz", + "integrity": "sha512-s2MPrmjovJcoMaHtx6K11Ra7oD05NT97w1IC5zpMkT6Atjr7H8LjaDd81iIxUYpMKSRRNMJE703M1Fhr/TctHw==", + "funding": { + "type": "opencollective", + "url": "https://opencollective.com/immer" + } + }, "node_modules/import-fresh": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", @@ -3891,6 +3999,14 @@ "url": "https://github.com/sponsors/sindresorhus" } }, + "node_modules/is-plain-object": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", + "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==", + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-promise": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", @@ -5115,6 +5231,14 @@ "node": ">=4" } }, + "node_modules/mrmime": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.0.tgz", + "integrity": "sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==", + "engines": { + "node": ">=10" + } + }, "node_modules/ms": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", @@ -5413,6 +5537,14 @@ "wrappy": "1" } }, + "node_modules/opener": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/opener/-/opener-1.5.2.tgz", + "integrity": "sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==", + "bin": { + "opener": "bin/opener-bin.js" + } + }, "node_modules/optionator": { "version": "0.9.3", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", @@ -6678,6 +6810,19 @@ "url": "https://github.com/sponsors/isaacs" } }, + "node_modules/sirv": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/sirv/-/sirv-2.0.4.tgz", + "integrity": "sha512-94Bdh3cC2PKrbgSOUqTiGPWVZeSiXfKOVZNJniWoqrWrRkB1CJzBU3NEbiTsPcYy1lDsANA/THzS+9WBiy5nfQ==", + "dependencies": { + "@polka/url": "^1.0.0-next.24", + "mrmime": "^2.0.0", + "totalist": "^3.0.0" + }, + "engines": { + "node": ">= 10" + } + }, "node_modules/slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -7207,6 +7352,14 @@ "node": ">=8.0" } }, + "node_modules/totalist": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz", + "integrity": "sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==", + "engines": { + "node": ">=6" + } + }, "node_modules/trim-lines": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/trim-lines/-/trim-lines-3.0.1.tgz", @@ -7584,6 +7737,32 @@ "url": "https://github.com/sponsors/wooorm" } }, + "node_modules/webpack-bundle-analyzer": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-4.10.1.tgz", + "integrity": "sha512-s3P7pgexgT/HTUSYgxJyn28A+99mmLq4HsJepMPzu0R8ImJc52QNqaFYW1Z2z2uIb1/J3eYgaAWVpaC+v/1aAQ==", + "dependencies": { + "@discoveryjs/json-ext": "0.5.7", + "acorn": "^8.0.4", + "acorn-walk": "^8.0.0", + "commander": "^7.2.0", + "debounce": "^1.2.1", + "escape-string-regexp": "^4.0.0", + "gzip-size": "^6.0.0", + "html-escaper": "^2.0.2", + "is-plain-object": "^5.0.0", + "opener": "^1.5.2", + "picocolors": "^1.0.0", + "sirv": "^2.0.3", + "ws": "^7.3.1" + }, + "bin": { + "webpack-bundle-analyzer": "lib/bin/analyzer.js" + }, + "engines": { + "node": ">= 10.13.0" + } + }, "node_modules/which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -7774,6 +7953,26 @@ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" }, + "node_modules/ws": { + "version": "7.5.10", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.10.tgz", + "integrity": "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==", + "engines": { + "node": ">=8.3.0" + }, + "peerDependencies": { + "bufferutil": "^4.0.1", + "utf-8-validate": "^5.0.2" + }, + "peerDependenciesMeta": { + "bufferutil": { + "optional": true + }, + "utf-8-validate": { + "optional": true + } + } + }, "node_modules/xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", @@ -7840,6 +8039,11 @@ "regenerator-runtime": "^0.14.0" } }, + "@discoveryjs/json-ext": { + "version": "0.5.7", + "resolved": "https://registry.npmjs.org/@discoveryjs/json-ext/-/json-ext-0.5.7.tgz", + "integrity": "sha512-dBVuXR082gk3jsFp7Rd/JI4kytwGHecnCoTtXFb7DB6CNHp4rg5k1bhg0nWdLGLnOV71lmDzGQaLMy8iPLY0pw==" + }, "@eslint-community/eslint-utils": { "version": "4.4.0", "resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.4.0.tgz", @@ -8001,6 +8205,14 @@ "is-promise": "^4.0.0" } }, + "@next/bundle-analyzer": { + "version": "14.2.5", + "resolved": "https://registry.npmjs.org/@next/bundle-analyzer/-/bundle-analyzer-14.2.5.tgz", + "integrity": "sha512-BtBbI8VUnB7s4m9ut6CkeJ8Hyx+aq+86mbH+uAld7ZGG0/eH4+5hcPnkHKsQM/yj74iClazS0fninI8yZbIZWA==", + "requires": { + "webpack-bundle-analyzer": "4.10.1" + } + }, "@next/env": { "version": "13.4.3", "resolved": "https://registry.npmjs.org/@next/env/-/env-13.4.3.tgz", @@ -8138,6 +8350,11 @@ "integrity": "sha512-cq8o4cWH0ibXh9VGi5P20Tu9XF/0fFXl9EUinr9QfTM7a7p0oTA4iJRCQWppXR1Pg8dSM0UCItCkPwsk9qWWYA==", "dev": true }, + "@polka/url": { + "version": "1.0.0-next.25", + "resolved": "https://registry.npmjs.org/@polka/url/-/url-1.0.0-next.25.tgz", + "integrity": "sha512-j7P6Rgr3mmtdkeDGTe0E/aYyWEWVtc5yFXtHCRHs28/jptDEWfaVOc5T7cblqy1XKPPfCxJc/8DwQ5YgLOZOVQ==" + }, "@popperjs/core": { "version": "2.11.8", "resolved": "https://registry.npmjs.org/@popperjs/core/-/core-2.11.8.tgz", @@ -8435,6 +8652,12 @@ "eslint-visitor-keys": "^3.4.1" } }, + "@uidotdev/usehooks": { + "version": "2.4.1", + "resolved": "https://registry.npmjs.org/@uidotdev/usehooks/-/usehooks-2.4.1.tgz", + "integrity": "sha512-1I+RwWyS+kdv3Mv0Vmc+p0dPYH0DTRAo04HLyXReYBL9AeseDWUJyi4THuksBJcu9F0Pih69Ak150VDnqbVnXg==", + "requires": {} + }, "@ungap/structured-clone": { "version": "1.2.0", "resolved": "https://registry.npmjs.org/@ungap/structured-clone/-/structured-clone-1.2.0.tgz", @@ -8444,8 +8667,7 @@ "acorn": { "version": "8.12.0", "resolved": "https://registry.npmjs.org/acorn/-/acorn-8.12.0.tgz", - "integrity": "sha512-RTvkC4w+KNXrM39/lWCUaG0IbRkWdCv7W/IOW9oU6SawyxulvkQy5HQPVTKxEjczcUvapcrw3cFx/60VN/NRNw==", - "dev": true + "integrity": "sha512-RTvkC4w+KNXrM39/lWCUaG0IbRkWdCv7W/IOW9oU6SawyxulvkQy5HQPVTKxEjczcUvapcrw3cFx/60VN/NRNw==" }, "acorn-jsx": { "version": "5.3.2", @@ -8454,6 +8676,14 @@ "dev": true, "requires": {} }, + "acorn-walk": { + "version": "8.3.3", + "resolved": "https://registry.npmjs.org/acorn-walk/-/acorn-walk-8.3.3.tgz", + "integrity": "sha512-MxXdReSRhGO7VlFe1bRG/oI7/mdLV9B9JJT0N8vZOhF7gFRR5l3M8W9G8JxmKV+JC5mGqJ0QvqfSOLsCPa4nUw==", + "requires": { + "acorn": "^8.11.0" + } + }, "ajv": { "version": "6.12.6", "resolved": "https://registry.npmjs.org/ajv/-/ajv-6.12.6.tgz", @@ -9045,6 +9275,11 @@ "is-data-view": "^1.0.1" } }, + "debounce": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/debounce/-/debounce-1.2.1.tgz", + "integrity": "sha512-XRRe6Glud4rd/ZGQfiV1ruXSfbvfJedlV9Y6zOlP+2K04vBYiJEte6stfFkCP03aMnY5tsipamumUjL14fofug==" + }, "debug": { "version": "4.3.4", "resolved": "https://registry.npmjs.org/debug/-/debug-4.3.4.tgz", @@ -9175,6 +9410,11 @@ "domhandler": "^4.2.0" } }, + "duplexer": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/duplexer/-/duplexer-0.1.2.tgz", + "integrity": "sha512-jtD6YG370ZCIi/9GTaJKQxWTZD045+4R4hTk/x1UyoqadyJ9x9CgSi1RlVDQF8U2sxLLSnFkCaMihqljHIWgMg==" + }, "eastasianwidth": { "version": "0.2.0", "resolved": "https://registry.npmjs.org/eastasianwidth/-/eastasianwidth-0.2.0.tgz", @@ -9346,8 +9586,7 @@ "escape-string-regexp": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz", - "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==", - "dev": true + "integrity": "sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA==" }, "eslint": { "version": "8.57.0", @@ -9898,6 +10137,14 @@ "integrity": "sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==", "dev": true }, + "framer-motion": { + "version": "11.2.12", + "resolved": "https://registry.npmjs.org/framer-motion/-/framer-motion-11.2.12.tgz", + "integrity": "sha512-lCjkV4nA9rWOy2bhR4RZzkp2xpB++kFmUZ6D44V9VQaxk+JDmbDd5lq+u58DjJIIllE8AZEXp9OG/TyDN4FB/w==", + "requires": { + "tslib": "^2.4.0" + } + }, "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -10042,6 +10289,14 @@ "integrity": "sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag==", "dev": true }, + "gzip-size": { + "version": "6.0.0", + "resolved": "https://registry.npmjs.org/gzip-size/-/gzip-size-6.0.0.tgz", + "integrity": "sha512-ax7ZYomf6jqPTQ4+XCpUGyXKHk5WweS+e05MBO4/y3WJ5RkmPXNKvX+bx1behVILVwr6JSQvZAku021CHPXG3Q==", + "requires": { + "duplexer": "^0.1.2" + } + }, "has-bigints": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/has-bigints/-/has-bigints-1.0.2.tgz", @@ -10227,6 +10482,11 @@ "resolved": "https://registry.npmjs.org/highlight.js/-/highlight.js-10.7.3.tgz", "integrity": "sha512-tzcUFauisWKNHaRkN4Wjl/ZA07gENAjFl3J/c480dprkGTg5EQstgaNFqBfUqCq54kZRIEcreTsAgF/m2quD7A==" }, + "html-escaper": { + "version": "2.0.2", + "resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz", + "integrity": "sha512-H2iMtd0I4Mt5eYiapRdIDjp+XzelXQ0tFE4JS7YFwFevXXMmOp9myNrUvCg0D6ws8iqkRPBfKHgbwig1SmlLfg==" + }, "html-void-elements": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/html-void-elements/-/html-void-elements-2.0.1.tgz", @@ -10238,6 +10498,11 @@ "integrity": "sha512-5Fytz/IraMjqpwfd34ke28PTVMjZjJG2MPn5t7OE4eUCUNf8BAa7b5WUS9/Qvr6mwOQS7Mk6vdsMno5he+T8Xw==", "dev": true }, + "immer": { + "version": "10.1.1", + "resolved": "https://registry.npmjs.org/immer/-/immer-10.1.1.tgz", + "integrity": "sha512-s2MPrmjovJcoMaHtx6K11Ra7oD05NT97w1IC5zpMkT6Atjr7H8LjaDd81iIxUYpMKSRRNMJE703M1Fhr/TctHw==" + }, "import-fresh": { "version": "3.3.0", "resolved": "https://registry.npmjs.org/import-fresh/-/import-fresh-3.3.0.tgz", @@ -10465,6 +10730,11 @@ "resolved": "https://registry.npmjs.org/is-plain-obj/-/is-plain-obj-4.1.0.tgz", "integrity": "sha512-+Pgi+vMuUNkJyExiMBt5IlFoMyKnr5zhJ4Uspz58WOhBF5QoIZkFyNHIbBAtHwzVAgk5RtndVNsDRN61/mmDqg==" }, + "is-plain-object": { + "version": "5.0.0", + "resolved": "https://registry.npmjs.org/is-plain-object/-/is-plain-object-5.0.0.tgz", + "integrity": "sha512-VRSzKkbMm5jMDoKLbltAkFQ5Qr7VDiTFGXxYFXXowVj387GeGNOCsOH6Msy00SGZ3Fp84b1Naa1psqgcCIEP5Q==" + }, "is-promise": { "version": "4.0.0", "resolved": "https://registry.npmjs.org/is-promise/-/is-promise-4.0.0.tgz", @@ -11276,6 +11546,11 @@ "resolved": "https://registry.npmjs.org/mri/-/mri-1.2.0.tgz", "integrity": "sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==" }, + "mrmime": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/mrmime/-/mrmime-2.0.0.tgz", + "integrity": "sha512-eu38+hdgojoyq63s+yTpN4XMBdt5l8HhMhc4VKLO9KM5caLIBvUm4thi7fFaxyTmCKeNnXZ5pAlBwCUnhA09uw==" + }, "ms": { "version": "2.1.2", "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.2.tgz", @@ -11464,6 +11739,11 @@ "wrappy": "1" } }, + "opener": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/opener/-/opener-1.5.2.tgz", + "integrity": "sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A==" + }, "optionator": { "version": "0.9.3", "resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.3.tgz", @@ -12276,6 +12556,16 @@ "integrity": "sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw==", "dev": true }, + "sirv": { + "version": "2.0.4", + "resolved": "https://registry.npmjs.org/sirv/-/sirv-2.0.4.tgz", + "integrity": "sha512-94Bdh3cC2PKrbgSOUqTiGPWVZeSiXfKOVZNJniWoqrWrRkB1CJzBU3NEbiTsPcYy1lDsANA/THzS+9WBiy5nfQ==", + "requires": { + "@polka/url": "^1.0.0-next.24", + "mrmime": "^2.0.0", + "totalist": "^3.0.0" + } + }, "slash": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/slash/-/slash-3.0.0.tgz", @@ -12643,6 +12933,11 @@ "is-number": "^7.0.0" } }, + "totalist": { + "version": "3.0.1", + "resolved": "https://registry.npmjs.org/totalist/-/totalist-3.0.1.tgz", + "integrity": "sha512-sf4i37nQ2LBx4m3wB74y+ubopq6W/dIzXg0FDGjsYnZHVa1Da8FH853wlL2gtUhg+xJXjfk3kUZS3BRoQeoQBQ==" + }, "trim-lines": { "version": "3.0.1", "resolved": "https://registry.npmjs.org/trim-lines/-/trim-lines-3.0.1.tgz", @@ -12897,6 +13192,26 @@ "resolved": "https://registry.npmjs.org/web-namespaces/-/web-namespaces-2.0.1.tgz", "integrity": "sha512-bKr1DkiNa2krS7qxNtdrtHAmzuYGFQLiQ13TsorsdT6ULTkPLKuu5+GsFpDlg6JFjUTwX2DyhMPG2be8uPrqsQ==" }, + "webpack-bundle-analyzer": { + "version": "4.10.1", + "resolved": "https://registry.npmjs.org/webpack-bundle-analyzer/-/webpack-bundle-analyzer-4.10.1.tgz", + "integrity": "sha512-s3P7pgexgT/HTUSYgxJyn28A+99mmLq4HsJepMPzu0R8ImJc52QNqaFYW1Z2z2uIb1/J3eYgaAWVpaC+v/1aAQ==", + "requires": { + "@discoveryjs/json-ext": "0.5.7", + "acorn": "^8.0.4", + "acorn-walk": "^8.0.0", + "commander": "^7.2.0", + "debounce": "^1.2.1", + "escape-string-regexp": "^4.0.0", + "gzip-size": "^6.0.0", + "html-escaper": "^2.0.2", + "is-plain-object": "^5.0.0", + "opener": "^1.5.2", + "picocolors": "^1.0.0", + "sirv": "^2.0.3", + "ws": "^7.3.1" + } + }, "which": { "version": "2.0.2", "resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz", @@ -13033,6 +13348,12 @@ "resolved": "https://registry.npmjs.org/wrappy/-/wrappy-1.0.2.tgz", "integrity": "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ==" }, + "ws": { + "version": "7.5.10", + "resolved": "https://registry.npmjs.org/ws/-/ws-7.5.10.tgz", + "integrity": "sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ==", + "requires": {} + }, "xtend": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/xtend/-/xtend-4.0.2.tgz", diff --git a/modelina-website/package.json b/modelina-website/package.json index 6e2f6b0320..c75e23f091 100644 --- a/modelina-website/package.json +++ b/modelina-website/package.json @@ -17,13 +17,17 @@ "dependencies": { "@monaco-editor/react": "^4.6.0", "@netlify/functions": "^1.4.0", + "@next/bundle-analyzer": "^14.2.5", "@tailwindcss/aspect-ratio": "^0.4.0", "@tailwindcss/forms": "^0.5.2", "@tailwindcss/line-clamp": "^0.4.0", "@tailwindcss/typography": "^0.5.9", "@tippyjs/react": "^4.2.6", + "@uidotdev/usehooks": "^2.4.1", "clsx": "^2.1.1", "cssnano": "^5.1.14", + "framer-motion": "^11.2.12", + "immer": "^10.1.1", "js-base64": "^3.7.4", "lodash": "^4.17.21", "lowlight": "^1.12.1", diff --git a/modelina-website/src/components/MonacoEditorWrapper.tsx b/modelina-website/src/components/MonacoEditorWrapper.tsx index d5f0ff8ba5..f3295d4df1 100644 --- a/modelina-website/src/components/MonacoEditorWrapper.tsx +++ b/modelina-website/src/components/MonacoEditorWrapper.tsx @@ -59,6 +59,8 @@ export default function MonacoEditorWrapper({ return ( >; - showOptions: boolean; - setShowOptions: Dispatch>; - showOutputNavigation: boolean; - setShowOutputNavigation: Dispatch>; config: ModelinaOptions; setConfig: Dispatch>; input: string; @@ -95,9 +89,6 @@ export const PlaygroundContextProvider: React.FC<{ kotlinPackageName: 'asyncapi.models' }; - const [showInputEditor, setShowInputEditor] = useState(true); - const [showOptions, setShowOptions] = useState(true); - const [showOutputNavigation, setShowOutputNavigation] = useState(true); const [config, setConfig] = useState(defaultConfig); const [input, setInput] = useState(JSON.stringify(defaultAsyncapiDocument, null, 4)); const [models, setModels] = useState([]); @@ -116,12 +107,6 @@ export const PlaygroundContextProvider: React.FC<{ const contextValue = useMemo( () => ({ - showInputEditor, - setShowInputEditor, - showOptions, - setShowOptions, - showOutputNavigation, - setShowOutputNavigation, config, setConfig, input, @@ -148,12 +133,6 @@ export const PlaygroundContextProvider: React.FC<{ setRenderModels }), [ - showInputEditor, - setShowInputEditor, - showOptions, - setShowOptions, - showOutputNavigation, - setShowOutputNavigation, config, setConfig, input, diff --git a/modelina-website/src/components/contexts/PlaygroundLayoutContext.tsx b/modelina-website/src/components/contexts/PlaygroundLayoutContext.tsx new file mode 100644 index 0000000000..17f1a5b873 --- /dev/null +++ b/modelina-website/src/components/contexts/PlaygroundLayoutContext.tsx @@ -0,0 +1,89 @@ +'use client'; + +import { useMeasure } from '@uidotdev/usehooks'; +import { createContext, useContext, useEffect, useReducer } from 'react'; + +import type { Dispatch, State } from '@/store/useLayoutStore'; +import { initialState, playgroundLayoutReducer } from '@/store/useLayoutStore'; + +const LocalStorageKey = 'PlaygroundLayout'; + +type PlaygroundtProviderProps = { children: React.ReactNode }; + +const PlaygroundtLayoutContext = createContext< + { state: State; dispatch: Dispatch } | undefined +>(undefined); + +const localStorageInitializer = (initialValue = initialState) => { + if (typeof window !== 'undefined') { + const resizablePercentageOpen = localStorage.getItem(LocalStorageKey); + + if (resizablePercentageOpen !== null) { + const size = JSON.parse(resizablePercentageOpen); + const newValue = { + ...initialValue, + editorSize: parseFloat(size.editorSize) + }; + + return newValue; + } + } + + return initialValue; +}; + +/** + * This component to consume Plaground Layout State. + * + * @returns {ReactNode} A React element that return state and update the state. + */ +function usePlaygroundLayout() { + const context = useContext(PlaygroundtLayoutContext); + + if (context === undefined) { + throw new Error('usePlaygroundLayout must be used within a PlaygroundLayoutProvider'); + } + + return context; +} + +/** + * This component to providing Plaground Layout State. + * + * @param {ReactNode} children The children element of the component. + * @returns {ReactNode} A React element that share context state. + */ +function PlaygroundLayoutProvider({ children }: PlaygroundtProviderProps) { + const [state, dispatch] = useReducer(playgroundLayoutReducer, initialState, localStorageInitializer); + const value = { state, dispatch }; + + const [ref, { width }] = useMeasure(); + + useEffect(() => { + localStorage.setItem(LocalStorageKey, JSON.stringify({ editorSize: state.editorSize })); + }, [state.editorSize]); + + useEffect(() => { + if (width !== null) { + if (width < 768) { + dispatch({ type: 'update-device', payload: 'mobile' }); + } else if (width <= 1024) { + dispatch({ type: 'update-device', payload: 'tablet' }); + } else if (width <= 1280) { + dispatch({ type: 'update-device', payload: 'notebook' }); + } else { + dispatch({ type: 'update-device', payload: 'desktop' }); + } + } + }, [width]); + + return ( + +
+ {children} +
+
+); +} + +export { PlaygroundLayoutProvider, usePlaygroundLayout }; diff --git a/modelina-website/src/components/playground/Content.tsx b/modelina-website/src/components/playground/Content.tsx index c9c584a2d8..e37cfa773c 100644 --- a/modelina-website/src/components/playground/Content.tsx +++ b/modelina-website/src/components/playground/Content.tsx @@ -6,11 +6,13 @@ import { useMemo } from 'react'; import { usePlaygroundContext } from '../contexts/PlaygroundContext'; import { PlaygroundGeneratedContext } from '../contexts/PlaygroundGeneratedContext'; +import { usePlaygroundLayout } from '../contexts/PlaygroundLayoutContext'; import CustomError from '../CustomError'; import MonacoEditorWrapper from '../MonacoEditorWrapper'; import GeneratedModelsComponent from './GeneratedModels'; import { OptionsNavigation } from './OptionsNavigation'; import OutputNavigation from './OutputNavigation'; +import Resizable from './Resizable'; interface ContentProps { setNewConfig: (config: string, configValue: any, updateCode?: boolean) => void; @@ -22,17 +24,20 @@ export const Content: FunctionComponent = ({ setNewConfig, setNewQ const { config, input, - showInputEditor, setInput, models, loaded, setLoaded, error, statusCode, - errorMessage, - showOptions, - showOutputNavigation + errorMessage } = usePlaygroundContext(); + const { state } = usePlaygroundLayout(); + + const hasInputOptionsOpen = state.sidebarItems.get('general-options')?.isOpen; + const hasOutputOptionsOpen = state.sidebarItems.get('output-options')?.isOpen; + const hasInputEditorOpen = state.sidebarItems.get('input-editor')?.isOpen; + const hasOutputEditorOpen = state.sidebarItems.get('output-editor')?.isOpen; const PlaygroundGeneratedContextValue = useMemo( () => ({ @@ -43,57 +48,61 @@ export const Content: FunctionComponent = ({ setNewConfig, setNewQ ); return ( -
+
-
- { - setInput(change); - generateNewCode(change); - }} - editorDidMount={() => { - setLoaded({ ...loaded, editorLoaded: true }); - }} - language='json' - /> -
-
-
- {error ? ( - - ) : ( - - - - )} + + { + setInput(change); + generateNewCode(change); + }} + editorDidMount={() => { + setLoaded({ ...loaded, editorLoaded: true }); + }} + language='json' + /> +
+ } + rightComponent={ +
+ {error ? ( + + ) : ( + + + + )} +
+ } /> +
); diff --git a/modelina-website/src/components/playground/Resizable.tsx b/modelina-website/src/components/playground/Resizable.tsx new file mode 100644 index 0000000000..7ddce51994 --- /dev/null +++ b/modelina-website/src/components/playground/Resizable.tsx @@ -0,0 +1,78 @@ + +import { useMeasure } from '@uidotdev/usehooks'; +import { motion, useMotionValue, useTransform } from 'framer-motion'; +import { memo, type ReactNode, useEffect } from 'react'; + +import { usePlaygroundLayout } from '../contexts/PlaygroundLayoutContext'; + +interface ResizableComponentProps { + leftComponent?: ReactNode; + rightComponent?: ReactNode; +} + +/** + * @description This is the resizable component. + * @type {React.FC} Props + * @property {React.ReactElement} leftComponent The left element which will be stretch. + * @property {React.ReactElement} rightComponent The right element which will be stretch. + */ +function Resizable({ leftComponent, rightComponent }: ResizableComponentProps) { + const { state, dispatch } = usePlaygroundLayout(); + + const [ref, { width: containerWidth }] = useMeasure(); + const DefaultWidth = 640; + + const dragableX = useMotionValue(DefaultWidth); + const width = useTransform(dragableX, (value) => { + if (containerWidth !== null) { + const visibleView = value / containerWidth; + + dispatch({ type: 'resizable-size', total: parseFloat(visibleView.toFixed(2)) }); + } + + return `${value + 0.5 * 4}px`; + }); + + useEffect(() => { + if (containerWidth !== null) { + dragableX.set(Math.round(containerWidth * state.editorSize)); + } + }, [containerWidth]); + + if (state.device === 'mobile') { + return
+ {leftComponent} + {rightComponent} +
; + } + + return ( +
+ + {leftComponent} + +
+ ); +} + +Resizable.displayName = 'Resizable Component'; + +export default memo(Resizable); diff --git a/modelina-website/src/components/playground/Sidebar.tsx b/modelina-website/src/components/playground/Sidebar.tsx index b4762bc312..0acb7c4c11 100644 --- a/modelina-website/src/components/playground/Sidebar.tsx +++ b/modelina-website/src/components/playground/Sidebar.tsx @@ -1,71 +1,15 @@ import clsx from 'clsx'; -import React from 'react'; -import { IoOptionsOutline } from 'react-icons/io5'; -import { LuFileInput, LuFileOutput } from 'react-icons/lu'; -import { VscListSelection } from 'react-icons/vsc'; +import React, { useCallback } from 'react'; -import { usePlaygroundContext } from '../contexts/PlaygroundContext'; +import { usePlaygroundLayout } from '../contexts/PlaygroundLayoutContext'; import { Tooltip } from './Tooltip'; -interface SidebarItem { - name: string; - title: string; - isActive: boolean; - isShow: boolean; - mobileOnly: boolean; - onClick: () => void; - icon: React.ReactNode; - tooltip: React.ReactNode; -} +export const Sidebar: React.FunctionComponent = () => { + const { state, dispatch } = usePlaygroundLayout(); -interface SidebarProps {} + const sidebarItems = Array.from(state.sidebarItems.values()); -export const Sidebar: React.FunctionComponent = () => { - const { setShowOptions, setShowOutputNavigation, setShowInputEditor, showInputEditor, showOptions } = - usePlaygroundContext(); - const sidebarItems: SidebarItem[] = [ - // Input/Output Editor - { - name: 'input-editor', - title: 'Input Editor', - isActive: false, - isShow: true, - mobileOnly: true, - onClick: () => { - setShowInputEditor((prevShowOptions) => !prevShowOptions); - setShowOutputNavigation(false); - setShowOptions(false); - }, - icon: showInputEditor ? : , - tooltip: `Show ${showInputEditor ? 'Input Editor' : 'Output Editor'}` - }, - // Options - { - name: 'options', - title: 'Options', - isActive: showOptions, - isShow: true, - mobileOnly: false, - onClick: () => { - setShowOptions((prevShowOptions) => !prevShowOptions); - }, - icon: , - tooltip: 'Show or hide all the options' - }, - // Output Explorer - { - name: 'outputExplorer', - title: 'Output', - isActive: false, - isShow: showInputEditor, - mobileOnly: false, - onClick: () => { - setShowOutputNavigation((prevShowOutputNavigation) => !prevShowOutputNavigation); - }, - icon: , - tooltip: 'Show or hide the list of output models' - } - ]; + const handleClick = useCallback(({ name }: { name: string }) => { dispatch({ type: 'open-option', name }); }, []); return (
@@ -74,15 +18,16 @@ export const Sidebar: React.FunctionComponent = () => {