@nikcli-ai/app 0.0.6
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.turbo/turbo-build.log +661 -0
- package/.turbo/turbo-typecheck.log +1 -0
- package/AGENTS.md +30 -0
- package/README.md +51 -0
- package/dist/_headers +17 -0
- package/dist/apple-touch-icon-v3.png +0 -0
- package/dist/apple-touch-icon.png +0 -0
- package/dist/assets/BlexMonoNerdFontMono-Bold-B8jzonSj.woff2 +0 -0
- package/dist/assets/BlexMonoNerdFontMono-Medium-BvtJB5kd.woff2 +0 -0
- package/dist/assets/BlexMonoNerdFontMono-Regular-DSJ7IWr2.woff2 +0 -0
- package/dist/assets/CaskaydiaCoveNerdFontMono-Bold-CxABrWmj.woff2 +0 -0
- package/dist/assets/CaskaydiaCoveNerdFontMono-Regular-C_H0OSLN.woff2 +0 -0
- package/dist/assets/FiraCodeNerdFontMono-Bold-BjAeM3gJ.woff2 +0 -0
- package/dist/assets/FiraCodeNerdFontMono-Regular-io3c92n9.woff2 +0 -0
- package/dist/assets/HackNerdFontMono-Bold-BNG4kp7w.woff2 +0 -0
- package/dist/assets/HackNerdFontMono-Regular-IcpSchWC.woff2 +0 -0
- package/dist/assets/InconsolataNerdFontMono-Bold-oTRjQesI.woff2 +0 -0
- package/dist/assets/InconsolataNerdFontMono-Regular-CRHGEvh2.woff2 +0 -0
- package/dist/assets/IntoneMonoNerdFontMono-Bold-BL6LrHzx.woff2 +0 -0
- package/dist/assets/IntoneMonoNerdFontMono-Regular-BwjBdmsJ.woff2 +0 -0
- package/dist/assets/JetBrainsMonoNerdFontMono-Bold-CU80ifuM.woff2 +0 -0
- package/dist/assets/JetBrainsMonoNerdFontMono-Regular-QVq88ZfU.woff2 +0 -0
- package/dist/assets/KaTeX_AMS-Regular-BQhdFMY1.woff2 +0 -0
- package/dist/assets/KaTeX_AMS-Regular-DMm9YOAa.woff +0 -0
- package/dist/assets/KaTeX_AMS-Regular-DRggAlZN.ttf +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-ATXxdsX0.ttf +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-BEiXGLvX.woff +0 -0
- package/dist/assets/KaTeX_Caligraphic-Bold-Dq_IR9rO.woff2 +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-CTRA-rTL.woff +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-Di6jR-x-.woff2 +0 -0
- package/dist/assets/KaTeX_Caligraphic-Regular-wX97UBjC.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-BdnERNNW.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-BsDP51OF.woff +0 -0
- package/dist/assets/KaTeX_Fraktur-Bold-CL6g_b3V.woff2 +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-CB_wures.ttf +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-CTYiF6lA.woff2 +0 -0
- package/dist/assets/KaTeX_Fraktur-Regular-Dxdc4cR9.woff +0 -0
- package/dist/assets/KaTeX_Main-Bold-Cx986IdX.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Bold-Jm3AIy58.woff +0 -0
- package/dist/assets/KaTeX_Main-Bold-waoOVXN0.ttf +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-DxDJ3AOS.woff2 +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-DzxPMmG6.ttf +0 -0
- package/dist/assets/KaTeX_Main-BoldItalic-SpSLRI95.woff +0 -0
- package/dist/assets/KaTeX_Main-Italic-3WenGoN9.ttf +0 -0
- package/dist/assets/KaTeX_Main-Italic-BMLOBm91.woff +0 -0
- package/dist/assets/KaTeX_Main-Italic-NWA7e6Wa.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Regular-B22Nviop.woff2 +0 -0
- package/dist/assets/KaTeX_Main-Regular-Dr94JaBh.woff +0 -0
- package/dist/assets/KaTeX_Main-Regular-ypZvNtVU.ttf +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-B3XSjfu4.ttf +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-CZnvNsCZ.woff2 +0 -0
- package/dist/assets/KaTeX_Math-BoldItalic-iY-2wyZ7.woff +0 -0
- package/dist/assets/KaTeX_Math-Italic-DA0__PXp.woff +0 -0
- package/dist/assets/KaTeX_Math-Italic-flOr_0UB.ttf +0 -0
- package/dist/assets/KaTeX_Math-Italic-t53AETM-.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-CFMepnvq.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-D1sUS0GD.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Bold-DbIhKOiC.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-C3H0VqGB.woff2 +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-DN2j7dab.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Italic-YYjJ1zSn.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-BNo7hRIc.ttf +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-CS6fqUqJ.woff +0 -0
- package/dist/assets/KaTeX_SansSerif-Regular-DDBCnlJ7.woff2 +0 -0
- package/dist/assets/KaTeX_Script-Regular-C5JkGWo-.ttf +0 -0
- package/dist/assets/KaTeX_Script-Regular-D3wIWfF6.woff2 +0 -0
- package/dist/assets/KaTeX_Script-Regular-D5yQViql.woff +0 -0
- package/dist/assets/KaTeX_Size1-Regular-C195tn64.woff +0 -0
- package/dist/assets/KaTeX_Size1-Regular-Dbsnue_I.ttf +0 -0
- package/dist/assets/KaTeX_Size1-Regular-mCD8mA8B.woff2 +0 -0
- package/dist/assets/KaTeX_Size2-Regular-B7gKUWhC.ttf +0 -0
- package/dist/assets/KaTeX_Size2-Regular-Dy4dx90m.woff2 +0 -0
- package/dist/assets/KaTeX_Size2-Regular-oD1tc_U0.woff +0 -0
- package/dist/assets/KaTeX_Size3-Regular-CTq5MqoE.woff +0 -0
- package/dist/assets/KaTeX_Size3-Regular-DgpXs0kz.ttf +0 -0
- package/dist/assets/KaTeX_Size4-Regular-BF-4gkZK.woff +0 -0
- package/dist/assets/KaTeX_Size4-Regular-DWFBv043.ttf +0 -0
- package/dist/assets/KaTeX_Size4-Regular-Dl5lxZxV.woff2 +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-C0xS9mPB.woff +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-CO6r4hn1.woff2 +0 -0
- package/dist/assets/KaTeX_Typewriter-Regular-D3Ib7_Hf.ttf +0 -0
- package/dist/assets/MesloLGSNerdFontMono-Bold-CrpVO3ec.woff2 +0 -0
- package/dist/assets/MesloLGSNerdFontMono-Regular-j-nTZDWZ.woff2 +0 -0
- package/dist/assets/RobotoMonoNerdFontMono-Bold-DNxuDepp.woff2 +0 -0
- package/dist/assets/RobotoMonoNerdFontMono-Regular-DvxS3QZC.woff2 +0 -0
- package/dist/assets/SauceCodeProNerdFontMono-Bold-DloEeUVQ.woff2 +0 -0
- package/dist/assets/SauceCodeProNerdFontMono-Regular-Ba96Bdne.woff2 +0 -0
- package/dist/assets/UbuntuMonoNerdFontMono-Bold-wLXUURqB.woff2 +0 -0
- package/dist/assets/UbuntuMonoNerdFontMono-Regular-tdnXLyap.woff2 +0 -0
- package/dist/assets/abap-BXYWi5l2.js +1 -0
- package/dist/assets/abap-BdImnpbu.js +1 -0
- package/dist/assets/actionscript-3-CfeIJUat.js +1 -0
- package/dist/assets/actionscript-3-LjD3sNyx.js +1 -0
- package/dist/assets/ada-DeReRNhg.js +1 -0
- package/dist/assets/ada-bCR0ucgS.js +1 -0
- package/dist/assets/alert-01-BuOD_o_q.aac +0 -0
- package/dist/assets/alert-02-CS75AsoP.aac +0 -0
- package/dist/assets/alert-04-CaGsIGFP.aac +0 -0
- package/dist/assets/alert-05-D2gbGoRH.aac +0 -0
- package/dist/assets/alert-06-GK5Tqy6u.aac +0 -0
- package/dist/assets/alert-07-0r0kiLGz.aac +0 -0
- package/dist/assets/alert-08-CZowNquU.aac +0 -0
- package/dist/assets/alert-10-Ck5hR7zH.aac +0 -0
- package/dist/assets/android-studio-D0DqtVA4.svg +1 -0
- package/dist/assets/andromeeda-C-Jbm3Hp.js +1 -0
- package/dist/assets/angular-html-6NEUuZou.js +1 -0
- package/dist/assets/angular-html-CU67Zn6k.js +1 -0
- package/dist/assets/angular-ts-BfBSdvnY.js +1 -0
- package/dist/assets/angular-ts-BwZT4LLn.js +1 -0
- package/dist/assets/antigravity-XfClf80L.svg +1 -0
- package/dist/assets/apache-CCUDgqAf.js +1 -0
- package/dist/assets/apache-Pmp26Uib.js +1 -0
- package/dist/assets/apex-DDbsPZ6N.js +1 -0
- package/dist/assets/apex-ynV6aGfE.js +1 -0
- package/dist/assets/apl-C-KxP0FL.js +1 -0
- package/dist/assets/apl-dKokRX4l.js +1 -0
- package/dist/assets/applescript-Co6uUVPk.js +1 -0
- package/dist/assets/applescript-Dd26Nb5t.js +1 -0
- package/dist/assets/ara-BRHolxvo.js +1 -0
- package/dist/assets/ara-D6vCOKVX.js +1 -0
- package/dist/assets/asciidoc-DZ9uQOoP.js +1 -0
- package/dist/assets/asciidoc-Dv7Oe6Be.js +1 -0
- package/dist/assets/asm-CbVMAfFB.js +1 -0
- package/dist/assets/asm-D_Q5rh1f.js +1 -0
- package/dist/assets/astro-CbQHKStN.js +1 -0
- package/dist/assets/astro-OEl6i-aI.js +1 -0
- package/dist/assets/aurora-x-D-2ljcwZ.js +1 -0
- package/dist/assets/awk-DMzUqQB5.js +1 -0
- package/dist/assets/awk-UmR4Ublb.js +1 -0
- package/dist/assets/ayu-dark-Cv9koXgw.js +1 -0
- package/dist/assets/ballerina-BFfxhgS-.js +1 -0
- package/dist/assets/ballerina-BMbY-1wP.js +1 -0
- package/dist/assets/bat-BkioyH1T.js +1 -0
- package/dist/assets/bat-Dcnh2ot_.js +1 -0
- package/dist/assets/beancount-B92nBv9q.js +1 -0
- package/dist/assets/beancount-k_qm7-4y.js +1 -0
- package/dist/assets/berry-C9fz7p0n.js +1 -0
- package/dist/assets/berry-uYugtg8r.js +1 -0
- package/dist/assets/bibtex-CHM0blh-.js +1 -0
- package/dist/assets/bibtex-DELXwv0g.js +1 -0
- package/dist/assets/bicep-Bmn6On1c.js +1 -0
- package/dist/assets/bicep-D2njIRU5.js +1 -0
- package/dist/assets/bip-bop-03-DXp7Zb0f.aac +0 -0
- package/dist/assets/bip-bop-04-CfVtpI7z.aac +0 -0
- package/dist/assets/bip-bop-06-BuvNosjK.aac +0 -0
- package/dist/assets/bip-bop-08-DBf7Bwjz.aac +0 -0
- package/dist/assets/bip-bop-09-CxEgoAHQ.aac +0 -0
- package/dist/assets/blade-CwZbp2Ca.js +1 -0
- package/dist/assets/blade-DVc8C-J4.js +1 -0
- package/dist/assets/bsl-BO_Y6i37.js +1 -0
- package/dist/assets/bsl-C4qeDxuX.js +1 -0
- package/dist/assets/c-BIGW1oBm.js +1 -0
- package/dist/assets/c-CyUOlKFr.js +1 -0
- package/dist/assets/cadence-Bv_4Rxtq.js +1 -0
- package/dist/assets/cadence-bIHbUj02.js +1 -0
- package/dist/assets/cairo-C_mOTUpN.js +1 -0
- package/dist/assets/cairo-KRGpt6FW.js +1 -0
- package/dist/assets/catppuccin-frappe-DFWUc33u.js +1 -0
- package/dist/assets/catppuccin-latte-C9dUb6Cb.js +1 -0
- package/dist/assets/catppuccin-macchiato-DQyhUUbL.js +1 -0
- package/dist/assets/catppuccin-mocha-D87Tk5Gz.js +1 -0
- package/dist/assets/clarity-A4x-rz24.js +1 -0
- package/dist/assets/clarity-D53aC0YG.js +1 -0
- package/dist/assets/clojure-Bb2-3A0I.js +1 -0
- package/dist/assets/clojure-P80f7IUj.js +1 -0
- package/dist/assets/cmake-Cv5ljclQ.js +1 -0
- package/dist/assets/cmake-D1j8_8rp.js +1 -0
- package/dist/assets/cobol-Cg3yRVOY.js +1 -0
- package/dist/assets/cobol-nwyudZeR.js +1 -0
- package/dist/assets/codeowners-Bp6g37R7.js +1 -0
- package/dist/assets/codeowners-CKT8dJip.js +1 -0
- package/dist/assets/codeql-DXd8ggn3.js +1 -0
- package/dist/assets/codeql-DsOJ9woJ.js +1 -0
- package/dist/assets/coffee-CSg58a7P.js +1 -0
- package/dist/assets/coffee-Ch7k5sss.js +1 -0
- package/dist/assets/common-lisp-Cg-RD9OK.js +1 -0
- package/dist/assets/common-lisp-v102bg8X.js +1 -0
- package/dist/assets/coq-CxCehFeR.js +1 -0
- package/dist/assets/coq-DkFqJrB1.js +1 -0
- package/dist/assets/cpp-C28rag7_.js +1 -0
- package/dist/assets/cpp-CofmeUqb.js +1 -0
- package/dist/assets/crystal-kLfOnxTY.js +1 -0
- package/dist/assets/crystal-tKQVLTB8.js +1 -0
- package/dist/assets/csharp-DFQpaWb6.js +1 -0
- package/dist/assets/csharp-K5feNrxe.js +1 -0
- package/dist/assets/css-BSK200cd.js +1 -0
- package/dist/assets/css-DPfMkruS.js +1 -0
- package/dist/assets/csv-AZNL9t8Q.js +1 -0
- package/dist/assets/csv-fuZLfV_i.js +1 -0
- package/dist/assets/cue-D82EKSYY.js +1 -0
- package/dist/assets/cue-nzsbQn5T.js +1 -0
- package/dist/assets/cypher-BM7Pwunj.js +1 -0
- package/dist/assets/cypher-COkxafJQ.js +1 -0
- package/dist/assets/d-85-TOEBH.js +1 -0
- package/dist/assets/d-TG9YRPoI.js +1 -0
- package/dist/assets/dark-plus-C3mMm8J8.js +1 -0
- package/dist/assets/dart-CF10PKvl.js +1 -0
- package/dist/assets/dart-DArdxSyj.js +1 -0
- package/dist/assets/dax-B28Cho4e.js +1 -0
- package/dist/assets/dax-CEL-wOlO.js +1 -0
- package/dist/assets/desktop-BmXAJ9_W.js +1 -0
- package/dist/assets/desktop-Z9LhWKWr.js +1 -0
- package/dist/assets/diff-BlzBhZx2.js +1 -0
- package/dist/assets/diff-D97Zzqfu.js +1 -0
- package/dist/assets/docker-BcOcwvcX.js +1 -0
- package/dist/assets/docker-DIG_IqVK.js +1 -0
- package/dist/assets/dotenv-Bf0GLMGc.js +1 -0
- package/dist/assets/dotenv-Da5cRb03.js +1 -0
- package/dist/assets/dracula-BzJJZx-M.js +1 -0
- package/dist/assets/dracula-soft-BXkSAIEj.js +1 -0
- package/dist/assets/dream-maker-BtqSS_iP.js +1 -0
- package/dist/assets/dream-maker-DBhfoDN-.js +1 -0
- package/dist/assets/edge-BkV0erSs.js +1 -0
- package/dist/assets/edge-C9mkmEUa.js +1 -0
- package/dist/assets/elixir-B1OcFvXE.js +1 -0
- package/dist/assets/elixir-CDX3lj18.js +1 -0
- package/dist/assets/elm-BX_9EnK8.js +1 -0
- package/dist/assets/elm-DbKCFpqz.js +1 -0
- package/dist/assets/emacs-lisp-C9XAeP06.js +1 -0
- package/dist/assets/emacs-lisp-fXIB3QbG.js +1 -0
- package/dist/assets/erb-BOJIQeun.js +1 -0
- package/dist/assets/erb-CjsbKF3w.js +1 -0
- package/dist/assets/erlang-Dh270pQQ.js +1 -0
- package/dist/assets/erlang-DsQrWhSR.js +1 -0
- package/dist/assets/everforest-dark-BgDCqdQA.js +1 -0
- package/dist/assets/everforest-light-C8M2exoo.js +1 -0
- package/dist/assets/fennel-BYunw83y.js +1 -0
- package/dist/assets/fennel-CKW7gc4E.js +1 -0
- package/dist/assets/finder-CGxGn0h3.png +0 -0
- package/dist/assets/fish-BiaFC76h.js +1 -0
- package/dist/assets/fish-BvzEVeQv.js +1 -0
- package/dist/assets/fluent-C1VbA1FI.js +1 -0
- package/dist/assets/fluent-C4IJs8-o.js +1 -0
- package/dist/assets/fortran-fixed-form-BZjJHVRy.js +1 -0
- package/dist/assets/fortran-fixed-form-VBUom2Po.js +1 -0
- package/dist/assets/fortran-free-form-D22FLkUw.js +1 -0
- package/dist/assets/fortran-free-form-xZZnushS.js +1 -0
- package/dist/assets/fsharp-B6CigNpP.js +1 -0
- package/dist/assets/fsharp-CXgrBDvD.js +1 -0
- package/dist/assets/gdresource-B57lXvxj.js +1 -0
- package/dist/assets/gdresource-B7Tvp0Sc.js +1 -0
- package/dist/assets/gdscript-DTMYz4Jt.js +1 -0
- package/dist/assets/gdscript-etqGfJNh.js +1 -0
- package/dist/assets/gdshader-DkwncUOv.js +1 -0
- package/dist/assets/gdshader-MbeiIFWW.js +1 -0
- package/dist/assets/genie-CTXGoN_z.js +1 -0
- package/dist/assets/genie-D0YGMca9.js +1 -0
- package/dist/assets/gherkin-DQeIZrj7.js +1 -0
- package/dist/assets/gherkin-DyxjwDmM.js +1 -0
- package/dist/assets/ghostty-web-CsYhFgbJ.js +13 -0
- package/dist/assets/git-commit-BicZTByO.js +1 -0
- package/dist/assets/git-commit-F4YmCXRG.js +1 -0
- package/dist/assets/git-rebase-Bzlpfr7M.js +1 -0
- package/dist/assets/git-rebase-r7XF79zn.js +1 -0
- package/dist/assets/github-dark-DHJKELXO.js +1 -0
- package/dist/assets/github-dark-default-Cuk6v7N8.js +1 -0
- package/dist/assets/github-dark-dimmed-DH5Ifo-i.js +1 -0
- package/dist/assets/github-dark-high-contrast-E3gJ1_iC.js +1 -0
- package/dist/assets/github-light-DAi9KRSo.js +1 -0
- package/dist/assets/github-light-default-D7oLnXFd.js +1 -0
- package/dist/assets/github-light-high-contrast-BfjtVDDH.js +1 -0
- package/dist/assets/gleam-B-IqXlZK.js +1 -0
- package/dist/assets/gleam-BspZqrRM.js +1 -0
- package/dist/assets/glimmer-js-Rg0-pVw9.js +1 -0
- package/dist/assets/glimmer-js-cZi37nA1.js +1 -0
- package/dist/assets/glimmer-ts-BY3QTAW9.js +1 -0
- package/dist/assets/glimmer-ts-U6CK756n.js +1 -0
- package/dist/assets/glsl-Bk7TyU8J.js +1 -0
- package/dist/assets/glsl-DplSGwfg.js +1 -0
- package/dist/assets/gnuplot-BBjkHhdK.js +1 -0
- package/dist/assets/gnuplot-DdkO51Og.js +1 -0
- package/dist/assets/go-BYCC9JQ2.js +1 -0
- package/dist/assets/go-Dn2_MT6a.js +1 -0
- package/dist/assets/graphql-By4-2MiP.js +1 -0
- package/dist/assets/graphql-ChdNCCLP.js +1 -0
- package/dist/assets/groovy-DjMbfbx0.js +1 -0
- package/dist/assets/groovy-gcz8RCvz.js +1 -0
- package/dist/assets/gruvbox-dark-hard-CFHQjOhq.js +1 -0
- package/dist/assets/gruvbox-dark-medium-GsRaNv29.js +1 -0
- package/dist/assets/gruvbox-dark-soft-CVdnzihN.js +1 -0
- package/dist/assets/gruvbox-light-hard-CH1njM8p.js +1 -0
- package/dist/assets/gruvbox-light-medium-DRw_LuNl.js +1 -0
- package/dist/assets/gruvbox-light-soft-hJgmCMqR.js +1 -0
- package/dist/assets/hack-CaT9iCJl.js +1 -0
- package/dist/assets/hack-Ce3HFLQu.js +1 -0
- package/dist/assets/haml-B8DHNrY2.js +1 -0
- package/dist/assets/haml-C-C08OMF.js +1 -0
- package/dist/assets/handlebars-BFxKmXBZ.js +1 -0
- package/dist/assets/handlebars-BL8al0AC.js +1 -0
- package/dist/assets/haskell-Df6bDoY_.js +1 -0
- package/dist/assets/haskell-DtebQ3Kv.js +1 -0
- package/dist/assets/haxe-CzTSHFRz.js +1 -0
- package/dist/assets/haxe-n0Q_eay_.js +1 -0
- package/dist/assets/hcl-BWvSN4gD.js +1 -0
- package/dist/assets/hcl-shhv74hy.js +1 -0
- package/dist/assets/hjson-BEMFYqr7.js +1 -0
- package/dist/assets/hjson-D5-asLiD.js +1 -0
- package/dist/assets/hlsl-D3lLCCz7.js +1 -0
- package/dist/assets/hlsl-Pefxhtaj.js +1 -0
- package/dist/assets/home-DDFZTO3p.js +1 -0
- package/dist/assets/houston-DnULxvSX.js +1 -0
- package/dist/assets/html-DBrZJmtb.js +1 -0
- package/dist/assets/html-GMplVEZG.js +1 -0
- package/dist/assets/html-derivative-BFtXZ54Q.js +1 -0
- package/dist/assets/html-derivative-DgyNW28a.js +1 -0
- package/dist/assets/http-D0p2QOuJ.js +1 -0
- package/dist/assets/http-jrhK8wxY.js +1 -0
- package/dist/assets/hurl-CeF7EoTQ.js +1 -0
- package/dist/assets/hurl-irOxFIW8.js +1 -0
- package/dist/assets/hxml-B_SuSTPP.js +1 -0
- package/dist/assets/hxml-Bvhsp5Yf.js +1 -0
- package/dist/assets/hy-CTpXmg0M.js +1 -0
- package/dist/assets/hy-DFXneXwc.js +1 -0
- package/dist/assets/imba-DGztddWO.js +1 -0
- package/dist/assets/imba-z_cUhu9e.js +1 -0
- package/dist/assets/index-D0jX0BNw.js +1844 -0
- package/dist/assets/index-DDCRe3lm.css +1 -0
- package/dist/assets/ini-BEwlwnbL.js +1 -0
- package/dist/assets/ini-DOGW4-0C.js +1 -0
- package/dist/assets/inter-FIwubZjA.woff2 +0 -0
- package/dist/assets/iosevka-nerd-font-DKH7rjGs.woff2 +0 -0
- package/dist/assets/iosevka-nerd-font-bold-BObfzjZJ.woff2 +0 -0
- package/dist/assets/java-B2AjWfgk.js +1 -0
- package/dist/assets/java-CylS5w8V.js +1 -0
- package/dist/assets/javascript-XPxc2APE.js +1 -0
- package/dist/assets/javascript-wDzz0qaB.js +1 -0
- package/dist/assets/jinja-4LBKfQ-Z.js +1 -0
- package/dist/assets/jinja-DlBE7Xgz.js +1 -0
- package/dist/assets/jison-DRGa8fbr.js +1 -0
- package/dist/assets/jison-wvAkD_A8.js +1 -0
- package/dist/assets/json-CjLXzM_z.js +1 -0
- package/dist/assets/json-Cp-IABpG.js +1 -0
- package/dist/assets/json5-C9tS-k6U.js +1 -0
- package/dist/assets/json5-DE06a0U2.js +1 -0
- package/dist/assets/jsonc-BYi4HS05.js +1 -0
- package/dist/assets/jsonc-Des-eS-w.js +1 -0
- package/dist/assets/jsonl-CuhIcKDk.js +1 -0
- package/dist/assets/jsonl-DcaNXYhu.js +1 -0
- package/dist/assets/jsonnet-DFQXde-d.js +1 -0
- package/dist/assets/jsonnet-DtQdHwK-.js +1 -0
- package/dist/assets/jssm-BlhucWo7.js +1 -0
- package/dist/assets/jssm-C2t-YnRu.js +1 -0
- package/dist/assets/jsx-BGEKFWBk.js +1 -0
- package/dist/assets/jsx-g9-lgVsj.js +1 -0
- package/dist/assets/julia-Bh5qjrIj.js +1 -0
- package/dist/assets/julia-C8NyazO9.js +1 -0
- package/dist/assets/kanagawa-dragon-CkXjmgJE.js +1 -0
- package/dist/assets/kanagawa-lotus-CfQXZHmo.js +1 -0
- package/dist/assets/kanagawa-wave-DWedfzmr.js +1 -0
- package/dist/assets/kdl-DJsF4U57.js +1 -0
- package/dist/assets/kdl-DV7GczEv.js +1 -0
- package/dist/assets/kotlin-BdnUsdx6.js +1 -0
- package/dist/assets/kotlin-DwKqsDKK.js +1 -0
- package/dist/assets/kusto-BvAqAH-y.js +1 -0
- package/dist/assets/kusto-w7PrLrBe.js +1 -0
- package/dist/assets/laserwave-DUszq2jm.js +1 -0
- package/dist/assets/latex-BdAV_C_H.js +1 -0
- package/dist/assets/latex-BuzO6rjU.js +1 -0
- package/dist/assets/lean-Bc6EcWN3.js +1 -0
- package/dist/assets/lean-Dle5D8N4.js +1 -0
- package/dist/assets/less-B1dDrJ26.js +1 -0
- package/dist/assets/less-C76OjOdF.js +1 -0
- package/dist/assets/light-plus-B7mTdjB0.js +1 -0
- package/dist/assets/liquid-DYVedYrR.js +1 -0
- package/dist/assets/liquid-DdraCjph.js +1 -0
- package/dist/assets/llvm-BtvRca6l.js +1 -0
- package/dist/assets/llvm-s1WuUfOe.js +1 -0
- package/dist/assets/log-2UxHyX5q.js +1 -0
- package/dist/assets/log-CbW7hOWU.js +1 -0
- package/dist/assets/logo-BbooMuDm.js +1 -0
- package/dist/assets/logo-BtOb2qkB.js +1 -0
- package/dist/assets/lua-BbnMAYS6.js +1 -0
- package/dist/assets/lua-tkfysgZP.js +1 -0
- package/dist/assets/luau-CBwFb-vO.js +1 -0
- package/dist/assets/luau-CXu1NL6O.js +1 -0
- package/dist/assets/make-CHLpvVh8.js +1 -0
- package/dist/assets/make-s9FKP0VE.js +1 -0
- package/dist/assets/markdown-CKXSK5AY.js +1 -0
- package/dist/assets/markdown-Cvjx9yec.js +1 -0
- package/dist/assets/marko-BsqRlKIY.js +1 -0
- package/dist/assets/marko-CPi9NSCl.js +1 -0
- package/dist/assets/material-theme-D5KoaKCx.js +1 -0
- package/dist/assets/material-theme-darker-BfHTSMKl.js +1 -0
- package/dist/assets/material-theme-lighter-B0m2ddpp.js +1 -0
- package/dist/assets/material-theme-ocean-CyktbL80.js +1 -0
- package/dist/assets/material-theme-palenight-Csfq5Kiy.js +1 -0
- package/dist/assets/matlab-BQKM-LA_.js +1 -0
- package/dist/assets/matlab-D7o27uSR.js +1 -0
- package/dist/assets/mdc-BBM-UqJy.js +1 -0
- package/dist/assets/mdc-DUICxH0z.js +1 -0
- package/dist/assets/mdx-BWRhHdFl.js +1 -0
- package/dist/assets/mdx-Cmh6b_Ma.js +1 -0
- package/dist/assets/mermaid-DKYwYmdq.js +1 -0
- package/dist/assets/mermaid-DrapLM8U.js +1 -0
- package/dist/assets/min-dark-CafNBF8u.js +1 -0
- package/dist/assets/min-light-CTRr51gU.js +1 -0
- package/dist/assets/mipsasm-BElnB_7Q.js +1 -0
- package/dist/assets/mipsasm-CKIfxQSi.js +1 -0
- package/dist/assets/mojo-1DNp92w6.js +1 -0
- package/dist/assets/mojo-Dz6zCmiK.js +1 -0
- package/dist/assets/monokai-D4h5O-jR.js +1 -0
- package/dist/assets/move-Bu9oaDYs.js +1 -0
- package/dist/assets/move-Dvu4btzz.js +1 -0
- package/dist/assets/narrat-Cy2k80UN.js +1 -0
- package/dist/assets/narrat-DRg8JJMk.js +1 -0
- package/dist/assets/nextflow-BO4ujnzu.js +1 -0
- package/dist/assets/nextflow-BrzmwbiE.js +1 -0
- package/dist/assets/nginx-DknmC5AR.js +1 -0
- package/dist/assets/nginx-XmDIhN78.js +1 -0
- package/dist/assets/night-owl-C39BiMTA.js +1 -0
- package/dist/assets/nim-CVrawwO9.js +1 -0
- package/dist/assets/nim-Cm6-Nayq.js +1 -0
- package/dist/assets/nix-DcZRBjkC.js +1 -0
- package/dist/assets/nix-c8nO5XWb.js +1 -0
- package/dist/assets/nope-02-EygnDbCM.aac +0 -0
- package/dist/assets/nope-05-DZsXzrQW.aac +0 -0
- package/dist/assets/nope-08-COPo0uNf.aac +0 -0
- package/dist/assets/nope-11-CVdXg8G-.aac +0 -0
- package/dist/assets/nope-12-BJR1Ka3c.aac +0 -0
- package/dist/assets/nord-Ddv68eIx.js +1 -0
- package/dist/assets/nushell-C-sUppwS.js +1 -0
- package/dist/assets/nushell-Dqv-0Q5X.js +1 -0
- package/dist/assets/objective-c-DXmwc3jG.js +1 -0
- package/dist/assets/objective-c-sgtl-PdD.js +1 -0
- package/dist/assets/objective-cpp-BTQ37cYs.js +1 -0
- package/dist/assets/objective-cpp-CLxacb5B.js +1 -0
- package/dist/assets/ocaml-BvBP9Lm0.js +1 -0
- package/dist/assets/ocaml-C0hk2d4L.js +1 -0
- package/dist/assets/one-dark-pro-DVMEJ2y_.js +1 -0
- package/dist/assets/one-light-PoHY5YXO.js +1 -0
- package/dist/assets/openscad-C4EeE6gA.js +1 -0
- package/dist/assets/openscad-Dmi2c_42.js +1 -0
- package/dist/assets/pascal-D93ZcfNL.js +1 -0
- package/dist/assets/pascal-DYjw0AuS.js +1 -0
- package/dist/assets/perl-C0TMdlhV.js +1 -0
- package/dist/assets/perl-Djxv1YJ9.js +1 -0
- package/dist/assets/php-CDn_0X-4.js +1 -0
- package/dist/assets/php-DEfiO3Bd.js +1 -0
- package/dist/assets/pierre-dark-BXuwtOqb.js +1 -0
- package/dist/assets/pierre-light-dckrK0oj.js +1 -0
- package/dist/assets/pkl-Cij4Vm9z.js +1 -0
- package/dist/assets/pkl-u5AG7uiY.js +1 -0
- package/dist/assets/plastic-3e1v2bzS.js +1 -0
- package/dist/assets/plsql-Bd6NdBJI.js +1 -0
- package/dist/assets/plsql-ChMvpjG-.js +1 -0
- package/dist/assets/po-BTJTHyun.js +1 -0
- package/dist/assets/po-CUhVi7la.js +1 -0
- package/dist/assets/poimandres-CS3Unz2-.js +1 -0
- package/dist/assets/polar-B-qJ3AIy.js +1 -0
- package/dist/assets/polar-C0HS_06l.js +1 -0
- package/dist/assets/postcss-BSGhOJ4k.js +1 -0
- package/dist/assets/postcss-CXtECtnM.js +1 -0
- package/dist/assets/powerquery-CC1XieQO.js +1 -0
- package/dist/assets/powerquery-CEu0bR-o.js +1 -0
- package/dist/assets/powershell-DdRAnR4Z.js +1 -0
- package/dist/assets/powershell-Dpen1YoG.js +1 -0
- package/dist/assets/prisma-C0hJlfqX.js +1 -0
- package/dist/assets/prisma-Dd19v3D-.js +1 -0
- package/dist/assets/prolog-CbFg5uaA.js +1 -0
- package/dist/assets/prolog-bSi0SE3D.js +1 -0
- package/dist/assets/proto-DIGPV07t.js +1 -0
- package/dist/assets/proto-DyJlTyXw.js +1 -0
- package/dist/assets/pug-CGlum2m_.js +1 -0
- package/dist/assets/pug-ClQUpJlV.js +1 -0
- package/dist/assets/puppet-BMWR74SV.js +1 -0
- package/dist/assets/puppet-ja2MI9-b.js +1 -0
- package/dist/assets/purescript-CklMAg4u.js +1 -0
- package/dist/assets/purescript-CxgsQa43.js +1 -0
- package/dist/assets/python-B6aJPvgy.js +1 -0
- package/dist/assets/python-DgE28U1V.js +1 -0
- package/dist/assets/qml-3beO22l8.js +1 -0
- package/dist/assets/qml-BEkX6VoX.js +1 -0
- package/dist/assets/qmldir-C8lEn-DE.js +1 -0
- package/dist/assets/qmldir-DNYHzgow.js +1 -0
- package/dist/assets/qss-BZsEMKxJ.js +1 -0
- package/dist/assets/qss-IeuSbFQv.js +1 -0
- package/dist/assets/r-BfG_C0qz.js +1 -0
- package/dist/assets/r-DiinP2Uv.js +1 -0
- package/dist/assets/racket-BqYA7rlc.js +1 -0
- package/dist/assets/racket-CzHBKLyy.js +1 -0
- package/dist/assets/raku-CPccoUN1.js +1 -0
- package/dist/assets/raku-DXvB9xmW.js +1 -0
- package/dist/assets/razor-8rj0VRTp.js +1 -0
- package/dist/assets/razor-CE9lU5zL.js +1 -0
- package/dist/assets/red-bN70gL4F.js +1 -0
- package/dist/assets/reg-C-SQnVFl.js +1 -0
- package/dist/assets/reg-DkqHJyC0.js +1 -0
- package/dist/assets/regexp-BW_zLBm-.js +1 -0
- package/dist/assets/regexp-CDVJQ6XC.js +1 -0
- package/dist/assets/rel-BIqKdwke.js +1 -0
- package/dist/assets/rel-C3B-1QV4.js +1 -0
- package/dist/assets/riscv-BM1_JUlF.js +1 -0
- package/dist/assets/riscv-s6spwTFp.js +1 -0
- package/dist/assets/rose-pine-dawn-DHQR4-dF.js +1 -0
- package/dist/assets/rose-pine-moon-D4_iv3hh.js +1 -0
- package/dist/assets/rose-pine-qdsjHGoJ.js +1 -0
- package/dist/assets/rosmsg-BJDFO7_C.js +1 -0
- package/dist/assets/rosmsg-TcL_zzr8.js +1 -0
- package/dist/assets/rst-8IYWyo0X.js +1 -0
- package/dist/assets/rst-B0xPkSld.js +1 -0
- package/dist/assets/ruby-BvKwtOVI.js +1 -0
- package/dist/assets/ruby-CHyTNJrF.js +1 -0
- package/dist/assets/rust-B1yitclQ.js +1 -0
- package/dist/assets/rust-DJErdhjk.js +1 -0
- package/dist/assets/sas-BZeeyq_z.js +1 -0
- package/dist/assets/sas-cz2c8ADy.js +1 -0
- package/dist/assets/sass-CIfQ9yqV.js +1 -0
- package/dist/assets/sass-Cj5Yp3dK.js +1 -0
- package/dist/assets/scala-C151Ov-r.js +1 -0
- package/dist/assets/scala-CWwlNtab.js +1 -0
- package/dist/assets/scheme-ByM66cy_.js +1 -0
- package/dist/assets/scheme-C98Dy4si.js +1 -0
- package/dist/assets/scss-Dk05w39G.js +1 -0
- package/dist/assets/scss-OYdSNvt2.js +1 -0
- package/dist/assets/sdbl-C5Q1usnU.js +1 -0
- package/dist/assets/sdbl-DVxCFoDh.js +1 -0
- package/dist/assets/session-fRmszlju.js +41 -0
- package/dist/assets/shaderlab-DNkMdb2B.js +1 -0
- package/dist/assets/shaderlab-Dg9Lc6iA.js +1 -0
- package/dist/assets/shellscript-CE-5zpiL.js +1 -0
- package/dist/assets/shellscript-Yzrsuije.js +1 -0
- package/dist/assets/shellsession-BADoaaVG.js +1 -0
- package/dist/assets/shellsession-BhddCRKA.js +1 -0
- package/dist/assets/slack-dark-BthQWCQV.js +1 -0
- package/dist/assets/slack-ochin-DqwNpetd.js +1 -0
- package/dist/assets/smalltalk-BERRCDM3.js +1 -0
- package/dist/assets/smalltalk-GKfhTxrk.js +1 -0
- package/dist/assets/snazzy-light-Bw305WKR.js +1 -0
- package/dist/assets/solarized-dark-DXbdFlpD.js +1 -0
- package/dist/assets/solarized-light-L9t79GZl.js +1 -0
- package/dist/assets/solidity-DanOLEvk.js +1 -0
- package/dist/assets/solidity-rGO070M0.js +1 -0
- package/dist/assets/soy-Brmx7dQM.js +1 -0
- package/dist/assets/soy-C6ERzLIv.js +1 -0
- package/dist/assets/sparql-CHK3vDkL.js +1 -0
- package/dist/assets/sparql-rVzFXLq3.js +1 -0
- package/dist/assets/splunk-BtCnVYZw.js +1 -0
- package/dist/assets/splunk-YU1HCmJX.js +1 -0
- package/dist/assets/sprite-BxALe73Z.svg +882 -0
- package/dist/assets/sprite-Fb-TFjRY.svg +11707 -0
- package/dist/assets/sql-BLtJtn59.js +1 -0
- package/dist/assets/sql-CKXjX23X.js +1 -0
- package/dist/assets/ssh-config-Cpnw3KEt.js +1 -0
- package/dist/assets/ssh-config-_ykCGR6B.js +1 -0
- package/dist/assets/staplebops-03-Aug82oH0.aac +0 -0
- package/dist/assets/staplebops-04-olyHi8qQ.aac +0 -0
- package/dist/assets/staplebops-06-Cj_2vOI4.aac +0 -0
- package/dist/assets/staplebops-07-cqQEvbIf.aac +0 -0
- package/dist/assets/stata-BH5u7GGu.js +1 -0
- package/dist/assets/stata-BJqnY-gO.js +1 -0
- package/dist/assets/stylus-BEDo0Tqx.js +1 -0
- package/dist/assets/stylus-CHekGIgu.js +1 -0
- package/dist/assets/svelte-3Dk4HxPD.js +1 -0
- package/dist/assets/svelte-DmnhoTkn.js +1 -0
- package/dist/assets/swift-Deru3G9w.js +1 -0
- package/dist/assets/swift-Dg5xB15N.js +1 -0
- package/dist/assets/synthwave-84-CbfX1IO0.js +1 -0
- package/dist/assets/system-verilog-CnnmHF94.js +1 -0
- package/dist/assets/system-verilog-rKNYENrf.js +1 -0
- package/dist/assets/systemd-4A_iFExJ.js +1 -0
- package/dist/assets/systemd-C1FfcI00.js +1 -0
- package/dist/assets/talonscript-CkByrt1z.js +1 -0
- package/dist/assets/talonscript-bksoRmW6.js +1 -0
- package/dist/assets/tasl-Bl4tsGRM.js +1 -0
- package/dist/assets/tasl-QIJgUcNo.js +1 -0
- package/dist/assets/tcl-B2_VAYrd.js +1 -0
- package/dist/assets/tcl-dwOrl1Do.js +1 -0
- package/dist/assets/templ-CyHhV9jl.js +1 -0
- package/dist/assets/templ-W15q3VgB.js +1 -0
- package/dist/assets/terminal-Cpd_WDea.png +0 -0
- package/dist/assets/terraform-BETggiCN.js +1 -0
- package/dist/assets/terraform-Ba94thAr.js +1 -0
- package/dist/assets/tex-CxkMU7Pf.js +1 -0
- package/dist/assets/tex-DZ7Smwga.js +1 -0
- package/dist/assets/textmate-B-hymv65.png +0 -0
- package/dist/assets/tokyo-night-hegEt444.js +1 -0
- package/dist/assets/toml-Btlk-g9G.js +1 -0
- package/dist/assets/toml-vGWfd6FD.js +1 -0
- package/dist/assets/ts-tags-DMPjx__B.js +1 -0
- package/dist/assets/ts-tags-zn1MmPIZ.js +1 -0
- package/dist/assets/tsv-B_m7g4N7.js +1 -0
- package/dist/assets/tsv-DYfmKMIW.js +1 -0
- package/dist/assets/tsx-COt5Ahok.js +1 -0
- package/dist/assets/tsx-Dio8iE-N.js +1 -0
- package/dist/assets/turtle-BsS91CYL.js +1 -0
- package/dist/assets/turtle-whXsfw69.js +1 -0
- package/dist/assets/twig-CO9l9SDP.js +1 -0
- package/dist/assets/twig-vsFUeHgy.js +1 -0
- package/dist/assets/typescript-BPQ3VLAy.js +1 -0
- package/dist/assets/typescript-DttGkNCf.js +1 -0
- package/dist/assets/typespec-BGHnOYBU.js +1 -0
- package/dist/assets/typespec-C_z-8Pri.js +1 -0
- package/dist/assets/typst-D2RORpaG.js +1 -0
- package/dist/assets/typst-DHCkPAjA.js +1 -0
- package/dist/assets/v-BcVCzyr7.js +1 -0
- package/dist/assets/v-CvM42WIw.js +1 -0
- package/dist/assets/vala-B_5xOBNr.js +1 -0
- package/dist/assets/vala-CsfeWuGM.js +1 -0
- package/dist/assets/vb-BVDm3zBt.js +1 -0
- package/dist/assets/vb-D17OF-Vu.js +1 -0
- package/dist/assets/verilog-BQ8w6xss.js +1 -0
- package/dist/assets/verilog-D5iownw6.js +1 -0
- package/dist/assets/vesper-DU1UobuO.js +1 -0
- package/dist/assets/vhdl-Bmgbkl_-.js +1 -0
- package/dist/assets/vhdl-CeAyd5Ju.js +1 -0
- package/dist/assets/viml-3ZODU9FQ.js +1 -0
- package/dist/assets/viml-CJc9bBzg.js +1 -0
- package/dist/assets/vitesse-black-Bkuqu6BP.js +1 -0
- package/dist/assets/vitesse-dark-D0r3Knsf.js +1 -0
- package/dist/assets/vitesse-light-CVO1_9PV.js +1 -0
- package/dist/assets/vue-BJuclMoA.js +1 -0
- package/dist/assets/vue-DnHKYNfI.js +1 -0
- package/dist/assets/vue-html-CChd_i61.js +1 -0
- package/dist/assets/vue-html-CRAum25o.js +1 -0
- package/dist/assets/vue-vine-8moa0y9V.js +1 -0
- package/dist/assets/vue-vine-CDS6dxm6.js +1 -0
- package/dist/assets/vyper-B6NKdirK.js +1 -0
- package/dist/assets/vyper-CDx5xZoG.js +1 -0
- package/dist/assets/wasm-CG6Dc4jp.js +1 -0
- package/dist/assets/wasm-LUzQYg3H.js +1 -0
- package/dist/assets/wasm-MzD3tlZU.js +1 -0
- package/dist/assets/wenyan-8fXatRyt.js +1 -0
- package/dist/assets/wenyan-BV7otONQ.js +1 -0
- package/dist/assets/wgsl-CECC3vJ5.js +1 -0
- package/dist/assets/wgsl-Dx-B1_4e.js +1 -0
- package/dist/assets/wikitext-BaRWL9sq.js +1 -0
- package/dist/assets/wikitext-BhOHFoWU.js +1 -0
- package/dist/assets/wit-5i3qLPDT.js +1 -0
- package/dist/assets/wit-MEd50GYB.js +1 -0
- package/dist/assets/wolfram-CmMaIVaG.js +1 -0
- package/dist/assets/wolfram-lXgVvXCa.js +1 -0
- package/dist/assets/worker-CKpRWWws.js +153 -0
- package/dist/assets/xcode-4zqeTePl.png +0 -0
- package/dist/assets/xml-DtyAt_xK.js +1 -0
- package/dist/assets/xml-sdJ4AIDG.js +1 -0
- package/dist/assets/xsl-Btrj9DhN.js +1 -0
- package/dist/assets/xsl-CtQFsRM5.js +1 -0
- package/dist/assets/yaml-B92AUxDB.js +1 -0
- package/dist/assets/yaml-Buea-lGh.js +1 -0
- package/dist/assets/yup-01-BtRq6dLN.aac +0 -0
- package/dist/assets/yup-03-BHLEoqSS.aac +0 -0
- package/dist/assets/yup-04-C7yadpJT.aac +0 -0
- package/dist/assets/yup-05-CuuaeyjC.aac +0 -0
- package/dist/assets/zenscript-DVFEvuxE.js +1 -0
- package/dist/assets/zenscript-Dja4fBOE.js +1 -0
- package/dist/assets/zig-LhWa2ADv.js +1 -0
- package/dist/assets/zig-VOosw3JB.js +1 -0
- package/dist/favicon-96x96-v3.png +0 -0
- package/dist/favicon-96x96.png +0 -0
- package/dist/favicon-v3.ico +0 -0
- package/dist/favicon-v3.svg +7 -0
- package/dist/favicon.ico +0 -0
- package/dist/favicon.svg +7 -0
- package/dist/index.html +24 -0
- package/dist/oc-theme-preload.js +28 -0
- package/dist/site.webmanifest +21 -0
- package/dist/social-share-zen.png +0 -0
- package/dist/social-share.png +0 -0
- package/dist/web-app-manifest-192x192.png +0 -0
- package/dist/web-app-manifest-512x512.png +0 -0
- package/e2e/AGENTS.md +176 -0
- package/e2e/actions.ts +421 -0
- package/e2e/app/home.spec.ts +21 -0
- package/e2e/app/navigation.spec.ts +10 -0
- package/e2e/app/palette.spec.ts +11 -0
- package/e2e/app/server-default.spec.ts +55 -0
- package/e2e/app/session.spec.ts +16 -0
- package/e2e/app/titlebar-history.spec.ts +42 -0
- package/e2e/files/file-open.spec.ts +18 -0
- package/e2e/files/file-tree.spec.ts +37 -0
- package/e2e/files/file-viewer.spec.ts +26 -0
- package/e2e/fixtures.ts +87 -0
- package/e2e/models/model-picker.spec.ts +42 -0
- package/e2e/models/models-visibility.spec.ts +61 -0
- package/e2e/projects/project-edit.spec.ts +53 -0
- package/e2e/projects/projects-close.spec.ts +74 -0
- package/e2e/projects/projects-switch.spec.ts +35 -0
- package/e2e/projects/workspaces.spec.ts +333 -0
- package/e2e/prompt/context.spec.ts +40 -0
- package/e2e/prompt/prompt-mention.spec.ts +26 -0
- package/e2e/prompt/prompt-slash-open.spec.ts +22 -0
- package/e2e/prompt/prompt.spec.ts +58 -0
- package/e2e/selectors.ts +57 -0
- package/e2e/session/session.spec.ts +157 -0
- package/e2e/settings/settings-keybinds.spec.ts +320 -0
- package/e2e/settings/settings-models.spec.ts +122 -0
- package/e2e/settings/settings-providers.spec.ts +136 -0
- package/e2e/settings/settings.spec.ts +292 -0
- package/e2e/sidebar/sidebar-session-links.spec.ts +31 -0
- package/e2e/sidebar/sidebar.spec.ts +14 -0
- package/e2e/status/status-popover.spec.ts +94 -0
- package/e2e/terminal/terminal-init.spec.ts +26 -0
- package/e2e/terminal/terminal.spec.ts +17 -0
- package/e2e/thinking-level.spec.ts +25 -0
- package/e2e/tsconfig.json +8 -0
- package/e2e/utils.ts +35 -0
- package/happydom.ts +75 -0
- package/index.html +23 -0
- package/package.json +72 -0
- package/playwright.config.ts +43 -0
- package/public/_headers +17 -0
- package/public/oc-theme-preload.js +28 -0
- package/script/e2e-local.ts +140 -0
- package/src/addons/serialize.test.ts +319 -0
- package/src/addons/serialize.ts +634 -0
- package/src/app.tsx +170 -0
- package/src/components/dialog-connect-provider.tsx +456 -0
- package/src/components/dialog-custom-provider.tsx +424 -0
- package/src/components/dialog-edit-project.tsx +245 -0
- package/src/components/dialog-fork.tsx +100 -0
- package/src/components/dialog-manage-models.tsx +76 -0
- package/src/components/dialog-release-notes.tsx +158 -0
- package/src/components/dialog-select-directory.tsx +326 -0
- package/src/components/dialog-select-file.tsx +404 -0
- package/src/components/dialog-select-mcp.tsx +96 -0
- package/src/components/dialog-select-model-unpaid.tsx +132 -0
- package/src/components/dialog-select-model.tsx +271 -0
- package/src/components/dialog-select-provider.tsx +87 -0
- package/src/components/dialog-select-server.tsx +536 -0
- package/src/components/dialog-settings.tsx +82 -0
- package/src/components/file-tree.test.ts +77 -0
- package/src/components/file-tree.tsx +468 -0
- package/src/components/link.tsx +17 -0
- package/src/components/model-tooltip.tsx +91 -0
- package/src/components/prompt-input/attachments.ts +145 -0
- package/src/components/prompt-input/build-request-parts.test.ts +277 -0
- package/src/components/prompt-input/build-request-parts.ts +190 -0
- package/src/components/prompt-input/context-items.tsx +82 -0
- package/src/components/prompt-input/drag-overlay.tsx +20 -0
- package/src/components/prompt-input/editor-dom.test.ts +51 -0
- package/src/components/prompt-input/editor-dom.ts +135 -0
- package/src/components/prompt-input/history.test.ts +69 -0
- package/src/components/prompt-input/history.ts +160 -0
- package/src/components/prompt-input/image-attachments.tsx +51 -0
- package/src/components/prompt-input/placeholder.test.ts +35 -0
- package/src/components/prompt-input/placeholder.ts +13 -0
- package/src/components/prompt-input/slash-popover.tsx +144 -0
- package/src/components/prompt-input/submit.ts +411 -0
- package/src/components/prompt-input.tsx +1199 -0
- package/src/components/question-dock.tsx +295 -0
- package/src/components/server/server-row.tsx +77 -0
- package/src/components/session/index.ts +5 -0
- package/src/components/session/session-context-metrics.test.ts +93 -0
- package/src/components/session/session-context-metrics.ts +94 -0
- package/src/components/session/session-context-tab.tsx +400 -0
- package/src/components/session/session-header.tsx +571 -0
- package/src/components/session/session-new-view.tsx +78 -0
- package/src/components/session/session-sortable-tab.tsx +63 -0
- package/src/components/session/session-sortable-terminal-tab.tsx +190 -0
- package/src/components/session-context-usage.tsx +100 -0
- package/src/components/settings-agents.tsx +15 -0
- package/src/components/settings-commands.tsx +15 -0
- package/src/components/settings-general.tsx +434 -0
- package/src/components/settings-keybinds.tsx +435 -0
- package/src/components/settings-mcp.tsx +15 -0
- package/src/components/settings-models.tsx +130 -0
- package/src/components/settings-permissions.tsx +228 -0
- package/src/components/settings-providers.tsx +264 -0
- package/src/components/status-popover.tsx +371 -0
- package/src/components/terminal.tsx +464 -0
- package/src/components/titlebar-history.test.ts +63 -0
- package/src/components/titlebar-history.ts +57 -0
- package/src/components/titlebar.tsx +261 -0
- package/src/context/command-keybind.test.ts +43 -0
- package/src/context/command.test.ts +25 -0
- package/src/context/command.tsx +365 -0
- package/src/context/comments.test.ts +111 -0
- package/src/context/comments.tsx +185 -0
- package/src/context/file/content-cache.ts +88 -0
- package/src/context/file/path.test.ts +352 -0
- package/src/context/file/path.ts +143 -0
- package/src/context/file/tree-store.ts +170 -0
- package/src/context/file/types.ts +41 -0
- package/src/context/file/view-cache.ts +136 -0
- package/src/context/file/watcher.test.ts +118 -0
- package/src/context/file/watcher.ts +52 -0
- package/src/context/file-content-eviction-accounting.test.ts +65 -0
- package/src/context/file.tsx +263 -0
- package/src/context/global-sdk.tsx +108 -0
- package/src/context/global-sync/bootstrap.ts +195 -0
- package/src/context/global-sync/child-store.ts +263 -0
- package/src/context/global-sync/event-reducer.test.ts +201 -0
- package/src/context/global-sync/event-reducer.ts +319 -0
- package/src/context/global-sync/eviction.ts +28 -0
- package/src/context/global-sync/queue.ts +83 -0
- package/src/context/global-sync/session-load.ts +26 -0
- package/src/context/global-sync/session-trim.test.ts +59 -0
- package/src/context/global-sync/session-trim.ts +56 -0
- package/src/context/global-sync/types.ts +134 -0
- package/src/context/global-sync/utils.ts +25 -0
- package/src/context/global-sync.test.ts +136 -0
- package/src/context/global-sync.tsx +365 -0
- package/src/context/highlights.tsx +225 -0
- package/src/context/language.tsx +226 -0
- package/src/context/layout-scroll.test.ts +36 -0
- package/src/context/layout-scroll.ts +118 -0
- package/src/context/layout.test.ts +69 -0
- package/src/context/layout.tsx +844 -0
- package/src/context/local.tsx +229 -0
- package/src/context/models.tsx +140 -0
- package/src/context/notification-index.ts +66 -0
- package/src/context/notification.test.ts +73 -0
- package/src/context/notification.tsx +199 -0
- package/src/context/permission.tsx +186 -0
- package/src/context/platform.tsx +75 -0
- package/src/context/prompt.tsx +246 -0
- package/src/context/sdk.tsx +48 -0
- package/src/context/server.tsx +208 -0
- package/src/context/settings.tsx +177 -0
- package/src/context/sync-optimistic.test.ts +56 -0
- package/src/context/sync.tsx +358 -0
- package/src/context/terminal.test.ts +38 -0
- package/src/context/terminal.tsx +283 -0
- package/src/entry.tsx +104 -0
- package/src/env.d.ts +8 -0
- package/src/hooks/use-providers.ts +31 -0
- package/src/i18n/ar.ts +716 -0
- package/src/i18n/br.ts +722 -0
- package/src/i18n/bs.ts +749 -0
- package/src/i18n/da.ts +721 -0
- package/src/i18n/de.ts +766 -0
- package/src/i18n/en.ts +792 -0
- package/src/i18n/es.ts +727 -0
- package/src/i18n/fr.ts +733 -0
- package/src/i18n/ja.ts +715 -0
- package/src/i18n/ko.ts +720 -0
- package/src/i18n/no.ts +723 -0
- package/src/i18n/parity.test.ts +31 -0
- package/src/i18n/pl.ts +723 -0
- package/src/i18n/ru.ts +727 -0
- package/src/i18n/th.ts +714 -0
- package/src/i18n/zh.ts +754 -0
- package/src/i18n/zht.ts +752 -0
- package/src/index.css +1 -0
- package/src/index.ts +3 -0
- package/src/pages/directory-layout.tsx +71 -0
- package/src/pages/error.tsx +290 -0
- package/src/pages/home.tsx +126 -0
- package/src/pages/layout/deep-links.ts +26 -0
- package/src/pages/layout/helpers.test.ts +63 -0
- package/src/pages/layout/helpers.ts +65 -0
- package/src/pages/layout/inline-editor.tsx +113 -0
- package/src/pages/layout/sidebar-items.tsx +328 -0
- package/src/pages/layout/sidebar-project-helpers.test.ts +63 -0
- package/src/pages/layout/sidebar-project-helpers.ts +11 -0
- package/src/pages/layout/sidebar-project.tsx +283 -0
- package/src/pages/layout/sidebar-shell-helpers.ts +1 -0
- package/src/pages/layout/sidebar-shell.test.ts +13 -0
- package/src/pages/layout/sidebar-shell.tsx +109 -0
- package/src/pages/layout/sidebar-workspace-helpers.ts +2 -0
- package/src/pages/layout/sidebar-workspace.test.ts +13 -0
- package/src/pages/layout/sidebar-workspace.tsx +410 -0
- package/src/pages/layout.tsx +1996 -0
- package/src/pages/session/file-tab-scroll.test.ts +40 -0
- package/src/pages/session/file-tab-scroll.ts +67 -0
- package/src/pages/session/file-tabs.tsx +516 -0
- package/src/pages/session/helpers.test.ts +61 -0
- package/src/pages/session/helpers.ts +38 -0
- package/src/pages/session/message-gesture.test.ts +62 -0
- package/src/pages/session/message-gesture.ts +21 -0
- package/src/pages/session/message-timeline.tsx +348 -0
- package/src/pages/session/review-tab.tsx +158 -0
- package/src/pages/session/scroll-spy.test.ts +127 -0
- package/src/pages/session/scroll-spy.ts +274 -0
- package/src/pages/session/session-command-helpers.ts +10 -0
- package/src/pages/session/session-mobile-tabs.tsx +36 -0
- package/src/pages/session/session-prompt-dock.test.ts +22 -0
- package/src/pages/session/session-prompt-dock.tsx +137 -0
- package/src/pages/session/session-prompt-helpers.ts +4 -0
- package/src/pages/session/session-side-panel.tsx +319 -0
- package/src/pages/session/terminal-label.ts +16 -0
- package/src/pages/session/terminal-panel.test.ts +25 -0
- package/src/pages/session/terminal-panel.tsx +169 -0
- package/src/pages/session/use-session-commands.test.ts +44 -0
- package/src/pages/session/use-session-commands.tsx +455 -0
- package/src/pages/session/use-session-hash-scroll.test.ts +16 -0
- package/src/pages/session/use-session-hash-scroll.ts +174 -0
- package/src/pages/session.tsx +1755 -0
- package/src/sst-env.d.ts +12 -0
- package/src/utils/agent.ts +11 -0
- package/src/utils/aim.ts +138 -0
- package/src/utils/base64.ts +10 -0
- package/src/utils/dom.ts +51 -0
- package/src/utils/id.ts +99 -0
- package/src/utils/index.ts +1 -0
- package/src/utils/perf.ts +135 -0
- package/src/utils/persist.ts +451 -0
- package/src/utils/prompt.ts +203 -0
- package/src/utils/runtime-adapters.test.ts +62 -0
- package/src/utils/runtime-adapters.ts +39 -0
- package/src/utils/same.ts +6 -0
- package/src/utils/scoped-cache.test.ts +69 -0
- package/src/utils/scoped-cache.ts +104 -0
- package/src/utils/server-health.test.ts +42 -0
- package/src/utils/server-health.ts +29 -0
- package/src/utils/solid-dnd.tsx +55 -0
- package/src/utils/sound.ts +117 -0
- package/src/utils/speech.ts +326 -0
- package/src/utils/time.ts +14 -0
- package/src/utils/worktree.test.ts +46 -0
- package/src/utils/worktree.ts +73 -0
- package/sst-env.d.ts +10 -0
- package/tsconfig.json +26 -0
- package/vite.config.ts +15 -0
- package/vite.js +26 -0
package/e2e/fixtures.ts
ADDED
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
import { test as base, expect, type Page } from "@playwright/test"
|
|
2
|
+
import { cleanupTestProject, createTestProject, seedProjects } from "./actions"
|
|
3
|
+
import { promptSelector } from "./selectors"
|
|
4
|
+
import { createSdk, dirSlug, getWorktree, sessionPath } from "./utils"
|
|
5
|
+
|
|
6
|
+
export const settingsKey = "settings.v3"
|
|
7
|
+
|
|
8
|
+
type TestFixtures = {
|
|
9
|
+
sdk: ReturnType<typeof createSdk>
|
|
10
|
+
gotoSession: (sessionID?: string) => Promise<void>
|
|
11
|
+
withProject: <T>(
|
|
12
|
+
callback: (project: {
|
|
13
|
+
directory: string
|
|
14
|
+
slug: string
|
|
15
|
+
gotoSession: (sessionID?: string) => Promise<void>
|
|
16
|
+
}) => Promise<T>,
|
|
17
|
+
options?: { extra?: string[] },
|
|
18
|
+
) => Promise<T>
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
type WorkerFixtures = {
|
|
22
|
+
directory: string
|
|
23
|
+
slug: string
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
export const test = base.extend<TestFixtures, WorkerFixtures>({
|
|
27
|
+
directory: [
|
|
28
|
+
async ({ }, use) => {
|
|
29
|
+
const directory = await getWorktree()
|
|
30
|
+
await use(directory)
|
|
31
|
+
},
|
|
32
|
+
{ scope: "worker" },
|
|
33
|
+
],
|
|
34
|
+
slug: [
|
|
35
|
+
async ({ directory }, use) => {
|
|
36
|
+
await use(dirSlug(directory))
|
|
37
|
+
},
|
|
38
|
+
{ scope: "worker" },
|
|
39
|
+
],
|
|
40
|
+
sdk: async ({ directory }, use) => {
|
|
41
|
+
await use(createSdk(directory))
|
|
42
|
+
},
|
|
43
|
+
gotoSession: async ({ page, directory }, use) => {
|
|
44
|
+
await seedStorage(page, { directory })
|
|
45
|
+
|
|
46
|
+
const gotoSession = async (sessionID?: string) => {
|
|
47
|
+
await page.goto(sessionPath(directory, sessionID))
|
|
48
|
+
await expect(page.locator(promptSelector)).toBeVisible()
|
|
49
|
+
}
|
|
50
|
+
await use(gotoSession)
|
|
51
|
+
},
|
|
52
|
+
withProject: async ({ page }, use) => {
|
|
53
|
+
await use(async (callback, options) => {
|
|
54
|
+
const directory = await createTestProject()
|
|
55
|
+
const slug = dirSlug(directory)
|
|
56
|
+
await seedStorage(page, { directory, extra: options?.extra })
|
|
57
|
+
|
|
58
|
+
const gotoSession = async (sessionID?: string) => {
|
|
59
|
+
await page.goto(sessionPath(directory, sessionID))
|
|
60
|
+
await expect(page.locator(promptSelector)).toBeVisible()
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
try {
|
|
64
|
+
await gotoSession()
|
|
65
|
+
return await callback({ directory, slug, gotoSession })
|
|
66
|
+
} finally {
|
|
67
|
+
await cleanupTestProject(directory)
|
|
68
|
+
}
|
|
69
|
+
})
|
|
70
|
+
},
|
|
71
|
+
})
|
|
72
|
+
|
|
73
|
+
async function seedStorage(page: Page, input: { directory: string; extra?: string[] }) {
|
|
74
|
+
await seedProjects(page, input)
|
|
75
|
+
await page.addInitScript(() => {
|
|
76
|
+
localStorage.setItem(
|
|
77
|
+
"nikcli.global.dat:model",
|
|
78
|
+
JSON.stringify({
|
|
79
|
+
recent: [{ providerID: "nikcli", modelID: "big-pickle" }],
|
|
80
|
+
user: [],
|
|
81
|
+
variant: {},
|
|
82
|
+
}),
|
|
83
|
+
)
|
|
84
|
+
})
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
export { expect }
|
|
@@ -0,0 +1,42 @@
|
|
|
1
|
+
import { test, expect } from "../fixtures"
|
|
2
|
+
import { promptSelector } from "../selectors"
|
|
3
|
+
import { clickListItem } from "../actions"
|
|
4
|
+
|
|
5
|
+
test("smoke model selection updates prompt footer", async ({ page, gotoSession }) => {
|
|
6
|
+
await gotoSession()
|
|
7
|
+
|
|
8
|
+
await page.locator(promptSelector).click()
|
|
9
|
+
await page.keyboard.type("/model")
|
|
10
|
+
|
|
11
|
+
const command = page.locator('[data-slash-id="model.choose"]')
|
|
12
|
+
await expect(command).toBeVisible()
|
|
13
|
+
await command.hover()
|
|
14
|
+
|
|
15
|
+
await page.keyboard.press("Enter")
|
|
16
|
+
|
|
17
|
+
const dialog = page.getByRole("dialog")
|
|
18
|
+
await expect(dialog).toBeVisible()
|
|
19
|
+
|
|
20
|
+
const input = dialog.getByRole("textbox").first()
|
|
21
|
+
|
|
22
|
+
const selected = dialog.locator('[data-slot="list-item"][data-selected="true"]').first()
|
|
23
|
+
await expect(selected).toBeVisible()
|
|
24
|
+
|
|
25
|
+
const other = dialog.locator('[data-slot="list-item"]:not([data-selected="true"])').first()
|
|
26
|
+
const target = (await other.count()) > 0 ? other : selected
|
|
27
|
+
|
|
28
|
+
const key = await target.getAttribute("data-key")
|
|
29
|
+
if (!key) throw new Error("Failed to resolve model key from list item")
|
|
30
|
+
|
|
31
|
+
const name = (await target.locator("span").first().innerText()).trim()
|
|
32
|
+
const model = key.split(":").slice(1).join(":")
|
|
33
|
+
|
|
34
|
+
await input.fill(model)
|
|
35
|
+
|
|
36
|
+
await clickListItem(dialog, { key })
|
|
37
|
+
|
|
38
|
+
await expect(dialog).toHaveCount(0)
|
|
39
|
+
|
|
40
|
+
const form = page.locator(promptSelector).locator("xpath=ancestor::form[1]")
|
|
41
|
+
await expect(form.locator('[data-component="button"]').filter({ hasText: name }).first()).toBeVisible()
|
|
42
|
+
})
|
|
@@ -0,0 +1,61 @@
|
|
|
1
|
+
import { test, expect } from "../fixtures"
|
|
2
|
+
import { promptSelector } from "../selectors"
|
|
3
|
+
import { closeDialog, openSettings, clickListItem } from "../actions"
|
|
4
|
+
|
|
5
|
+
test("hiding a model removes it from the model picker", async ({ page, gotoSession }) => {
|
|
6
|
+
await gotoSession()
|
|
7
|
+
|
|
8
|
+
await page.locator(promptSelector).click()
|
|
9
|
+
await page.keyboard.type("/model")
|
|
10
|
+
|
|
11
|
+
const command = page.locator('[data-slash-id="model.choose"]')
|
|
12
|
+
await expect(command).toBeVisible()
|
|
13
|
+
await command.hover()
|
|
14
|
+
await page.keyboard.press("Enter")
|
|
15
|
+
|
|
16
|
+
const picker = page.getByRole("dialog")
|
|
17
|
+
await expect(picker).toBeVisible()
|
|
18
|
+
|
|
19
|
+
const target = picker.locator('[data-slot="list-item"]').first()
|
|
20
|
+
await expect(target).toBeVisible()
|
|
21
|
+
|
|
22
|
+
const key = await target.getAttribute("data-key")
|
|
23
|
+
if (!key) throw new Error("Failed to resolve model key from list item")
|
|
24
|
+
|
|
25
|
+
const name = (await target.locator("span").first().innerText()).trim()
|
|
26
|
+
if (!name) throw new Error("Failed to resolve model name from list item")
|
|
27
|
+
|
|
28
|
+
await page.keyboard.press("Escape")
|
|
29
|
+
await expect(picker).toHaveCount(0)
|
|
30
|
+
|
|
31
|
+
const settings = await openSettings(page)
|
|
32
|
+
|
|
33
|
+
await settings.getByRole("tab", { name: "Models" }).click()
|
|
34
|
+
const search = settings.getByPlaceholder("Search models")
|
|
35
|
+
await expect(search).toBeVisible()
|
|
36
|
+
await search.fill(name)
|
|
37
|
+
|
|
38
|
+
const toggle = settings.locator('[data-component="switch"]').filter({ hasText: name }).first()
|
|
39
|
+
const input = toggle.locator('[data-slot="switch-input"]')
|
|
40
|
+
await expect(toggle).toBeVisible()
|
|
41
|
+
await expect(input).toHaveAttribute("aria-checked", "true")
|
|
42
|
+
await toggle.locator('[data-slot="switch-control"]').click()
|
|
43
|
+
await expect(input).toHaveAttribute("aria-checked", "false")
|
|
44
|
+
|
|
45
|
+
await closeDialog(page, settings)
|
|
46
|
+
|
|
47
|
+
await page.locator(promptSelector).click()
|
|
48
|
+
await page.keyboard.type("/model")
|
|
49
|
+
await expect(command).toBeVisible()
|
|
50
|
+
await command.hover()
|
|
51
|
+
await page.keyboard.press("Enter")
|
|
52
|
+
|
|
53
|
+
const pickerAgain = page.getByRole("dialog")
|
|
54
|
+
await expect(pickerAgain).toBeVisible()
|
|
55
|
+
await expect(pickerAgain.locator('[data-slot="list-item"]').first()).toBeVisible()
|
|
56
|
+
|
|
57
|
+
await expect(pickerAgain.locator(`[data-slot="list-item"][data-key="${key}"]`)).toHaveCount(0)
|
|
58
|
+
|
|
59
|
+
await page.keyboard.press("Escape")
|
|
60
|
+
await expect(pickerAgain).toHaveCount(0)
|
|
61
|
+
})
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { test, expect } from "../fixtures"
|
|
2
|
+
import { openSidebar } from "../actions"
|
|
3
|
+
|
|
4
|
+
test("dialog edit project updates name and startup script", async ({ page, withProject }) => {
|
|
5
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
6
|
+
|
|
7
|
+
await withProject(async () => {
|
|
8
|
+
await openSidebar(page)
|
|
9
|
+
|
|
10
|
+
const open = async () => {
|
|
11
|
+
const header = page.locator(".group\\/project").first()
|
|
12
|
+
await header.hover()
|
|
13
|
+
const trigger = header.getByRole("button", { name: "More options" }).first()
|
|
14
|
+
await expect(trigger).toBeVisible()
|
|
15
|
+
await trigger.click({ force: true })
|
|
16
|
+
|
|
17
|
+
const menu = page.locator('[data-component="dropdown-menu-content"]').first()
|
|
18
|
+
await expect(menu).toBeVisible()
|
|
19
|
+
|
|
20
|
+
const editItem = menu.getByRole("menuitem", { name: "Edit" }).first()
|
|
21
|
+
await expect(editItem).toBeVisible()
|
|
22
|
+
await editItem.click({ force: true })
|
|
23
|
+
|
|
24
|
+
const dialog = page.getByRole("dialog")
|
|
25
|
+
await expect(dialog).toBeVisible()
|
|
26
|
+
await expect(dialog.getByRole("heading", { level: 2 })).toHaveText("Edit project")
|
|
27
|
+
return dialog
|
|
28
|
+
}
|
|
29
|
+
|
|
30
|
+
const name = `e2e project ${Date.now()}`
|
|
31
|
+
const startup = `echo e2e_${Date.now()}`
|
|
32
|
+
|
|
33
|
+
const dialog = await open()
|
|
34
|
+
|
|
35
|
+
const nameInput = dialog.getByLabel("Name")
|
|
36
|
+
await nameInput.fill(name)
|
|
37
|
+
|
|
38
|
+
const startupInput = dialog.getByLabel("Workspace startup script")
|
|
39
|
+
await startupInput.fill(startup)
|
|
40
|
+
|
|
41
|
+
await dialog.getByRole("button", { name: "Save" }).click()
|
|
42
|
+
await expect(dialog).toHaveCount(0)
|
|
43
|
+
|
|
44
|
+
const header = page.locator(".group\\/project").first()
|
|
45
|
+
await expect(header).toContainText(name)
|
|
46
|
+
|
|
47
|
+
const reopened = await open()
|
|
48
|
+
await expect(reopened.getByLabel("Name")).toHaveValue(name)
|
|
49
|
+
await expect(reopened.getByLabel("Workspace startup script")).toHaveValue(startup)
|
|
50
|
+
await reopened.getByRole("button", { name: "Cancel" }).click()
|
|
51
|
+
await expect(reopened).toHaveCount(0)
|
|
52
|
+
})
|
|
53
|
+
})
|
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { test, expect } from "../fixtures"
|
|
2
|
+
import { createTestProject, cleanupTestProject, openSidebar, clickMenuItem } from "../actions"
|
|
3
|
+
import { projectCloseHoverSelector, projectCloseMenuSelector, projectSwitchSelector } from "../selectors"
|
|
4
|
+
import { dirSlug } from "../utils"
|
|
5
|
+
|
|
6
|
+
test("can close a project via hover card close button", async ({ page, withProject }) => {
|
|
7
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
8
|
+
|
|
9
|
+
const other = await createTestProject()
|
|
10
|
+
const otherSlug = dirSlug(other)
|
|
11
|
+
|
|
12
|
+
try {
|
|
13
|
+
await withProject(
|
|
14
|
+
async () => {
|
|
15
|
+
await openSidebar(page)
|
|
16
|
+
|
|
17
|
+
const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
|
|
18
|
+
await expect(otherButton).toBeVisible()
|
|
19
|
+
await otherButton.hover()
|
|
20
|
+
|
|
21
|
+
const close = page.locator(projectCloseHoverSelector(otherSlug)).first()
|
|
22
|
+
await expect(close).toBeVisible()
|
|
23
|
+
await close.click()
|
|
24
|
+
|
|
25
|
+
await expect(otherButton).toHaveCount(0)
|
|
26
|
+
},
|
|
27
|
+
{ extra: [other] },
|
|
28
|
+
)
|
|
29
|
+
} finally {
|
|
30
|
+
await cleanupTestProject(other)
|
|
31
|
+
}
|
|
32
|
+
})
|
|
33
|
+
|
|
34
|
+
test("can close a project via project header more options menu", async ({ page, withProject }) => {
|
|
35
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
36
|
+
|
|
37
|
+
const other = await createTestProject()
|
|
38
|
+
const otherName = other.split("/").pop() ?? other
|
|
39
|
+
const otherSlug = dirSlug(other)
|
|
40
|
+
|
|
41
|
+
try {
|
|
42
|
+
await withProject(
|
|
43
|
+
async () => {
|
|
44
|
+
await openSidebar(page)
|
|
45
|
+
|
|
46
|
+
const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
|
|
47
|
+
await expect(otherButton).toBeVisible()
|
|
48
|
+
await otherButton.click()
|
|
49
|
+
|
|
50
|
+
await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
|
|
51
|
+
|
|
52
|
+
const header = page
|
|
53
|
+
.locator(".group\\/project")
|
|
54
|
+
.filter({ has: page.locator(`[data-action="project-menu"][data-project="${otherSlug}"]`) })
|
|
55
|
+
.first()
|
|
56
|
+
await expect(header).toContainText(otherName)
|
|
57
|
+
|
|
58
|
+
const trigger = header.locator(`[data-action="project-menu"][data-project="${otherSlug}"]`).first()
|
|
59
|
+
await expect(trigger).toHaveCount(1)
|
|
60
|
+
await trigger.focus()
|
|
61
|
+
await page.keyboard.press("Enter")
|
|
62
|
+
|
|
63
|
+
const menu = page.locator('[data-component="dropdown-menu-content"]').first()
|
|
64
|
+
await expect(menu).toBeVisible({ timeout: 10_000 })
|
|
65
|
+
|
|
66
|
+
await clickMenuItem(menu, /^Close$/i, { force: true })
|
|
67
|
+
await expect(otherButton).toHaveCount(0)
|
|
68
|
+
},
|
|
69
|
+
{ extra: [other] },
|
|
70
|
+
)
|
|
71
|
+
} finally {
|
|
72
|
+
await cleanupTestProject(other)
|
|
73
|
+
}
|
|
74
|
+
})
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { test, expect } from "../fixtures"
|
|
2
|
+
import { defocus, createTestProject, cleanupTestProject } from "../actions"
|
|
3
|
+
import { projectSwitchSelector } from "../selectors"
|
|
4
|
+
import { dirSlug } from "../utils"
|
|
5
|
+
|
|
6
|
+
test("can switch between projects from sidebar", async ({ page, withProject }) => {
|
|
7
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
8
|
+
|
|
9
|
+
const other = await createTestProject()
|
|
10
|
+
const otherSlug = dirSlug(other)
|
|
11
|
+
|
|
12
|
+
try {
|
|
13
|
+
await withProject(
|
|
14
|
+
async ({ directory }) => {
|
|
15
|
+
await defocus(page)
|
|
16
|
+
|
|
17
|
+
const currentSlug = dirSlug(directory)
|
|
18
|
+
const otherButton = page.locator(projectSwitchSelector(otherSlug)).first()
|
|
19
|
+
await expect(otherButton).toBeVisible()
|
|
20
|
+
await otherButton.click()
|
|
21
|
+
|
|
22
|
+
await expect(page).toHaveURL(new RegExp(`/${otherSlug}/session`))
|
|
23
|
+
|
|
24
|
+
const currentButton = page.locator(projectSwitchSelector(currentSlug)).first()
|
|
25
|
+
await expect(currentButton).toBeVisible()
|
|
26
|
+
await currentButton.click()
|
|
27
|
+
|
|
28
|
+
await expect(page).toHaveURL(new RegExp(`/${currentSlug}/session`))
|
|
29
|
+
},
|
|
30
|
+
{ extra: [other] },
|
|
31
|
+
)
|
|
32
|
+
} finally {
|
|
33
|
+
await cleanupTestProject(other)
|
|
34
|
+
}
|
|
35
|
+
})
|
|
@@ -0,0 +1,333 @@
|
|
|
1
|
+
import { base64Decode } from "@nikcli-ai/util/encode"
|
|
2
|
+
import fs from "node:fs/promises"
|
|
3
|
+
import path from "node:path"
|
|
4
|
+
import type { Page } from "@playwright/test"
|
|
5
|
+
|
|
6
|
+
import { test, expect } from "../fixtures"
|
|
7
|
+
|
|
8
|
+
test.describe.configure({ mode: "serial" })
|
|
9
|
+
import {
|
|
10
|
+
cleanupTestProject,
|
|
11
|
+
clickMenuItem,
|
|
12
|
+
confirmDialog,
|
|
13
|
+
openSidebar,
|
|
14
|
+
openWorkspaceMenu,
|
|
15
|
+
setWorkspacesEnabled,
|
|
16
|
+
} from "../actions"
|
|
17
|
+
import { inlineInputSelector, workspaceItemSelector } from "../selectors"
|
|
18
|
+
|
|
19
|
+
function slugFromUrl(url: string) {
|
|
20
|
+
return /\/([^/]+)\/session(?:\/|$)/.exec(url)?.[1] ?? ""
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
async function setupWorkspaceTest(page: Page, project: { slug: string }) {
|
|
24
|
+
const rootSlug = project.slug
|
|
25
|
+
await openSidebar(page)
|
|
26
|
+
|
|
27
|
+
await setWorkspacesEnabled(page, rootSlug, true)
|
|
28
|
+
|
|
29
|
+
await page.getByRole("button", { name: "New workspace" }).first().click()
|
|
30
|
+
await expect
|
|
31
|
+
.poll(
|
|
32
|
+
() => {
|
|
33
|
+
const slug = slugFromUrl(page.url())
|
|
34
|
+
return slug.length > 0 && slug !== rootSlug
|
|
35
|
+
},
|
|
36
|
+
{ timeout: 45_000 },
|
|
37
|
+
)
|
|
38
|
+
.toBe(true)
|
|
39
|
+
|
|
40
|
+
const slug = slugFromUrl(page.url())
|
|
41
|
+
const dir = base64Decode(slug)
|
|
42
|
+
|
|
43
|
+
await openSidebar(page)
|
|
44
|
+
|
|
45
|
+
await expect
|
|
46
|
+
.poll(
|
|
47
|
+
async () => {
|
|
48
|
+
const item = page.locator(workspaceItemSelector(slug)).first()
|
|
49
|
+
try {
|
|
50
|
+
await item.hover({ timeout: 500 })
|
|
51
|
+
return true
|
|
52
|
+
} catch {
|
|
53
|
+
return false
|
|
54
|
+
}
|
|
55
|
+
},
|
|
56
|
+
{ timeout: 60_000 },
|
|
57
|
+
)
|
|
58
|
+
.toBe(true)
|
|
59
|
+
|
|
60
|
+
return { rootSlug, slug, directory: dir }
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
test("can enable and disable workspaces from project menu", async ({ page, withProject }) => {
|
|
64
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
65
|
+
|
|
66
|
+
await withProject(async ({ slug }) => {
|
|
67
|
+
await openSidebar(page)
|
|
68
|
+
|
|
69
|
+
await expect(page.getByRole("button", { name: "New session" }).first()).toBeVisible()
|
|
70
|
+
await expect(page.getByRole("button", { name: "New workspace" })).toHaveCount(0)
|
|
71
|
+
|
|
72
|
+
await setWorkspacesEnabled(page, slug, true)
|
|
73
|
+
await expect(page.getByRole("button", { name: "New workspace" }).first()).toBeVisible()
|
|
74
|
+
await expect(page.locator(workspaceItemSelector(slug)).first()).toBeVisible()
|
|
75
|
+
|
|
76
|
+
await setWorkspacesEnabled(page, slug, false)
|
|
77
|
+
await expect(page.getByRole("button", { name: "New session" }).first()).toBeVisible()
|
|
78
|
+
await expect(page.locator(workspaceItemSelector(slug))).toHaveCount(0)
|
|
79
|
+
})
|
|
80
|
+
})
|
|
81
|
+
|
|
82
|
+
test("can create a workspace", async ({ page, withProject }) => {
|
|
83
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
84
|
+
|
|
85
|
+
await withProject(async ({ slug }) => {
|
|
86
|
+
await openSidebar(page)
|
|
87
|
+
await setWorkspacesEnabled(page, slug, true)
|
|
88
|
+
|
|
89
|
+
await expect(page.getByRole("button", { name: "New workspace" }).first()).toBeVisible()
|
|
90
|
+
|
|
91
|
+
await page.getByRole("button", { name: "New workspace" }).first().click()
|
|
92
|
+
|
|
93
|
+
await expect
|
|
94
|
+
.poll(
|
|
95
|
+
() => {
|
|
96
|
+
const currentSlug = slugFromUrl(page.url())
|
|
97
|
+
return currentSlug.length > 0 && currentSlug !== slug
|
|
98
|
+
},
|
|
99
|
+
{ timeout: 45_000 },
|
|
100
|
+
)
|
|
101
|
+
.toBe(true)
|
|
102
|
+
|
|
103
|
+
const workspaceSlug = slugFromUrl(page.url())
|
|
104
|
+
const workspaceDir = base64Decode(workspaceSlug)
|
|
105
|
+
|
|
106
|
+
await openSidebar(page)
|
|
107
|
+
|
|
108
|
+
await expect
|
|
109
|
+
.poll(
|
|
110
|
+
async () => {
|
|
111
|
+
const item = page.locator(workspaceItemSelector(workspaceSlug)).first()
|
|
112
|
+
try {
|
|
113
|
+
await item.hover({ timeout: 500 })
|
|
114
|
+
return true
|
|
115
|
+
} catch {
|
|
116
|
+
return false
|
|
117
|
+
}
|
|
118
|
+
},
|
|
119
|
+
{ timeout: 60_000 },
|
|
120
|
+
)
|
|
121
|
+
.toBe(true)
|
|
122
|
+
|
|
123
|
+
await expect(page.locator(workspaceItemSelector(workspaceSlug)).first()).toBeVisible()
|
|
124
|
+
|
|
125
|
+
await cleanupTestProject(workspaceDir)
|
|
126
|
+
})
|
|
127
|
+
})
|
|
128
|
+
|
|
129
|
+
test("can rename a workspace", async ({ page, withProject }) => {
|
|
130
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
131
|
+
|
|
132
|
+
await withProject(async (project) => {
|
|
133
|
+
const { slug } = await setupWorkspaceTest(page, project)
|
|
134
|
+
|
|
135
|
+
const rename = `e2e workspace ${Date.now()}`
|
|
136
|
+
const menu = await openWorkspaceMenu(page, slug)
|
|
137
|
+
await clickMenuItem(menu, /^Rename$/i, { force: true })
|
|
138
|
+
|
|
139
|
+
await expect(menu).toHaveCount(0)
|
|
140
|
+
|
|
141
|
+
const item = page.locator(workspaceItemSelector(slug)).first()
|
|
142
|
+
await expect(item).toBeVisible()
|
|
143
|
+
const input = item.locator(inlineInputSelector).first()
|
|
144
|
+
await expect(input).toBeVisible()
|
|
145
|
+
await input.fill(rename)
|
|
146
|
+
await input.press("Enter")
|
|
147
|
+
await expect(item).toContainText(rename)
|
|
148
|
+
})
|
|
149
|
+
})
|
|
150
|
+
|
|
151
|
+
test("can reset a workspace", async ({ page, sdk, withProject }) => {
|
|
152
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
153
|
+
|
|
154
|
+
await withProject(async (project) => {
|
|
155
|
+
const { slug, directory: createdDir } = await setupWorkspaceTest(page, project)
|
|
156
|
+
|
|
157
|
+
const readme = path.join(createdDir, "README.md")
|
|
158
|
+
const extra = path.join(createdDir, `e2e_reset_${Date.now()}.txt`)
|
|
159
|
+
const original = await fs.readFile(readme, "utf8")
|
|
160
|
+
const dirty = `${original.trimEnd()}\n\nchange_${Date.now()}\n`
|
|
161
|
+
await fs.writeFile(readme, dirty, "utf8")
|
|
162
|
+
await fs.writeFile(extra, `created_${Date.now()}\n`, "utf8")
|
|
163
|
+
|
|
164
|
+
await expect
|
|
165
|
+
.poll(async () => {
|
|
166
|
+
return await fs
|
|
167
|
+
.stat(extra)
|
|
168
|
+
.then(() => true)
|
|
169
|
+
.catch(() => false)
|
|
170
|
+
})
|
|
171
|
+
.toBe(true)
|
|
172
|
+
|
|
173
|
+
await expect
|
|
174
|
+
.poll(async () => {
|
|
175
|
+
const files = await sdk.file
|
|
176
|
+
.status({ directory: createdDir })
|
|
177
|
+
.then((r) => r.data ?? [])
|
|
178
|
+
.catch(() => [])
|
|
179
|
+
return files.length
|
|
180
|
+
})
|
|
181
|
+
.toBeGreaterThan(0)
|
|
182
|
+
|
|
183
|
+
const menu = await openWorkspaceMenu(page, slug)
|
|
184
|
+
await clickMenuItem(menu, /^Reset$/i, { force: true })
|
|
185
|
+
await confirmDialog(page, /^Reset workspace$/i)
|
|
186
|
+
|
|
187
|
+
await expect
|
|
188
|
+
.poll(
|
|
189
|
+
async () => {
|
|
190
|
+
const files = await sdk.file
|
|
191
|
+
.status({ directory: createdDir })
|
|
192
|
+
.then((r) => r.data ?? [])
|
|
193
|
+
.catch(() => [])
|
|
194
|
+
return files.length
|
|
195
|
+
},
|
|
196
|
+
{ timeout: 60_000 },
|
|
197
|
+
)
|
|
198
|
+
.toBe(0)
|
|
199
|
+
|
|
200
|
+
await expect.poll(() => fs.readFile(readme, "utf8"), { timeout: 60_000 }).toBe(original)
|
|
201
|
+
|
|
202
|
+
await expect
|
|
203
|
+
.poll(async () => {
|
|
204
|
+
return await fs
|
|
205
|
+
.stat(extra)
|
|
206
|
+
.then(() => true)
|
|
207
|
+
.catch(() => false)
|
|
208
|
+
})
|
|
209
|
+
.toBe(false)
|
|
210
|
+
})
|
|
211
|
+
})
|
|
212
|
+
|
|
213
|
+
test("can delete a workspace", async ({ page, withProject }) => {
|
|
214
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
215
|
+
|
|
216
|
+
await withProject(async (project) => {
|
|
217
|
+
const { rootSlug, slug } = await setupWorkspaceTest(page, project)
|
|
218
|
+
|
|
219
|
+
const menu = await openWorkspaceMenu(page, slug)
|
|
220
|
+
await clickMenuItem(menu, /^Delete$/i, { force: true })
|
|
221
|
+
await confirmDialog(page, /^Delete workspace$/i)
|
|
222
|
+
|
|
223
|
+
await expect(page).toHaveURL(new RegExp(`/${rootSlug}/session`))
|
|
224
|
+
await expect(page.locator(workspaceItemSelector(slug))).toHaveCount(0)
|
|
225
|
+
await expect(page.locator(workspaceItemSelector(rootSlug)).first()).toBeVisible()
|
|
226
|
+
})
|
|
227
|
+
})
|
|
228
|
+
|
|
229
|
+
test("can reorder workspaces by drag and drop", async ({ page, withProject }) => {
|
|
230
|
+
await page.setViewportSize({ width: 1400, height: 800 })
|
|
231
|
+
await withProject(async ({ slug: rootSlug }) => {
|
|
232
|
+
const workspaces = [] as { directory: string; slug: string }[]
|
|
233
|
+
|
|
234
|
+
const listSlugs = async () => {
|
|
235
|
+
const nodes = page.locator('[data-component="sidebar-nav-desktop"] [data-component="workspace-item"]')
|
|
236
|
+
const slugs = await nodes.evaluateAll((els) => {
|
|
237
|
+
return els.map((el) => el.getAttribute("data-workspace") ?? "").filter((x) => x.length > 0)
|
|
238
|
+
})
|
|
239
|
+
return slugs
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
const waitReady = async (slug: string) => {
|
|
243
|
+
await expect
|
|
244
|
+
.poll(
|
|
245
|
+
async () => {
|
|
246
|
+
const item = page.locator(workspaceItemSelector(slug)).first()
|
|
247
|
+
try {
|
|
248
|
+
await item.hover({ timeout: 500 })
|
|
249
|
+
return true
|
|
250
|
+
} catch {
|
|
251
|
+
return false
|
|
252
|
+
}
|
|
253
|
+
},
|
|
254
|
+
{ timeout: 60_000 },
|
|
255
|
+
)
|
|
256
|
+
.toBe(true)
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
const drag = async (from: string, to: string) => {
|
|
260
|
+
const src = page.locator(workspaceItemSelector(from)).first()
|
|
261
|
+
const dst = page.locator(workspaceItemSelector(to)).first()
|
|
262
|
+
|
|
263
|
+
await src.scrollIntoViewIfNeeded()
|
|
264
|
+
await dst.scrollIntoViewIfNeeded()
|
|
265
|
+
|
|
266
|
+
const a = await src.boundingBox()
|
|
267
|
+
const b = await dst.boundingBox()
|
|
268
|
+
if (!a || !b) throw new Error("Failed to resolve workspace drag bounds")
|
|
269
|
+
|
|
270
|
+
await page.mouse.move(a.x + a.width / 2, a.y + a.height / 2)
|
|
271
|
+
await page.mouse.down()
|
|
272
|
+
await page.mouse.move(b.x + b.width / 2, b.y + b.height / 2, { steps: 12 })
|
|
273
|
+
await page.mouse.up()
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
try {
|
|
277
|
+
await openSidebar(page)
|
|
278
|
+
|
|
279
|
+
await setWorkspacesEnabled(page, rootSlug, true)
|
|
280
|
+
|
|
281
|
+
for (const _ of [0, 1]) {
|
|
282
|
+
const prev = slugFromUrl(page.url())
|
|
283
|
+
await page.getByRole("button", { name: "New workspace" }).first().click()
|
|
284
|
+
await expect
|
|
285
|
+
.poll(
|
|
286
|
+
() => {
|
|
287
|
+
const slug = slugFromUrl(page.url())
|
|
288
|
+
return slug.length > 0 && slug !== rootSlug && slug !== prev
|
|
289
|
+
},
|
|
290
|
+
{ timeout: 45_000 },
|
|
291
|
+
)
|
|
292
|
+
.toBe(true)
|
|
293
|
+
|
|
294
|
+
const slug = slugFromUrl(page.url())
|
|
295
|
+
const dir = base64Decode(slug)
|
|
296
|
+
workspaces.push({ slug, directory: dir })
|
|
297
|
+
|
|
298
|
+
await openSidebar(page)
|
|
299
|
+
}
|
|
300
|
+
|
|
301
|
+
if (workspaces.length !== 2) throw new Error("Expected two created workspaces")
|
|
302
|
+
|
|
303
|
+
const a = workspaces[0].slug
|
|
304
|
+
const b = workspaces[1].slug
|
|
305
|
+
|
|
306
|
+
await waitReady(a)
|
|
307
|
+
await waitReady(b)
|
|
308
|
+
|
|
309
|
+
const list = async () => {
|
|
310
|
+
const slugs = await listSlugs()
|
|
311
|
+
return slugs.filter((s) => s !== rootSlug && (s === a || s === b)).slice(0, 2)
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
await expect
|
|
315
|
+
.poll(async () => {
|
|
316
|
+
const slugs = await list()
|
|
317
|
+
return slugs.length === 2
|
|
318
|
+
})
|
|
319
|
+
.toBe(true)
|
|
320
|
+
|
|
321
|
+
const before = await list()
|
|
322
|
+
const from = before[1]
|
|
323
|
+
const to = before[0]
|
|
324
|
+
if (!from || !to) throw new Error("Failed to resolve initial workspace order")
|
|
325
|
+
|
|
326
|
+
await drag(from, to)
|
|
327
|
+
|
|
328
|
+
await expect.poll(async () => await list()).toEqual([from, to])
|
|
329
|
+
} finally {
|
|
330
|
+
await Promise.all(workspaces.map((w) => cleanupTestProject(w.directory)))
|
|
331
|
+
}
|
|
332
|
+
})
|
|
333
|
+
})
|