diff --git a/package.json b/package.json index 39eed93..439d21d 100644 --- a/package.json +++ b/package.json @@ -25,6 +25,12 @@ "@push.rocks/smarti18n": "^1.0.4", "@push.rocks/smartpromise": "^4.2.0", "@push.rocks/smartstring": "^4.0.15", + "@tiptap/core": "^2.22.3", + "@tiptap/extension-link": "^2.22.3", + "@tiptap/extension-text-align": "^2.22.3", + "@tiptap/extension-typography": "^2.22.3", + "@tiptap/extension-underline": "^2.22.3", + "@tiptap/starter-kit": "^2.22.3", "@tsclass/tsclass": "^9.2.0", "@webcontainer/api": "1.2.0", "apexcharts": "^4.7.0", diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index d3cd8ac..7673b2c 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -38,6 +38,24 @@ importers: '@push.rocks/smartstring': specifier: ^4.0.15 version: 4.0.15 + '@tiptap/core': + specifier: ^2.22.3 + version: 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/extension-link': + specifier: ^2.22.3 + version: 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-text-align': + specifier: ^2.22.3 + version: 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-typography': + specifier: ^2.22.3 + version: 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-underline': + specifier: ^2.22.3 + version: 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/starter-kit': + specifier: ^2.22.3 + version: 2.22.3 '@tsclass/tsclass': specifier: ^9.2.0 version: 9.2.0 @@ -1217,6 +1235,9 @@ packages: resolution: {integrity: sha512-6KGnf2vHR7hW4mQpAD7gkDVL3QVML3jb/No/Uw+qCqvs0TaQr60Yjm+CXoLxJNCKwmrL+I1yx8mhAHBHfYJiJA==} deprecated: This package has been deprecated in favour of the new package at @push.rocks/smarturl + '@remirror/core-constants@3.0.0': + resolution: {integrity: sha512-42aWfPrimMfDKDi4YegyS7x+/0tlzaqwPQCULLanv3DMIlu96KTJR0fM5isWX2UViOqlGnX6YFgqWepcX+XMNg==} + '@rolldown/binding-darwin-arm64@1.0.0-beta.18': resolution: {integrity: sha512-F1kqKxIuh9XM6ViC3/Ltz6ARpyUo6b1b2Lo1BhMwR5KwQ06OdOAOY9fmVW5XJ9hHYzABGgvH4hfjtYad0KshAA==} cpu: [arm64] @@ -1601,6 +1622,138 @@ packages: '@tempfix/watcher@2.3.0': resolution: {integrity: sha512-a2qVQffcrnetehvwsN+LdipxQ6jejwZLgAvS9/91+C0gP4CKyikY01c0tSs0I4tSL7qHdCw1Fx0quLw+A9uyLA==} + '@tiptap/core@2.22.3': + resolution: {integrity: sha512-czyBPXZG/ZFyObZEF1kyusGf58Ai3X8TnaxlUUn3gqLLWPy0idXZg85NETCidzi/gAxWxL9j6Pcy+zwS4pbZYQ==} + peerDependencies: + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-blockquote@2.22.3': + resolution: {integrity: sha512-HvTXvqeGaANg0owk0Xxkgyc4lJMO5CZES2Lc3JJp8u5kV+HZIwd78eJ7fbKBMtkpKb4zOk4xQsHQ/TuhghJaeA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-bold@2.22.3': + resolution: {integrity: sha512-J3GxKwijD42eqCwU1SS7PK5aSgnp0wgQDetLz9izAD0RQBrKj5WZA13GnPoTTlzLU4qwjcPRV+6mvF+llH6b6A==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-bullet-list@2.22.3': + resolution: {integrity: sha512-SYvLIxqmuV0kTj4/3ZFlnZ1fr9Y233qX00BKuIpGnczeFsWQmzBJo8vGm3d1IlKPCQN+jTRtDdDE1aSum8Kv2w==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-code-block@2.22.3': + resolution: {integrity: sha512-twPCBpb/ygNixlSBAXgvfo+t56Ucpb8lvPDiZn+cH8OjmmO0ayBoSfSrjKWgaEWGPcXBrFAfsBRbYHyoHj7pXg==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-code@2.22.3': + resolution: {integrity: sha512-s+W6jHezq+n9cC40xZ3hZF6cGGSl+fBELik1b2x8+cb0WoIlqmcdWin1dgeMNrWlRZUw1aD2DNwy/PdXI5vn2g==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-document@2.22.3': + resolution: {integrity: sha512-7MnILbhRZRyROlMUgyntzRZ/EZlqNB8fO761RNjJxR2WMb49R4yc04fz7/+f/QH/hwxoS13bKfsNUDAsDxA5Aw==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-dropcursor@2.22.3': + resolution: {integrity: sha512-yQxSfTWjdUQS+bh6KiNLR9KIMsn1SElzycQe4XE+0eoaetapGtKqxfwkTbbQdNgQOU5wQG1KOda221mnPvkpAA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-gapcursor@2.22.3': + resolution: {integrity: sha512-6Q8TLL4PVGcZLn27eQazCC+be8LP8uzuz5Z5e4TpIeswPAju49cerQOdEGNFKkuYv/FelWIhXNtkWFMf4eSmyw==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-hard-break@2.22.3': + resolution: {integrity: sha512-tbEji/V4Za3UhxYwB36amYhyonwe5j66iYTNRWzgjNixjrcGDbWk6cfaF9jMAgPgIDBmmtQLJY+moKskwgpnZg==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-heading@2.22.3': + resolution: {integrity: sha512-+MexJD+kXtNwMDbNTFa7jCFipx1DqAdT+n9GgInqebAN9bK+CWjC+SskzZNRqeMrQ0Er7QTsi6YC09M+74sevA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-history@2.22.3': + resolution: {integrity: sha512-F9sC45zPw7vbjKrwSKuSLZ0ODyc/X3bGPeCa6HYLEHKfgqsdt2v2fQLvxjpmlwO2ZMrnkBkg76KDxHfVyrZ2zQ==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-horizontal-rule@2.22.3': + resolution: {integrity: sha512-3GvY798p9pCXUBbCebIdSmi1q80l7VZz/B6NN4uUMQ9iwxWopd8yaZ0O7xx2hM2UBzPEtY3M4FAhhpYUTXNFgQ==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-italic@2.22.3': + resolution: {integrity: sha512-W/rQDo7qFL7MfwfaYEcdtbk862fOmBv30qIEwVdqElBye7BFJYKtRuWBzNbG2BwKanjwMbVc/tBXF5W1sqfT7Q==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-link@2.22.3': + resolution: {integrity: sha512-05whzrw8uuRHpFah27a+K3XVjZ78aifO1C5ncQiEgjG8oMbaPG0FpU7yJ9awNtr1U5dvFUEbGvkg1WDW2fLTzA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + '@tiptap/pm': ^2.7.0 + + '@tiptap/extension-list-item@2.22.3': + resolution: {integrity: sha512-B7Fze+eM1sYbGOZtDDAwAivnj1ow2wN5RqaQPC1la3wdTK4Wgp7bdzGjvUbrN6gp3zMFCEWlqP2toc/mRAHCtA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-ordered-list@2.22.3': + resolution: {integrity: sha512-pHGkuZhV/uAAHI9vzk/lpAkbdpMT4wUR1FI17/GE3zNrogfzx0VopCQrXq4+sQVsLUW4I6Cj6VeBjm9wB6qlIw==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-paragraph@2.22.3': + resolution: {integrity: sha512-TYvgS7CweNFo/xVxsKWSt0wnm46Y8OtsfDSjnLbSC4Pj4ZNa6PU3zpvDTW+UxYakr+8zIPvI2WgLBkyTHq6oQA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-strike@2.22.3': + resolution: {integrity: sha512-I+s2Csw2cTHae2vFJiojnHK+NnQjDr6441mSlAd+e7kEly1kjZ4g7J+JMj02ajNQhr/ob8/hb5r6EdIyv2xtoA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-text-align@2.22.3': + resolution: {integrity: sha512-UZ8803BOrHLHSNFfooqgkm2AQsaK/7eE1deQGSazxft89KksAv1kZkEKFypOE8yw85Bg2NHH2Lp6n4tyz2n6/g==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-text-style@2.22.3': + resolution: {integrity: sha512-M3FLOUPcO8fR+rM97mR2gQ54KFkdlAUQtEPKQpO1f312gtcVdBNxgq0WgqTnBY7thWLyqQSKiAsL6y88+JddSA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-text@2.22.3': + resolution: {integrity: sha512-07cymWkPTfq6nuum88Yf90YYArbowed8nNiu0Tw3jCvwpzf9J9TDaovT+LAKuSKtrOsnNpFB/9IqUwFxZepOGw==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-typography@2.22.3': + resolution: {integrity: sha512-pkZUMSDnt1vAR8XwoO9um7WDzBkclSYvWEBdqUN9pnH7Fpz2kbuer/Hqyk94vvWz2D/svndrg1Fb2dLEvfQopA==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/extension-underline@2.22.3': + resolution: {integrity: sha512-floLjh1UbQ2pKgdwfw7qCAJ5VojvH1uqj7xW2RCv79aWYUuJCPD6UBpaBOt/jv7gXDJJ9EeV3m2Hga49CXBrEQ==} + peerDependencies: + '@tiptap/core': ^2.7.0 + + '@tiptap/pm@2.22.3': + resolution: {integrity: sha512-uWPeIScnpQVCYdTnL140XgcvbT1qH288CstMJ6S0Y11lC5PclPK9CxfAipsqgWWrIK7yatxKUVCg6TzfG9zpmA==} + + '@tiptap/starter-kit@2.22.3': + resolution: {integrity: sha512-GkvheaR2ORnHJ9g9R6xIT38w2uppGja/iAIrXLZ9vY1QuR+0cya/ZZ5vKU6r9C2PeyBs3aKYxRD1/j3HDhuGXw==} + '@tokenizer/token@0.3.0': resolution: {integrity: sha512-OvjF+z51L3ov0OyAU0duzsYuvO01PH7x4t6DJx+guahgTnBHkhJdG7soQeTSFLWN3efnHyibZ4Z8l2EuWwJN3A==} @@ -1755,9 +1908,18 @@ packages: '@types/koa@2.15.0': resolution: {integrity: sha512-7QFsywoE5URbuVnG3loe03QXuGajrnotr3gQkXcEBShORai23MePfFYdhz90FEtBBpkyIYQbVD+evKtloCgX3g==} + '@types/linkify-it@5.0.0': + resolution: {integrity: sha512-sVDA58zAw4eWAffKOaQH5/5j3XeayukzDk+ewSsnv3p4yJEZHCCzMDiZM8e0OUrRvmpGZ85jf4yDHkHsgBNr9Q==} + + '@types/markdown-it@14.1.2': + resolution: {integrity: sha512-promo4eFwuiW+TfGxhi+0x3czqTYJkG8qB17ZUJiVF10Xm7NLVRSLUsfRTU/6h1e24VvRnXCx+hG7li58lkzog==} + '@types/mdast@4.0.4': resolution: {integrity: sha512-kGaNbPh1k7AFzgpud/gMdvIm5xuECykRR+JnWKQno9TAXVa6WIVCGTPvYGekIDL4uwCZQSYbUxNBSb1aUo79oA==} + '@types/mdurl@2.0.0': + resolution: {integrity: sha512-RGdgjQUZba5p6QEFAVx2OGb8rQDL/cPRG7GiedRzMcJ1tYnUANBncjbSB1NRGwbvjcPeikRABz2nshyPk1bhWg==} + '@types/mime-types@2.1.4': resolution: {integrity: sha512-lfU4b34HOri+kAY5UheuFMWPDOI+OPceBSHZKp69gEyTL/mmJ4cnU6Y/rlme3UL3GyOn6Y42hyIEw0/q8sWx5w==} @@ -2300,6 +2462,9 @@ packages: typescript: optional: true + crelt@1.0.6: + resolution: {integrity: sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g==} + croner@4.4.1: resolution: {integrity: sha512-aqVeeIPCf5/NZFlz4mN4MLEOs9xf4ODCmHQDs+577JFj8mK3RkKJz77h7+Rn94AijUqKdFNOUHM+v88d8p02UQ==} @@ -3230,6 +3395,12 @@ packages: lines-and-columns@1.2.4: resolution: {integrity: sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==} + linkify-it@5.0.0: + resolution: {integrity: sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ==} + + linkifyjs@4.3.1: + resolution: {integrity: sha512-DRSlB9DKVW04c4SUdGvKK5FR6be45lTU9M76JnngqPeeGDqPwYc0zdUErtsNVMtxPXgUWV4HbXbnC4sNyBxkYg==} + lit-element@4.2.0: resolution: {integrity: sha512-MGrXJVAI5x+Bfth/pU9Kst1iWID6GHDLEzFEnyULB/sFiRLgkd8NPK/PeeXxktA3T6EIIaq8U3KcbTU5XFcP2Q==} @@ -3332,6 +3503,10 @@ packages: make-error@1.3.6: resolution: {integrity: sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw==} + markdown-it@14.1.0: + resolution: {integrity: sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==} + hasBin: true + markdown-table@3.0.4: resolution: {integrity: sha512-wiYz4+JrLyb/DqW2hkFJxP7Vd7JuTDm77fvbM8VfEQdmSMqcImWeeRbHwZjBjIFki/VaMK2BhFi7oUUZeM5bqw==} @@ -3389,6 +3564,9 @@ packages: mdast-util-to-string@4.0.0: resolution: {integrity: sha512-0H44vDimn51F0YwvxSJSm0eCDOJTRlmN0R1yBh4HLj9wiV1Dn0QoXGbvFAWj2hSItVTlCmBF1hqKlIyUBVFLPg==} + mdurl@2.0.0: + resolution: {integrity: sha512-Lf+9+2r+Tdp5wXDXC4PcIBjTDtq4UKjCPMQhKIuzpJNW0b96kVqSwW0bT7FhRSfmAiFYgP+SCRvdrDozfh0U5w==} + media-typer@0.3.0: resolution: {integrity: sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=} engines: {node: '>= 0.6'} @@ -3711,6 +3889,9 @@ packages: resolution: {integrity: sha512-7x81NCL719oNbsq/3mh+hVrAWmFuEYUqrq/Iw3kUzH8ReypT9QQ0BLoJS7/G9k6N81XjW4qHWtjWwe/9eLy1EQ==} engines: {node: '>=12'} + orderedmap@2.1.1: + resolution: {integrity: sha512-TvAWxi0nDe1j/rtMcWcIj94+Ffe6n7zhow33h40SKxmsmozs6dz/e+EajymfoFcHd7sxNn8yHM8839uixMOV6g==} + p-cancelable@3.0.0: resolution: {integrity: sha512-mlVgR3PGuzlo0MmTdk4cXqXWlwQDLnONTAg6sm62XkMJEiRxN3GL3SffkYvqwonbkJBcrI7Uvv5Zh9yjvn2iUw==} engines: {node: '>=12.20'} @@ -3896,6 +4077,64 @@ packages: property-information@7.1.0: resolution: {integrity: sha512-TwEZ+X+yCJmYfL7TPUOcvBZ4QfoT5YenQiJuX//0th53DE6w0xxLEtfK3iyryQFddXuvkIk51EEgrJQ0WJkOmQ==} + prosemirror-changeset@2.3.1: + resolution: {integrity: sha512-j0kORIBm8ayJNl3zQvD1TTPHJX3g042et6y/KQhZhnPrruO8exkTgG8X+NRpj7kIyMMEx74Xb3DyMIBtO0IKkQ==} + + prosemirror-collab@1.3.1: + resolution: {integrity: sha512-4SnynYR9TTYaQVXd/ieUvsVV4PDMBzrq2xPUWutHivDuOshZXqQ5rGbZM84HEaXKbLdItse7weMGOUdDVcLKEQ==} + + prosemirror-commands@1.7.1: + resolution: {integrity: sha512-rT7qZnQtx5c0/y/KlYaGvtG411S97UaL6gdp6RIZ23DLHanMYLyfGBV5DtSnZdthQql7W+lEVbpSfwtO8T+L2w==} + + prosemirror-dropcursor@1.8.2: + resolution: {integrity: sha512-CCk6Gyx9+Tt2sbYk5NK0nB1ukHi2ryaRgadV/LvyNuO3ena1payM2z6Cg0vO1ebK8cxbzo41ku2DE5Axj1Zuiw==} + + prosemirror-gapcursor@1.3.2: + resolution: {integrity: sha512-wtjswVBd2vaQRrnYZaBCbyDqr232Ed4p2QPtRIUK5FuqHYKGWkEwl08oQM4Tw7DOR0FsasARV5uJFvMZWxdNxQ==} + + prosemirror-history@1.4.1: + resolution: {integrity: sha512-2JZD8z2JviJrboD9cPuX/Sv/1ChFng+xh2tChQ2X4bB2HeK+rra/bmJ3xGntCcjhOqIzSDG6Id7e8RJ9QPXLEQ==} + + prosemirror-inputrules@1.5.0: + resolution: {integrity: sha512-K0xJRCmt+uSw7xesnHmcn72yBGTbY45vm8gXI4LZXbx2Z0jwh5aF9xrGQgrVPu0WbyFVFF3E/o9VhJYz6SQWnA==} + + prosemirror-keymap@1.2.3: + resolution: {integrity: sha512-4HucRlpiLd1IPQQXNqeo81BGtkY8Ai5smHhKW9jjPKRc2wQIxksg7Hl1tTI2IfT2B/LgX6bfYvXxEpJl7aKYKw==} + + prosemirror-markdown@1.13.2: + resolution: {integrity: sha512-FPD9rHPdA9fqzNmIIDhhnYQ6WgNoSWX9StUZ8LEKapaXU9i6XgykaHKhp6XMyXlOWetmaFgGDS/nu/w9/vUc5g==} + + prosemirror-menu@1.2.5: + resolution: {integrity: sha512-qwXzynnpBIeg1D7BAtjOusR+81xCp53j7iWu/IargiRZqRjGIlQuu1f3jFi+ehrHhWMLoyOQTSRx/IWZJqOYtQ==} + + prosemirror-model@1.25.1: + resolution: {integrity: sha512-AUvbm7qqmpZa5d9fPKMvH1Q5bqYQvAZWOGRvxsB6iFLyycvC9MwNemNVjHVrWgjaoxAfY8XVg7DbvQ/qxvI9Eg==} + + prosemirror-schema-basic@1.2.4: + resolution: {integrity: sha512-ELxP4TlX3yr2v5rM7Sb70SqStq5NvI15c0j9j/gjsrO5vaw+fnnpovCLEGIcpeGfifkuqJwl4fon6b+KdrODYQ==} + + prosemirror-schema-list@1.5.1: + resolution: {integrity: sha512-927lFx/uwyQaGwJxLWCZRkjXG0p48KpMj6ueoYiu4JX05GGuGcgzAy62dfiV8eFZftgyBUvLx76RsMe20fJl+Q==} + + prosemirror-state@1.4.3: + resolution: {integrity: sha512-goFKORVbvPuAQaXhpbemJFRKJ2aixr+AZMGiquiqKxaucC6hlpHNZHWgz5R7dS4roHiwq9vDctE//CZ++o0W1Q==} + + prosemirror-tables@1.7.1: + resolution: {integrity: sha512-eRQ97Bf+i9Eby99QbyAiyov43iOKgWa7QCGly+lrDt7efZ1v8NWolhXiB43hSDGIXT1UXgbs4KJN3a06FGpr1Q==} + + prosemirror-trailing-node@3.0.0: + resolution: {integrity: sha512-xiun5/3q0w5eRnGYfNlW1uU9W6x5MoFKWwq/0TIRgt09lv7Hcser2QYV8t4muXbEr+Fwo0geYn79Xs4GKywrRQ==} + peerDependencies: + prosemirror-model: ^1.22.1 + prosemirror-state: ^1.4.2 + prosemirror-view: ^1.33.8 + + prosemirror-transform@1.10.4: + resolution: {integrity: sha512-pwDy22nAnGqNR1feOQKHxoFkkUtepoFAd3r2hbEDsnf4wp57kKA36hXsB3njA9FtONBEwSDnDeCiJe+ItD+ykw==} + + prosemirror-view@1.40.0: + resolution: {integrity: sha512-2G3svX0Cr1sJjkD/DYWSe3cfV5VPVTBOxI9XQEGWJDFEpsZb/gh4MV29ctv+OJx2RFX4BLt09i+6zaGM/ldkCw==} + proto-list@1.2.4: resolution: {integrity: sha1-IS1b/hMYMGpCD2QCuOJv85ZHqEk=} @@ -3923,6 +4162,10 @@ packages: pumpify@1.5.1: resolution: {integrity: sha512-oClZI37HvuUJJxSKKrC17bZ9Cu0ZYhEAGPsPUy9KlMUmv9dKX2o77RUmq7f3XjIxbwyGwYzbzQ1L2Ks8sIradQ==} + punycode.js@2.3.1: + resolution: {integrity: sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA==} + engines: {node: '>=6'} + punycode@1.4.1: resolution: {integrity: sha1-wNWmOycYgArY4esPpSachN1BhF4=} @@ -4056,6 +4299,9 @@ packages: resolution: {integrity: sha512-8svdqTMfF/LJ9ZS8NVT4pXAQDFXRrZFVyh9h+qbBprQ4Bge2dj1HkMl3b5LTJdvQY2ioWIBYsMBPw5TJ86j72Q==} hasBin: true + rope-sequence@1.3.4: + resolution: {integrity: sha512-UT5EDe2cu2E/6O4igUr5PSFs23nvvukicWHx6GnOPlHAiiYbzNuCRQCuiUdHJQcqKalLKlrYJnjY0ySGsXNQXQ==} + rss-parser@3.13.0: resolution: {integrity: sha512-7jWUBV5yGN3rqMMj7CZufl/291QAhvrrGpDNE4k/02ZchL0npisiYYqULF71jCEKoIiHvK/Q2e6IkDwPziT7+w==} @@ -4417,6 +4663,9 @@ packages: engines: {node: '>=14.17'} hasBin: true + uc.micro@2.1.0: + resolution: {integrity: sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A==} + uglify-js@3.19.3: resolution: {integrity: sha512-v3Xu+yuwBXisp6QYTcH4UbH+xYJXqnq2m/LtQVWKWzYc1iehYnLixoQDN9FH6/j9/oybfd6W9Ghwkl8+UMKTKQ==} engines: {node: '>=0.8.0'} @@ -4490,6 +4739,9 @@ packages: vfile@6.0.3: resolution: {integrity: sha512-KzIbH/9tXat2u30jf+smMwFCsno4wHVdNmzFyL+T/L3UGqqk6JKfVqOFOZEpZSHADH1k40ab6NUIXZq422ov3Q==} + w3c-keyname@2.2.8: + resolution: {integrity: sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ==} + webidl-conversions@7.0.0: resolution: {integrity: sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g==} engines: {node: '>=12'} @@ -6766,6 +7018,8 @@ snapshots: '@pushrocks/smarturl@2.0.1': {} + '@remirror/core-constants@3.0.0': {} + '@rolldown/binding-darwin-arm64@1.0.0-beta.18': optional: true @@ -7223,6 +7477,154 @@ snapshots: dependencies: stubborn-fs: 1.2.5 + '@tiptap/core@2.22.3(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-blockquote@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-bold@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-bullet-list@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-code-block@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-code@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-document@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-dropcursor@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-gapcursor@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-hard-break@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-heading@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-history@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-horizontal-rule@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + + '@tiptap/extension-italic@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-link@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3)': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/pm': 2.22.3 + linkifyjs: 4.3.1 + + '@tiptap/extension-list-item@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-ordered-list@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-paragraph@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-strike@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-text-align@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-text-style@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-text@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-typography@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/extension-underline@2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + + '@tiptap/pm@2.22.3': + dependencies: + prosemirror-changeset: 2.3.1 + prosemirror-collab: 1.3.1 + prosemirror-commands: 1.7.1 + prosemirror-dropcursor: 1.8.2 + prosemirror-gapcursor: 1.3.2 + prosemirror-history: 1.4.1 + prosemirror-inputrules: 1.5.0 + prosemirror-keymap: 1.2.3 + prosemirror-markdown: 1.13.2 + prosemirror-menu: 1.2.5 + prosemirror-model: 1.25.1 + prosemirror-schema-basic: 1.2.4 + prosemirror-schema-list: 1.5.1 + prosemirror-state: 1.4.3 + prosemirror-tables: 1.7.1 + prosemirror-trailing-node: 3.0.0(prosemirror-model@1.25.1)(prosemirror-state@1.4.3)(prosemirror-view@1.40.0) + prosemirror-transform: 1.10.4 + prosemirror-view: 1.40.0 + + '@tiptap/starter-kit@2.22.3': + dependencies: + '@tiptap/core': 2.22.3(@tiptap/pm@2.22.3) + '@tiptap/extension-blockquote': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-bold': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-bullet-list': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-code': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-code-block': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-document': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-dropcursor': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-gapcursor': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-hard-break': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-heading': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-history': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-horizontal-rule': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3))(@tiptap/pm@2.22.3) + '@tiptap/extension-italic': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-list-item': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-ordered-list': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-paragraph': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-strike': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-text': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/extension-text-style': 2.22.3(@tiptap/core@2.22.3(@tiptap/pm@2.22.3)) + '@tiptap/pm': 2.22.3 + '@tokenizer/token@0.3.0': {} '@tootallnate/quickjs-emscripten@0.23.0': {} @@ -7418,10 +7820,19 @@ snapshots: '@types/koa-compose': 3.2.8 '@types/node': 22.15.31 + '@types/linkify-it@5.0.0': {} + + '@types/markdown-it@14.1.2': + dependencies: + '@types/linkify-it': 5.0.0 + '@types/mdurl': 2.0.0 + '@types/mdast@4.0.4': dependencies: '@types/unist': 3.0.3 + '@types/mdurl@2.0.0': {} + '@types/mime-types@2.1.4': {} '@types/mime@1.3.5': {} @@ -8009,6 +8420,8 @@ snapshots: optionalDependencies: typescript: 5.8.3 + crelt@1.0.6: {} + croner@4.4.1: {} croner@5.7.0: {} @@ -9071,6 +9484,12 @@ snapshots: lines-and-columns@1.2.4: {} + linkify-it@5.0.0: + dependencies: + uc.micro: 2.1.0 + + linkifyjs@4.3.1: {} + lit-element@4.2.0: dependencies: '@lit-labs/ssr-dom-shim': 1.3.0 @@ -9178,6 +9597,15 @@ snapshots: make-error@1.3.6: {} + markdown-it@14.1.0: + dependencies: + argparse: 2.0.1 + entities: 4.5.0 + linkify-it: 5.0.0 + mdurl: 2.0.0 + punycode.js: 2.3.1 + uc.micro: 2.1.0 + markdown-table@3.0.4: {} matcher@3.0.0: @@ -9317,6 +9745,8 @@ snapshots: dependencies: '@types/mdast': 4.0.4 + mdurl@2.0.0: {} + media-typer@0.3.0: {} memory-pager@1.5.0: {} @@ -9721,6 +10151,8 @@ snapshots: is-docker: 2.2.1 is-wsl: 2.2.0 + orderedmap@2.1.1: {} + p-cancelable@3.0.0: {} p-event@4.2.0: @@ -9887,6 +10319,109 @@ snapshots: property-information@7.1.0: {} + prosemirror-changeset@2.3.1: + dependencies: + prosemirror-transform: 1.10.4 + + prosemirror-collab@1.3.1: + dependencies: + prosemirror-state: 1.4.3 + + prosemirror-commands@1.7.1: + dependencies: + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + + prosemirror-dropcursor@1.8.2: + dependencies: + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + prosemirror-view: 1.40.0 + + prosemirror-gapcursor@1.3.2: + dependencies: + prosemirror-keymap: 1.2.3 + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-view: 1.40.0 + + prosemirror-history@1.4.1: + dependencies: + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + prosemirror-view: 1.40.0 + rope-sequence: 1.3.4 + + prosemirror-inputrules@1.5.0: + dependencies: + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + + prosemirror-keymap@1.2.3: + dependencies: + prosemirror-state: 1.4.3 + w3c-keyname: 2.2.8 + + prosemirror-markdown@1.13.2: + dependencies: + '@types/markdown-it': 14.1.2 + markdown-it: 14.1.0 + prosemirror-model: 1.25.1 + + prosemirror-menu@1.2.5: + dependencies: + crelt: 1.0.6 + prosemirror-commands: 1.7.1 + prosemirror-history: 1.4.1 + prosemirror-state: 1.4.3 + + prosemirror-model@1.25.1: + dependencies: + orderedmap: 2.1.1 + + prosemirror-schema-basic@1.2.4: + dependencies: + prosemirror-model: 1.25.1 + + prosemirror-schema-list@1.5.1: + dependencies: + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + + prosemirror-state@1.4.3: + dependencies: + prosemirror-model: 1.25.1 + prosemirror-transform: 1.10.4 + prosemirror-view: 1.40.0 + + prosemirror-tables@1.7.1: + dependencies: + prosemirror-keymap: 1.2.3 + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + prosemirror-view: 1.40.0 + + prosemirror-trailing-node@3.0.0(prosemirror-model@1.25.1)(prosemirror-state@1.4.3)(prosemirror-view@1.40.0): + dependencies: + '@remirror/core-constants': 3.0.0 + escape-string-regexp: 4.0.0 + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-view: 1.40.0 + + prosemirror-transform@1.10.4: + dependencies: + prosemirror-model: 1.25.1 + + prosemirror-view@1.40.0: + dependencies: + prosemirror-model: 1.25.1 + prosemirror-state: 1.4.3 + prosemirror-transform: 1.10.4 + proto-list@1.2.4: {} proxy-addr@2.0.7: @@ -9932,6 +10467,8 @@ snapshots: inherits: 2.0.4 pump: 2.0.1 + punycode.js@2.3.1: {} + punycode@1.4.1: {} punycode@2.3.1: {} @@ -10124,6 +10661,8 @@ snapshots: '@rolldown/binding-win32-ia32-msvc': 1.0.0-beta.18 '@rolldown/binding-win32-x64-msvc': 1.0.0-beta.18 + rope-sequence@1.3.4: {} + rss-parser@3.13.0: dependencies: entities: 2.2.0 @@ -10547,6 +11086,8 @@ snapshots: typescript@5.8.3: {} + uc.micro@2.1.0: {} + uglify-js@3.19.3: {} uint8array-extras@1.4.0: {} @@ -10619,6 +11160,8 @@ snapshots: '@types/unist': 3.0.3 vfile-message: 4.0.2 + w3c-keyname@2.2.8: {} + webidl-conversions@7.0.0: {} whatwg-mimetype@3.0.0: {} diff --git a/ts_web/elements/dees-input-richtext.demo.ts b/ts_web/elements/dees-input-richtext.demo.ts new file mode 100644 index 0000000..518a7da --- /dev/null +++ b/ts_web/elements/dees-input-richtext.demo.ts @@ -0,0 +1,174 @@ +import { html, css } from '@design.estate/dees-element'; +import '@design.estate/dees-wcctools/demotools'; + +export const demoFunc = () => html` + + + +
+
+

Basic Rich Text Editor

+

A full-featured rich text editor with formatting toolbar

+ + Welcome to the Rich Text Editor!

This is a feature-rich editor built with TipTap. You can:

  • Format text with various styles
  • Create different heading levels
  • Add links to external resources
  • Write inline code or code blocks

Use the toolbar above to explore all the formatting options available!

Start typing to see the magic happen...

'} + .description=${'Use the toolbar to format your content with headings, lists, links, and more'} + .showWordCount=${true} + >
+
+ +
+

With Placeholder

+

Empty editor with placeholder text

+ + +
+ +
+

Different Heights

+

Editors with different minimum heights for various use cases

+ +
+ + + +
+
+ +
+

Code Examples

+

Editor pre-filled with code examples

+ + Installation Guide

To install the package, run the following command:

npm install @design.estate/dees-catalog

Then import the component in your TypeScript file:

import { DeesInputRichtext } from "@design.estate/dees-catalog";

You can now use the <dees-input-richtext> element in your templates.

'} + .minHeight=${250} + .showWordCount=${true} + >
+
+ +
+

Disabled State

+

Read-only rich text content

+ + The Future of Web Components

Web Components have revolutionized how we build modern web applications...

"The future of web development lies in reusable, encapsulated components."

'} + .disabled=${true} + .showWordCount=${true} + >
+
+ +
+

Interactive Demo

+

Type in the editor below and see the HTML output

+ + { + const output = document.querySelector('#output-preview'); + if (output) { + output.textContent = e.detail.value; + } + }} + > + +
+ HTML output will appear here... +
+
+
+
+`; \ No newline at end of file diff --git a/ts_web/elements/dees-input-richtext.ts b/ts_web/elements/dees-input-richtext.ts new file mode 100644 index 0000000..b0d6870 --- /dev/null +++ b/ts_web/elements/dees-input-richtext.ts @@ -0,0 +1,704 @@ +import * as colors from './00colors.js'; +import { DeesInputBase } from './dees-input-base.js'; +import { demoFunc } from './dees-input-richtext.demo.js'; + +import { + customElement, + type TemplateResult, + property, + html, + css, + cssManager, + state, + query, +} from '@design.estate/dees-element'; +import * as domtools from '@design.estate/dees-domtools'; + +import { Editor } from '@tiptap/core'; +import StarterKit from '@tiptap/starter-kit'; +import Underline from '@tiptap/extension-underline'; +import TextAlign from '@tiptap/extension-text-align'; +import Link from '@tiptap/extension-link'; +import Typography from '@tiptap/extension-typography'; + +declare global { + interface HTMLElementTagNameMap { + 'dees-input-richtext': DeesInputRichtext; + } +} + +interface IToolbarButton { + name: string; + icon: string; + action?: () => void; + isActive?: () => boolean; + title: string; + isDivider?: boolean; +} + +@customElement('dees-input-richtext') +export class DeesInputRichtext extends DeesInputBase { + public static demo = demoFunc; + + // INSTANCE + @property({ + type: String, + reflect: true, + }) + public value: string = ''; + + @property({ + type: String, + }) + public placeholder: string = ''; + + @property({ + type: Boolean, + }) + public showWordCount: boolean = true; + + @property({ + type: Number, + }) + public minHeight: number = 200; + + @state() + private showLinkInput: boolean = false; + + @state() + private wordCount: number = 0; + + @query('.editor-content') + private editorElement: HTMLElement; + + @query('.link-input input') + private linkInputElement: HTMLInputElement; + + private editor: Editor; + + public static styles = [ + ...DeesInputBase.baseStyles, + cssManager.defaultStyles, + css` + :host { + display: block; + position: relative; + font-family: Inter, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; + } + + .input-wrapper { + position: relative; + } + + .label { + display: block; + margin-bottom: 8px; + font-size: 14px; + font-weight: 500; + color: ${cssManager.bdTheme('#374151', '#e4e4e7')}; + } + + .editor-container { + display: flex; + flex-direction: column; + min-height: ${cssManager.bdTheme('200px', '200px')}; + border: 1px solid ${cssManager.bdTheme('#e1e5e9', '#2c2c2c')}; + border-radius: 8px; + background: ${cssManager.bdTheme('#ffffff', '#141414')}; + overflow: hidden; + transition: all 0.2s ease; + } + + .editor-container:hover { + border-color: ${cssManager.bdTheme('#d1d5db', '#404040')}; + } + + .editor-container.focused { + border-color: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + box-shadow: 0 0 0 3px ${cssManager.bdTheme('rgba(0, 80, 185, 0.1)', 'rgba(0, 105, 242, 0.1)')}; + } + + .editor-toolbar { + display: flex; + flex-wrap: wrap; + gap: 4px; + padding: 8px 12px; + background: ${cssManager.bdTheme('#f8f9fa', '#1a1a1a')}; + border-bottom: 1px solid ${cssManager.bdTheme('#e1e5e9', '#2c2c2c')}; + align-items: center; + position: relative; + } + + .toolbar-button { + display: flex; + align-items: center; + justify-content: center; + width: 32px; + height: 32px; + border: none; + border-radius: 4px; + background: transparent; + cursor: pointer; + font-size: 14px; + font-weight: 500; + color: ${cssManager.bdTheme('#374151', '#9ca3af')}; + transition: all 0.2s; + user-select: none; + } + + .toolbar-button:hover { + background: ${cssManager.bdTheme('#e5e7eb', '#2c2c2c')}; + color: ${cssManager.bdTheme('#1f2937', '#e4e4e7')}; + } + + .toolbar-button.active { + background: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + color: white; + } + + .toolbar-button:disabled { + opacity: 0.5; + cursor: not-allowed; + } + + .toolbar-divider { + width: 1px; + height: 24px; + background: ${cssManager.bdTheme('#d1d5db', '#404040')}; + margin: 0 4px; + } + + .editor-content { + flex: 1; + padding: 16px; + overflow-y: auto; + min-height: var(--min-height, 200px); + } + + .editor-content .ProseMirror { + outline: none; + line-height: 1.6; + color: ${cssManager.bdTheme('#374151', '#e4e4e7')}; + min-height: 100%; + } + + .editor-content .ProseMirror p { + margin: 0.5em 0; + } + + .editor-content .ProseMirror p:first-child { + margin-top: 0; + } + + .editor-content .ProseMirror p:last-child { + margin-bottom: 0; + } + + .editor-content .ProseMirror h1 { + font-size: 2em; + font-weight: bold; + margin: 1em 0 0.5em 0; + line-height: 1.2; + } + + .editor-content .ProseMirror h2 { + font-size: 1.5em; + font-weight: bold; + margin: 1em 0 0.5em 0; + line-height: 1.3; + } + + .editor-content .ProseMirror h3 { + font-size: 1.25em; + font-weight: bold; + margin: 1em 0 0.5em 0; + line-height: 1.4; + } + + .editor-content .ProseMirror ul, + .editor-content .ProseMirror ol { + padding-left: 1.5em; + margin: 0.5em 0; + } + + .editor-content .ProseMirror li { + margin: 0.25em 0; + } + + .editor-content .ProseMirror blockquote { + border-left: 4px solid ${cssManager.bdTheme('#d1d5db', '#404040')}; + margin: 1em 0; + padding-left: 1em; + color: ${cssManager.bdTheme('#6b7280', '#9ca3af')}; + font-style: italic; + } + + .editor-content .ProseMirror code { + background: ${cssManager.bdTheme('#f3f4f6', '#2c2c2c')}; + border-radius: 4px; + padding: 0.2em 0.4em; + font-family: 'Fira Code', 'SF Mono', Monaco, 'Cascadia Code', 'Roboto Mono', Consolas, 'Courier New', monospace; + font-size: 0.9em; + color: ${cssManager.bdTheme('#e11d48', '#f87171')}; + } + + .editor-content .ProseMirror pre { + background: ${cssManager.bdTheme('#1f2937', '#0a0a0a')}; + color: ${cssManager.bdTheme('#f9fafb', '#e4e4e7')}; + border-radius: 6px; + padding: 1em; + margin: 1em 0; + overflow-x: auto; + } + + .editor-content .ProseMirror pre code { + background: none; + color: inherit; + padding: 0; + border-radius: 0; + } + + .editor-content .ProseMirror a { + color: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + text-decoration: underline; + cursor: pointer; + } + + .editor-content .ProseMirror a:hover { + color: ${cssManager.bdTheme('#0069f2', '#0084ff')}; + } + + .editor-footer { + padding: 8px 12px; + background: ${cssManager.bdTheme('#f8f9fa', '#1a1a1a')}; + border-top: 1px solid ${cssManager.bdTheme('#e1e5e9', '#2c2c2c')}; + font-size: 12px; + color: ${cssManager.bdTheme('#6b7280', '#9ca3af')}; + display: flex; + justify-content: space-between; + align-items: center; + } + + .word-count { + font-weight: 500; + } + + .link-input { + display: none; + position: absolute; + top: 100%; + left: 0; + right: 0; + background: ${cssManager.bdTheme('#ffffff', '#1a1a1a')}; + border: 1px solid ${cssManager.bdTheme('#d1d5db', '#404040')}; + border-radius: 6px; + box-shadow: 0 4px 6px -1px rgba(0, 0, 0, 0.1); + padding: 12px; + z-index: 1000; + } + + .link-input.show { + display: block; + } + + .link-input input { + width: 100%; + padding: 8px 12px; + border: 1px solid ${cssManager.bdTheme('#d1d5db', '#404040')}; + border-radius: 4px; + outline: none; + font-size: 14px; + background: ${cssManager.bdTheme('#ffffff', '#0a0a0a')}; + color: ${cssManager.bdTheme('#374151', '#e4e4e7')}; + } + + .link-input input:focus { + border-color: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + box-shadow: 0 0 0 2px ${cssManager.bdTheme('rgba(0, 80, 185, 0.1)', 'rgba(0, 105, 242, 0.1)')}; + } + + .link-input-buttons { + display: flex; + gap: 8px; + margin-top: 8px; + } + + .link-input-buttons button { + padding: 6px 12px; + border: 1px solid ${cssManager.bdTheme('#d1d5db', '#404040')}; + border-radius: 4px; + background: ${cssManager.bdTheme('#ffffff', '#1a1a1a')}; + cursor: pointer; + font-size: 12px; + color: ${cssManager.bdTheme('#374151', '#e4e4e7')}; + transition: all 0.2s; + } + + .link-input-buttons button:hover { + background: ${cssManager.bdTheme('#f3f4f6', '#2c2c2c')}; + } + + .link-input-buttons button.primary { + background: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + color: white; + border-color: ${cssManager.bdTheme('#0050b9', '#0069f2')}; + } + + .link-input-buttons button.primary:hover { + background: ${cssManager.bdTheme('#0069f2', '#0084ff')}; + } + + .description { + margin-top: 8px; + font-size: 12px; + color: ${cssManager.bdTheme('#6b7280', '#9ca3af')}; + line-height: 1.4; + } + + :host([disabled]) .editor-container { + opacity: 0.6; + cursor: not-allowed; + } + + :host([disabled]) .toolbar-button, + :host([disabled]) .editor-content { + pointer-events: none; + } + `, + ]; + + public render(): TemplateResult { + return html` +
+ ${this.label ? html`` : ''} +
+
+ ${this.renderToolbar()} + +
+
+ ${this.showWordCount + ? html` + + ` + : ''} +
+ ${this.description ? html`
${this.description}
` : ''} +
+ `; + } + + private renderToolbar(): TemplateResult { + const buttons: IToolbarButton[] = this.getToolbarButtons(); + + return html` + ${buttons.map((button) => { + if (button.isDivider) { + return html`
`; + } + return html` + + `; + })} + `; + } + + private getToolbarButtons(): IToolbarButton[] { + if (!this.editor) return []; + + return [ + { + name: 'bold', + icon: '𝐁', + title: 'Bold (Ctrl+B)', + action: () => this.editor.chain().focus().toggleBold().run(), + isActive: () => this.editor.isActive('bold'), + }, + { + name: 'italic', + icon: '𝐼', + title: 'Italic (Ctrl+I)', + action: () => this.editor.chain().focus().toggleItalic().run(), + isActive: () => this.editor.isActive('italic'), + }, + { + name: 'underline', + icon: '𝐔', + title: 'Underline (Ctrl+U)', + action: () => this.editor.chain().focus().toggleUnderline().run(), + isActive: () => this.editor.isActive('underline'), + }, + { + name: 'strike', + icon: '𝐒', + title: 'Strikethrough', + action: () => this.editor.chain().focus().toggleStrike().run(), + isActive: () => this.editor.isActive('strike'), + }, + { name: 'divider1', icon: '', title: '', isDivider: true }, + { + name: 'h1', + icon: 'H₁', + title: 'Heading 1', + action: () => this.editor.chain().focus().toggleHeading({ level: 1 }).run(), + isActive: () => this.editor.isActive('heading', { level: 1 }), + }, + { + name: 'h2', + icon: 'H₂', + title: 'Heading 2', + action: () => this.editor.chain().focus().toggleHeading({ level: 2 }).run(), + isActive: () => this.editor.isActive('heading', { level: 2 }), + }, + { + name: 'h3', + icon: 'H₃', + title: 'Heading 3', + action: () => this.editor.chain().focus().toggleHeading({ level: 3 }).run(), + isActive: () => this.editor.isActive('heading', { level: 3 }), + }, + { name: 'divider2', icon: '', title: '', isDivider: true }, + { + name: 'bulletList', + icon: '• ', + title: 'Bullet List', + action: () => this.editor.chain().focus().toggleBulletList().run(), + isActive: () => this.editor.isActive('bulletList'), + }, + { + name: 'orderedList', + icon: '1.', + title: 'Numbered List', + action: () => this.editor.chain().focus().toggleOrderedList().run(), + isActive: () => this.editor.isActive('orderedList'), + }, + { + name: 'blockquote', + icon: '"', + title: 'Quote', + action: () => this.editor.chain().focus().toggleBlockquote().run(), + isActive: () => this.editor.isActive('blockquote'), + }, + { + name: 'code', + icon: '<>', + title: 'Code', + action: () => this.editor.chain().focus().toggleCode().run(), + isActive: () => this.editor.isActive('code'), + }, + { + name: 'codeBlock', + icon: '{}', + title: 'Code Block', + action: () => this.editor.chain().focus().toggleCodeBlock().run(), + isActive: () => this.editor.isActive('codeBlock'), + }, + { name: 'divider3', icon: '', title: '', isDivider: true }, + { + name: 'link', + icon: '🔗', + title: 'Add Link', + action: () => this.toggleLink(), + isActive: () => this.editor.isActive('link'), + }, + { + name: 'alignLeft', + icon: '⬅', + title: 'Align Left', + action: () => this.editor.chain().focus().setTextAlign('left').run(), + isActive: () => this.editor.isActive({ textAlign: 'left' }), + }, + { + name: 'alignCenter', + icon: '⬄', + title: 'Align Center', + action: () => this.editor.chain().focus().setTextAlign('center').run(), + isActive: () => this.editor.isActive({ textAlign: 'center' }), + }, + { + name: 'alignRight', + icon: '➡', + title: 'Align Right', + action: () => this.editor.chain().focus().setTextAlign('right').run(), + isActive: () => this.editor.isActive({ textAlign: 'right' }), + }, + { name: 'divider4', icon: '', title: '', isDivider: true }, + { + name: 'undo', + icon: '↶', + title: 'Undo (Ctrl+Z)', + action: () => this.editor.chain().focus().undo().run(), + }, + { + name: 'redo', + icon: '↷', + title: 'Redo (Ctrl+Y)', + action: () => this.editor.chain().focus().redo().run(), + }, + ]; + } + + public async firstUpdated() { + await this.updateComplete; + this.initializeEditor(); + } + + private initializeEditor(): void { + if (this.disabled) return; + + this.editor = new Editor({ + element: this.editorElement, + extensions: [ + StarterKit.configure({ + heading: { + levels: [1, 2, 3], + }, + }), + Underline, + TextAlign.configure({ + types: ['heading', 'paragraph'], + }), + Link.configure({ + openOnClick: false, + HTMLAttributes: { + class: 'editor-link', + }, + }), + Typography, + ], + content: this.value || (this.placeholder ? `

${this.placeholder}

` : ''), + onUpdate: ({ editor }) => { + this.value = editor.getHTML(); + this.updateWordCount(); + this.dispatchEvent( + new CustomEvent('input', { + detail: { value: this.value }, + bubbles: true, + composed: true, + }) + ); + this.dispatchEvent( + new CustomEvent('change', { + detail: { value: this.value }, + bubbles: true, + composed: true, + }) + ); + }, + onSelectionUpdate: () => { + this.requestUpdate(); + }, + onFocus: () => { + this.requestUpdate(); + }, + onBlur: () => { + this.requestUpdate(); + }, + }); + + this.updateWordCount(); + } + + private updateWordCount(): void { + if (!this.editor) return; + const text = this.editor.getText(); + this.wordCount = text.trim() ? text.trim().split(/\s+/).length : 0; + } + + private toggleLink(): void { + if (!this.editor) return; + + if (this.editor.isActive('link')) { + const href = this.editor.getAttributes('link').href; + this.showLinkInput = true; + requestAnimationFrame(() => { + if (this.linkInputElement) { + this.linkInputElement.value = href || ''; + this.linkInputElement.focus(); + this.linkInputElement.select(); + } + }); + } else { + this.showLinkInput = true; + requestAnimationFrame(() => { + if (this.linkInputElement) { + this.linkInputElement.value = ''; + this.linkInputElement.focus(); + } + }); + } + } + + private saveLink(): void { + if (!this.editor || !this.linkInputElement) return; + + const url = this.linkInputElement.value; + if (url) { + this.editor.chain().focus().setLink({ href: url }).run(); + } + this.hideLinkInput(); + } + + private removeLink(): void { + if (!this.editor) return; + this.editor.chain().focus().unsetLink().run(); + this.hideLinkInput(); + } + + private hideLinkInput(): void { + this.showLinkInput = false; + this.editor?.commands.focus(); + } + + private handleLinkInputKeydown(e: KeyboardEvent): void { + if (e.key === 'Enter') { + e.preventDefault(); + this.saveLink(); + } else if (e.key === 'Escape') { + e.preventDefault(); + this.hideLinkInput(); + } + } + + public setValue(value: string): void { + this.value = value; + if (this.editor && value !== this.editor.getHTML()) { + this.editor.commands.setContent(value); + } + } + + public getValue(): string { + return this.value; + } + + public clear(): void { + this.setValue(''); + } + + public focus(): void { + this.editor?.commands.focus(); + } + + public async disconnectedCallback(): Promise { + await super.disconnectedCallback(); + if (this.editor) { + this.editor.destroy(); + } + } +} \ No newline at end of file diff --git a/ts_web/elements/index.ts b/ts_web/elements/index.ts index 141bcb8..ae46b56 100644 --- a/ts_web/elements/index.ts +++ b/ts_web/elements/index.ts @@ -35,6 +35,7 @@ export * from './dees-input-wysiwyg.js'; export * from './dees-progressbar.js'; export * from './dees-input-quantityselector.js'; export * from './dees-input-radio.js'; +export * from './dees-input-richtext.js'; export * from './dees-input-text.js'; export * from './dees-label.js'; export * from './dees-mobilenavigation.js';